data:image/s3,"s3://crabby-images/11005/11005d402f9e41942bf67cbe59349737841af26b" alt=""
data:image/s3,"s3://crabby-images/e054e/e054eff009c8f99982e04e83271ae0ee9bedb422" alt=""
data:image/s3,"s3://crabby-images/b0612/b0612c797e19209a3d0535167da4325f19dbd21a" alt=""
/me
William Correa
@wilcorrea
data:image/s3,"s3://crabby-images/e70ca/e70ca8b01349824e88e519ef71f086076b4bb58a" alt=""
data:image/s3,"s3://crabby-images/3e9fd/3e9fd2a3de1b339d86d7da22ccc1b41b4905c9ed" alt=""
data:image/s3,"s3://crabby-images/bf666/bf66694837336b957d66c5ead1dd0bc197fdd340" alt=""
data:image/s3,"s3://crabby-images/11fab/11fabdc5ba9d8c59ea97761d4bf01d652247579b" alt=""
-> Agenda
- Gestão de estados
- Aplicando extends e mixin
- Segredos do watch
- Templates limpos
data:image/s3,"s3://crabby-images/c41a4/c41a4b9917d2d59ddac0ee6b208172d92780e763" alt=""
Aviso:
Este material contém opiniões pessoais
data:image/s3,"s3://crabby-images/86fc7/86fc7349f9b99c0dd1ebb57d975541cf9e9ec8b5" alt=""
->Gestão de estados
- $emit & props
- data & computed
- $options
- Vuex
- Event Bus
-> $emit & props
- $attrs vs $props
- sync & update
- $emit & on
data:image/s3,"s3://crabby-images/7f878/7f8786fcc5d794c84791c38691c1ef8d58a37cbc" alt=""
-> $emit & props
data:image/s3,"s3://crabby-images/39930/39930867bd7fa9b2c77efb90c9e46c052b30a946" alt=""
- $attrs vs $props
-> $emit & props
data:image/s3,"s3://crabby-images/986d4/986d4ea1613c60f3e3fe38439f0ff734bfed0df4" alt=""
- sync & update
-> $emit & props
- $emit & on
data:image/s3,"s3://crabby-images/45a1b/45a1b6a6a1d7e89ea41a1c6b4c515b15cfa6367e" alt=""
-> data & computed
- data: propriedades de primeira classe
data:image/s3,"s3://crabby-images/e524f/e524fd782fe1c9cd5a3fbc7ad71837b7921ba46e" alt=""
-> data & computed
- computed: estados derivados
data:image/s3,"s3://crabby-images/47ea3/47ea3361926277ae678c5a4a794d468b97a4abf5" alt=""
-> data & computed
- computed: estados derivados
data:image/s3,"s3://crabby-images/9a908/9a908e6f4edd35949d3213c74fe214c35e9c833f" alt=""
-> data & computed
- computed: estados derivados
data:image/s3,"s3://crabby-images/81751/81751cc87c7d28db0962cdae96344598bcc3b874" alt=""
-> data & computed
- computed: estados derivados
data:image/s3,"s3://crabby-images/184ae/184ae6f6d5545447def662a15ffca8b8ccbc0d3b" alt=""
-> data & computed
- computed: cache e reatividade
computed: {
now: function () {
return Date.now()
}
}
computed: {
now: function () {
return this.ticker > 0 ? Date.now() : ''
}
}
-> $options
- referência para as opções do construtor
data:image/s3,"s3://crabby-images/43c2a/43c2a062779cc4794a578e00336357b5c4066ede" alt=""
// analog to
new Vue({ ... })
-> $options
data:image/s3,"s3://crabby-images/a822a/a822a989de6d057cec4be11ab32a6a51d16d99e6" alt=""
-> Vuex
- Variável global gourmetizada
- Sessão
- Reatividade
data:image/s3,"s3://crabby-images/71e59/71e59c27d881e2c612ef0b2d91129072cb5e9f7b" alt=""
-> Vuex
- Observables
- Pub/Sub
data:image/s3,"s3://crabby-images/3aa8d/3aa8d950602af9667029f4addc8f6c17db28a89e" alt=""
-> Vuex
data:image/s3,"s3://crabby-images/73f7e/73f7ee6d1c8bf99eb4f954171bcbbc33e6df429b" alt=""
-> Vuex
data:image/s3,"s3://crabby-images/b6121/b612163e3f9d996f6c39f5925720c8d0340ca585" alt=""
-> $emit & prop vs vuex
data:image/s3,"s3://crabby-images/ef331/ef331bde4554eabfd67d25bc58b6a4b17c39746a" alt=""
- Propagação de estados com $emit & prop
-> $emit & prop vs vuex
data:image/s3,"s3://crabby-images/b9fee/b9fee2b7a6d3129517f12f9ce36824735b19f298" alt=""
- Propagação de estados com vuex
-> Event Bus
- Usar um mesmo componente
- Manipulação de escopo
data:image/s3,"s3://crabby-images/b3d7e/b3d7e7d59728655f642dd566c267486fc4a129ae" alt=""
import Vue from 'vue';
export const EventBus = new Vue();
// Import the EventBus we just created.
import { EventBus } from './event-bus.js';
// emitter
EventBus.$emit('drawer::setState', true)
// receiver
EventBus.$on('drawer::setState', (state) => {
if (state) {
this.openDrawer()
return
}
this.closeDrawer()
})
// receiver
EventBus.$off('drawer::setState')
-> Aplicando extends e mixin
- Extends
- Mixin
- Design Patterns
data:image/s3,"s3://crabby-images/ea264/ea264ee796d74dfbc9ec32f1f0e0cdedf0590661" alt=""
-> Extends
<template>
<button @click="click">Click me</button>
</template>
<script>
export default {
methods: {
click () {
window.alert('Clica ni mim!')
}
}
}
</script>
import AppButton from './AppButton'
export default {
extends: AppButton,
methods: {
click() {
window.alert('Clica aqui no pai!')
}
}
}
-> Extends
data:image/s3,"s3://crabby-images/2b302/2b302bb06b602a4e9e4d4f91a31059005b2b4138" alt=""
data:image/s3,"s3://crabby-images/95337/9533715dd4269a6346728836a69ee0a4a6ab3b0f" alt=""
-> Mixin
data:image/s3,"s3://crabby-images/9b682/9b682eeadba45e83732479fe4da4f59b10bd9240" alt=""
export default {
methods: {
send() {
window.alert('O pai resolveu já!')
}
}
}
-> Design Pattern
- Chain Of Responsibilities
data:image/s3,"s3://crabby-images/6f79c/6f79cf9aecd39ee18f47d0a385ee90db05bb5e06" alt=""
<template>
<button @click="click">Click me</button>
</template>
<script>
export default {
methods: {
click () {
// use here to check in runtinme
this.checkHandler()
this.handler()
},
checkHandler () {
if (this.handler) {
return
}
throw new Error('The "handler" method is required')
}
},
created () {
// check on create component
this.checkHandler()
}
}
</script>
-> Design Pattern
- Chain Of Responsibilities
import AppButton from "./AppButton";
export default {
extends: AppButton,
methods: {
handler() {
window.alert("Ai sim ein?!");
}
}
};
-> Design Pattern
- Strategy
<template>
<button @click="click">Click me</button>
</template>
<script>
export default {
$handler: undefined,
methods: {
click() {
this.$options.$handler.execute();
},
checkHandler() {
if (this.$options.$handler) {
return;
}
throw new Error('The "handler" method is required');
}
},
created() {
this.checkHandler();
}
};
</script>
-> Design Pattern
- Strategy
import AppButton from "./AppButton";
export default {
extends: AppButton,
$handler: {
execute() {
window.alert("Ai sim ein?!");
}
}
};
-> Segredos do watch
- $watch
- immediate
- deep
- Notação usando . (dot notation)
data:image/s3,"s3://crabby-images/b727c/b727cbe2cf6f1eb5511e9f2bbb033503e56fac3d" alt=""
-> Segredos do watch
- $watch
<template>
...
</template>
<script>
export default {
created () {
if (!this.prop) {
return
}
this.unWatch = this.$watch(this.prop, () => this.$emit('it-works'))
}
}
</script>
-> Segredos do watch
- immediate
- deep
<template>
...
</template>
<script>
export default {
watch: {
prop: {
immediate: true,
handler () {
this.$emit('it-works')
}
}
}
}
</script>
<script>
export default {
watch: {
prop: {
deep: true,
handler () {
this.$emit('it-works')
}
}
}
}
</script>
-> Segredos do watch
- Notação usando . (dot notation)
<script>
export default {
watch: {
'$router.fullPath' () {
this.$emit('crazy-day')
}
},
created () {
this.unWatch = this.$watch('orderItem.product.price', ...)
}
}
</script>
-> Templates limpos
- componentes de uma linha
- peças menores compõe maiores
- estados organizados
- eventos bem alinhados
- proxy components podem ajudar
-> Componentes especialistas
<form class="grid">
<AppInput
class="width-6"
label="Name"
v-model="record.name"
/>
<AppInput
class="width-6"
label="Last Name"
v-model="record.lastName"
/>
<AppSelect
class="width-12"
label="Estado"
v-model="record.state"
v-bind="schema.state"
@input="onInputState"
/>
<AppSelect
class="width-12"
label="Cidade"
v-model="record.city"
v-bind="schema.city"
/>
</form>
<template>
<div class="cell">
<label>{{ label }}</label>
<input v-bind="bind" :value="value" @input="input">
</div>
</template>
<script>
export default {
props: ["label", "value"],
computed: {
bind() {
return { ...this.$props, ...this.$attrs };
}
},
methods: {
input($event) {
this.$emit("input", $event.target.value);
}
}
};
</script>
-> Componentes compostos
export default {
name: "AppField",
props: ['label', 'value'],
computed: {
bind() {
return { ...this.$props, ...this.$attrs }
}
},
methods: {
input($event) {
this.$emit('input', this.parseValue($event))
},
parseValue($event) {
return $event.target.value
}
}
}
<template>
<div class="cell">
<label>{{ label }}</label>
<input
v-bind="bind"
:value="value"
@input="input"
/>
</div>
</template>
<script>
import AppField from "./AppField";
export default {
extends: AppField,
name: "AppInput"
};
</script>
<template>
<div class="cell">
<label>{{ label }}</label>
<select v-bind="bind" v-model="model" @input="input">
<option
v-for="(option, index) in options"
:value="option[optionValue]"
>
{{ option[optionLabel] }}
</option>
</select>
</div>
</template>
<script>
import AppField from "./AppField";
export default {
extends: AppField,
methods: {
parseValue($event) {
return this.getOption($event.target.value);
},
// ...
},
watch: {
value: {
immediate: true,
handler(value) {
this.model = value[this.optionValue];
}
}
}
};
</script>
-> Extender componentes
import AppForm from "./AppForm";
import { getStates, getPerson, getCities } from "../services";
export default {
extends: AppForm,
name: "MyForm",
record: { name: undefined, lastName: undefined, state: undefined, city: undefined },
schema: { ... },
computed: {
recordComputed() {
return {
fullName: `${this.record.name} ${this.record.lastName}`
};
}
},
methods: {
async initialize() {
this.schema.state.options = await getStates();
this.record = await getPerson();
}
},
watch: {
async "record.state"(value) {
this.schema.city.options = await getCities(value);
}
}
};
-> Componentes reusáveis
import AppInput from './AppInput'
import AppSelect from './AppSelect'
export default {
name: 'AppForm',
components: { AppInput, AppSelect },
data: () => ({ record: this.$options.record, schema: this.$options.schema }),
computed: {
payload() {
return { ...this.record, ...this.recordComputed }
},
recordComputed() {
return {}
}
},
methods: {
initialize() {}
},
created() {
this.initialize()
}
}
-> Mixins para operações
<template>
<div class="AppForm">
<form class="grid">
<!-- ... -->
<div class="cell buttons">
<button @click="save">Save Me</button>
</div>
</form>
<pre>{{ payload }}</pre>
</div>
</template>
<script>
import AppForm from "./AppForm";
import MixinOperations from "./MixinOperations";
import { getStates, getPerson, getCities } from "../services";
export default {
extends: AppForm,
mixins: [MixinOperations],
name: "MyForm",
// ...
};
</script>
hora das palmas
Links
Dicas e truques para criar códigos manuteníveis usando Vue.js
By William Correa
Dicas e truques para criar códigos manuteníveis usando Vue.js
- 1,200