Guia
Essenciais
- Instalação
- Introdução
- A Instância Vue
- Sintaxe de Templates
- Dados Computados e Observadores
- Interligações de Classe e Estilo
- Renderização Condicional
- Renderização de Listas
- Manipulação de Eventos
- Interligações em Formulários
- Básico sobre Componentes
Componentes em Detalhes
- Registro de Componentes
- Propriedades
- Eventos Personalizados
- Slots
- Dinâmicos & Assíncronos
- Lidando com Casos Extremos
Transições & Animações
- Transições de Visibilidade e Listas
- Transições de Estado
Reuso & Composição
- Mixins
- Diretivas Personalizadas
- Funções de Renderização & JSX
- Plugins
- Filtros
Ferramentas
- Componentes Single-File
- Testes Unitários
- Testing
- Suporte ao TypeScript
- Publicando em Produção
Escalonando
- Roteamento
- Gerenciamento de Estado
- Renderizando no Lado do Servidor
- Segurança
Internamente
- Reatividade em Profundidade
Migração
- Migração do Vue 1.x
- Migração do Vue Router 0.7.x
- Migração do Vuex 0.6.x para 1.0
Diversos
- Comparação com Outros Frameworks
- Junte-se à Comunidade Vue.js!
- Conheça a Equipe
Você está navegando a documentação da v2.x e anterior. Para a v3.x, clique aqui.
Diretivas Personalizadas
Introdução
Adicionalmente ao conjunto de diretivas incluídas em seu núcleo (v-model
e v-show
), Vue permite registrar suas próprias diretivas personalizadas. Note que no Vue 2.0, a forma primária de abstração e reuso de código são componentes - no entanto, pode haver casos em que você só precisa de um acesso de baixo nível ao DOM em elementos simples, e aí diretivas personalizadas seriam úteis. Um exemplo seria colocar o foco a um elemento input como este:
Quando a página carrega, o elemento ganha o foco (nota: o atributo nativo autofocus
não funciona no Safari para dispositivos móveis). Na verdade, se você ainda não clicou em nada desde que visitou esta página do guia, o input acima deverá estar com o foco. Agora vamos construir a diretiva que realiza isto:
// Registra a diretiva personalizada global chamada `v-focus`
Vue.directive('focus', {
// Quando o elemento vinculado é inserido no DOM...
inserted: function (el) {
// Coloque o foco no elemento
el.focus()
}
})
Se você deseja registrar uma diretiva localmente em vez disso, os componentes também aceitam uma opção directives
:
directives: {
focus: {
// definição da diretiva
inserted: function (el) {
el.focus()
}
}
}
Então, em um template você pode usar o novo atributo v-focus
para qualquer elemento, por exemplo:
<input v-focus>
Funções de Gatilhos
Um objeto de definição de diretiva pode prover algumas funções de gatilhos (todas opcionais):
bind
: chamada apenas uma vez, quando a directiva é interligada pela primeira vez ao elemento. Aí é onde você pode fazer o trabalho de configuração inicial.inserted
: chamada quando o elemento for inserido no nó pai (garante a presença no nó pai, mas não necessariamente no documento).update
: chamada após a atualização do VNode que contém o componente, mas possivelmente antes da atualização de seus filhos. O valor da diretiva pode ou não ter mudado, mas você pode evitar atualizações desnecessárias comparando os valores atuais com os antigos (veja abaixo, em argumentos dos gatilhos).
Abordaremos o VNodes com mais detalhes depois, quando discutirmos funções de renderização.
componentUpdated
: chamada após a atualização do Vnode que contém o componente, inclusive de seus filhos.unbind
: chamada somente uma vez, quando a diretiva é desvinculada do elemento.
Iremos a seguir explorar os argumentos que podem ser passados nessas funções de gatilho (el
, binding
, vnode
, e oldVnode
).
Argumentos nos Gatilhos
Os seguintes argumentos são esperados nas funções de gatilho das diretivas:
el
: O elemento a que a diretiva está vinculada. Isso pode ser usado para manipular o DOM diretamente.binding
: Um objeto contendo as seguintes propriedades:name
: O nome da diretiva, sem o prefixov-
.value
: O valor passado para a diretiva. Por exemplo emv-my-directive="1 + 1"
, o valor passado seria2
.oldValue
: O valor anterior, somente disponível emupdate
ecomponentUpdated
. Está presente tanto se o valor foi alterado quanto não alterado.expression
: A expressão de vinculação como uma String. Por exemplo emv-my-directive="1 + 1"
, a expressão seria"1 + 1"
.arg
: O argumento passado para a diretiva, se houver algum. Por exemplo emv-my-directive:foo
, o argumento seria"foo"
.modifiers
: Um objeto contendo modificadores, se houver algum. Por exemplo emv-my-directive.foo.bar
, o objeto seria{ foo: true, bar: true }
.
vnode
: O nó virtual produzido pelo compilador do Vue. Veja VNode API para mais detalhes.oldVnode
: O nó virtual anterior, somente disponível emupdate
ecomponentUpdated
.
Exceto el
, você deve tratar os outros argumentos como somente leitura e nunca modificá-los. Se você precisar compartilhar informações entre gatilhos, é aconselhável utilizar um atributo dataset.
Eis um exemplo de diretiva personalizada utilizando algumas dessas propriedades:
<div id="hook-arguments-example" v-demo:foo.a.b="message"></div>
Vue.directive('demo', {
bind: function (el, binding, vnode) {
var s = JSON.stringify
el.innerHTML =
'name: ' + s(binding.name) + '<br>' +
'value: ' + s(binding.value) + '<br>' +
'expression: ' + s(binding.expression) + '<br>' +
'argument: ' + s(binding.arg) + '<br>' +
'modifiers: ' + s(binding.modifiers) + '<br>' +
'vnode keys: ' + Object.keys(vnode).join(', ')
}
})
new Vue({
el: '#hook-arguments-example',
data: {
message: 'Olá!'
}
})
value: "Olá!"
expression: "message"
argument: "foo"
modifiers: {"a":true,"b":true}
vnode keys: tag, data, children, text, elm, ns, context, fnContext, fnOptions, fnScopeId, key, componentOptions, componentInstance, parent, raw, isStatic, isRootInsert, isComment, isCloned, isOnce, asyncFactory, asyncMeta, isAsyncPlaceholder
Argumentos de Diretiva Dinâmicos
Os argumentos de diretiva podem ser dinâmicos. Por exemplo, em v-mydirective:[argument]="value"
, o argument
pode ser atualizado baseado em propriedades de dados em nossa instância de componente! Isso torna nossas diretivas personalizadas flexíveis para serem utilizadas ao longo da aplicação.
Digamos que você quer fazer uma diretiva personalizada que permite fixar elementos em sua página através de posicionamento fixed
. Poderíamos criar uma diretiva personalizada em que os valores atualizassem o posicionamento vertical em pixels, desta forma:
<div id="baseexample">
<p>Role a página para baixo</p>
<p v-pin="200">Me prenda 200px a partir do topo da página</p>
</div>
Vue.directive('pin', {
bind: function (el, binding, vnode) {
el.style.position = 'fixed'
el.style.top = binding.value + 'px'
}
})
new Vue({
el: '#baseexample'
})
Isto fixaria o elemento 200px a partir do topo da página. Mas e se encontrássemos um cenário em que precisássemos fixar o elemento a partir da esquerda, ao invés do topo? Aqui está um argumento dinâmico que pode ser atualizado em cada instância de componente:
<div id="dynamicexample">
<h3>Role para baixo dentro desta seção ↓</h3>
<p v-pin:[direction]="200">Estou fixo na página 200px a partir da esquerda.</p>
</div>
Vue.directive('pin', {
bind: function (el, binding, vnode) {
el.style.position = 'fixed'
var s = (binding.arg == 'left' ? 'left' : 'top')
el.style[s] = binding.value + 'px'
}
})
new Vue({
el: '#dynamicexample',
data: function () {
return {
direction: 'left'
}
}
})
Resultado:
Nossa diretiva personalizada agora é flexível o suficiente para suportar alguns casos de uso diferentes.
Forma Abreviada de Funções
Em muitos casos, você pode querer ter o mesmo comportamento nos gatilhos bind
e update
, e não se importar com os outros gatilhos. Por exemplo:
Vue.directive('color-swatch', function (el, binding) {
el.style.backgroundColor = binding.value
})
Objetos Literais
Se sua diretiva precisa de múltiplos valores, você pode passá-los em um objeto literal JavaScript. Lembre-se que as diretivas podem ter qualquer objeto JavaScript válido.
<div v-demo="{ color: 'white', text: 'Olá!' }"></div>
Vue.directive('demo', function (el, binding) {
console.log(binding.value.color) // => "white"
console.log(binding.value.text) // => "Olá!"
})