Check out this simple shopping cart demo:
http://plnkr.co/edit/CHt2iNSRJAJ6OWs7xmiP?p=preview
A user can pick a veggie and a fruit, and it will be added into the cart array. The function that adds a fruit/veggie is very similar, and I want to combine it into a function that can be shared across both components.
selectFruit: function(product){
var cart = this.cart
for(p in cart){
if (cart[p]["type"] == "fruit"){
console.log("We already got a fruit!, Let's remove " + cart[p]["name"] + " and add in " + product["name"]);
this.cart.$remove(cart[p])
}
}
console.log("Adding " + product.name + " to cart.");
var productName = product.name
var cartFruit = {name: product.name, type: 'fruit'}
this.cart.push(cartFruit)
}
selectVeggie: function(product){
var cart = this.cart
for(p in cart){
if (cart[p]["type"] == "veggie"){
console.log("We already got a veggie!, Let's remove " + cart[p]["name"] + " and add in " + product["name"]);
this.cart.$remove(cart[p])
}
}
console.log("Adding " + product.name + " to cart.");
var productName = product.name
var cartVeggie = {name: product.name, type: 'veggie'}
this.cart.push(cartVeggie)
}
How can I make it so I can alter this method and have it used globally? I'm using the Vue Router with this project btw, thanks for any help!
I found this technique to be more simple/satisfactory, as I prefer composition over inheritance:
src/shared.js
src/yourcomponent.vue
This will also allow you to write Vue-agnostic tests.
Option 1
One approach for sharing your method across components is to use a mixin. Here's a
cartMixin
that contains aselectProduct
method:You can reference this in each component like this:
... and then use it in your templates like this:
Here's a fork of your Plunker.
Option 2
After thinking about this some more, another option you might consider is to create a base
Product
component and extend that to create yourFruit
andVegetable
components. You would then put your common functionality in the base component.Here's a Plunker with this approach.
Given that your Fruit and Vegetable templates are so similar, you might be able to take this idea even further and use a common template from the base component.
You can put the method in your root Vue instance and then dispatch an event from the child instance when a veggie is selected, or when a fruit is selected. Events look for a handler on their parent component, and if they don't find an event handler they keep going up the chain until they do. So on your root instance:
Then on the child instance: