A Vue.js wrapper component that turns everything into fun scratch cards. Includes touch support without additional dependencies.
It can also calculate percentage value of the scratchables's cleared area.
Install it with npm:
npm i vue-scratchable
Or directly in the browser:
<script src="https://unpkg.com/vue-scratchable@latest/dist/vue-scratchable.umd.min.js"></script>
Register the component globally:
import VueScratchable from 'vue-scratchable';
Vue.component('vue-scratchable', VueScratchable);
Or use it locally in your Single File Components:
import VueScratchable from 'vue-scratchable';
export default {
/* ... */
components: {
VueScratchable,
},
/* ... */
};
In both cases you are now able to use it in the <template>
of a component:
<vue-scratchable>
<h1>Hello Scratchable World</h1>
</vue-scratchable>
This code is taken from this project's App.vue
file to showcase the component's easy way of use.
<template>
<div id="app">
<h1>A beautiful parrot got trapped behind some paper.</h1>
<h2>Scratch them free!</h2>
<vue-scratchable
v-slot="{ init }"
:brush-options="brush"
:hide-options="hide"
get-percentage-cleared
@percentage-update="updatePoints"
>
<div class="wrapper">
<img
:src="require('./assets/mehmet-turgut-kirkgoz-vnayhPykN5Q-unsplash.jpg')"
@load="init()"
>
<h3>{{ subline }}</h3>
</div>
</vue-scratchable>
<p>You scratched {{ percentage }}% free.</p>
<pre>Photo by <a href="https://unsplash.com/@tkirkgoz?utm_source=unsplash&utm_medium=referral&utm_content=creditCopyText">Mehmet Turgut Kirkgoz</a> on <a href="https://unsplash.com/t/animals?utm_source=unsplash&utm_medium=referral&utm_content=creditCopyText">Unsplash</a></pre>
</div>
</template>
<script>
import VueScratchable from 'vue-scratchable';
import paperPattern from './assets/natural-paper-texture.jpg';
export default {
name: 'App',
components: {
VueScratchable,
},
computed: {
subline() {
return this.percentage < 100
? `π There is still ${100 - this.percentage}% left for me to be free... π`
: 'π Thank you for scratching me free! π';
},
},
data() {
return {
percentage: 0,
hide: {
type: 'pattern',
src: paperPattern,
repeat: 'repeat',
},
brush: {
size: 60,
shape: 'round',
},
};
},
methods: {
updatePoints(percentage) {
this.percentage = percentage;
},
},
};
</script>
<style>
body {
background-color: #333;
margin: 0;
}
#app {
font-family: 'Open Sans', sans-serif;
color: white;
display: flex;
flex-direction: column;
align-items: center;
max-width: 1200px;
margin: 0 auto;
margin-top: 50px;
}
.vue-scratchable-wrapper {
background-color: white;
}
h3 {
color: #2c3e50;
text-align: center;
}
a {
color: #2196f3;
}
</style>
Slot name | Description |
---|---|
default | The content to be scratched free. |
Property | Type | Description |
---|---|---|
brush-options | Object | Configuration object of the "scratcher". See below for possible options. |
hide-options | Object | Configuration object of the scratchable layer. See below for possible options. |
get-percentage-cleared | Boolean | Flag to enable the percentage-update event which emits the amount of cleared paint as percentage. |
percentage-stride | Number | A stride used while calculating the cleared percentage to reduce calculation time. |
Property | Type | Default | Description |
---|---|---|---|
size | Number | 20 | Defines the lineWidth attribute. |
shape | String | round | Defines the lineJoin attribute. |
There are two different types of fill that can be applied to the scratchable area: a solid colour or a canvas pattern. These are differentiated by the type
property:
Property | Type | Description |
---|---|---|
type | String | Can be 'color' or 'pattern'. If you want it to be a solid colour you should set it to 'color'. |
value | String | The colour you want for the fill. Can be any type of colour: hex or rgba. |
Example:
const hide = {
type: 'color',
value: '#333',
};
Property | Type | Description |
---|---|---|
type | String | Can be 'color' or 'pattern'. If you want it to be a pattern you should set it to 'pattern'. |
src | String | A link to an image (best case is a repeatable texture). Can be an external Link as well as an imported local asset. |
repeat | String | Defines whether and in which direction the image should be repeated. Possible values are "repeat", "no-repeat", "repeat-x" and "repeat-y". |
Example:
const hide = {
type: 'pattern',
src: 'https://mdn.mozillademos.org/files/222/Canvas_createpattern.png',
repeat: 'repeat',
};
Be aware that if you will be referencing an external image in src
, you should guarantee the remote source does respond with an Access-Control-Allow-Origin
Header being equal to one of the following:
- to the value of the Request Header
Origin
- automatically set by the browser, being the base url of your website - to the value
*
- allowing any client-side application to request your image as well
Failing to do this will show CORS-related issues in the console of your application.
Event name | Parameter type | Description |
---|---|---|
percentage-update | Number | If the get-percentage-cleared flag is set the component will emit this event and pass a number calculated from the percentage amount of the cleared area. |
- Img tags with local assets
When using img tags with local imported assets you have to call the init
method of the component corresponding to the img tag's @load
event. The init
method can be accessed through the component's scoped slot.
Example:
<vue-scratchable v-slot="{ init }">
<img
:src="require('./img.jpg')"
@load="init()"
>
</vue-scratchable>
- Percentage calculation is very taxing on performance
The cleared area percentage calculation has to take every pixel of the canvas into consideration and analyzes whether they are cleared or not. Since this calculation gets called on every draw step this needs a lot of processing power. Because of that this feature is disabled by default and needs to be explicitly activated.
That's also why the percentage-stride
property should be set wisely and adjusted to your needs. It defines how many pixels should be skipped while calculating. This obviously also decreases the percentage's value accuracy.
If you find a bug or want to contribute to the code or documentation, you can help by submitting an issue or a pull request.