Monterail Academy
Co dziś będziemy robić
Habit Tracker
data:image/s3,"s3://crabby-images/10c3b/10c3bda32574600ee2fc5a68384488023f5a6e29" alt=""
co jest potrzebne żeby zacząć
data:image/s3,"s3://crabby-images/4a0df/4a0df28d822e8892a48de7fae0328296c56f303e" alt=""
data:image/s3,"s3://crabby-images/949ff/949ff22ecaa46d160c1d20268d9e9800472f2fd8" alt=""
data:image/s3,"s3://crabby-images/d5b4a/d5b4a29082aee82aad8a4ad0c8f9880f355a83e4" alt=""
Środowisko
instalowane razem
co jest potrzebne żeby zacząć
npm install --global poi
data:image/s3,"s3://crabby-images/30304/30304e841a4071171a0bfc54a38932cdcb400530" alt=""
TWORZYMY PROJEKT
Nowy folder i dwa pliki:
data:image/s3,"s3://crabby-images/937c0/937c0ed1fba0d59d67bfd4294f85fc8d72ee452b" alt=""
Zadanie 1. TWORZYMY PROJEKT
- Uruchom poi
- Włącz przeglądarkę (biała strona)
$ npm install --global poi
$ poi index.js
data:image/s3,"s3://crabby-images/c9803/c980366517bed67cda91285de119079c7ec6813a" alt=""
Hello World!
"montowanie" komponentu, wersja fiddle
data:image/s3,"s3://crabby-images/fceda/fceda994d40104ba86f9a4d42862d390d3c0a678" alt=""
template: '<...>'
render () { ... }
- fiddle dostarcza "Vue" jako global
Hello World!
"montowanie" komponentu, wersja poi
data:image/s3,"s3://crabby-images/c4f61/c4f616f9df0bbf1a74af92d663f88b13e4f8ae5e" alt=""
nazwa zmiennej w pliku
ścieżka/nazwa modułu
- poi dostarcza "Vue" jako moduł
Hello World!
SFC - Single File Component
data:image/s3,"s3://crabby-images/394bc/394bca35babb64d606ec976193b45a3735433313" alt=""
- ścieżka lokalna zawsze od ./
- bez rozszerzenia .vue lub .js
Hello World!
SFC - Single File Component
data:image/s3,"s3://crabby-images/394bc/394bca35babb64d606ec976193b45a3735433313" alt=""
- to samo co `<App />` w html
Hello World!
SFC - Single File Component
data:image/s3,"s3://crabby-images/35093/3509370a4c5646dac12f982133f1076fa131eea4" alt=""
-
index.js - montowanie Vue i render
- App.vue - SFC - kod komponentu
zadanie 2. Hello vue world
data:image/s3,"s3://crabby-images/07ff7/07ff706b8b55d0e426d00d2f8b423fcb232a8d63" alt=""
zadanie dodatkowe: input field
data:image/s3,"s3://crabby-images/394bc/394bca35babb64d606ec976193b45a3735433313" alt=""
data i v-for
lista elementów
Stan aplikacji
Reprezentacja wizualna
render
template
data
v-for="element in list"
list: ["a", "b", "c"]
lista elementów
count | label |
---|---|
2 | pizza |
1 | kebab |
3 | pierogi |
data () {
return {
items: [
{ count: 2, label: 'pizza' },
{ count: 1, label: 'kebab' },
{ count: 3, label: 'pierogi' }
]
}
}
dane
lista elementów
2x pizza
<ul>
<li v-for="item in list">
{{ item.count }}x {{ item.label }}
</li>
</ul>
1x kebab
3x pierogi
3x pierogi
- usunięcie ostatniego elementu
- dostosowanie treści pozostałych
1x kebab
3x pierogi
data () {
return {
items: [
{ count: 2, label: 'pizza' },
{ count: 1, label: 'kebab' },
{ count: 3, label: 'pierogi' }
]
}
}
lista elementów
2x pizza
<ul>
<li v-for="item in list" :key="element.label">
{{ item.count }}x {{ item.label }}
</li>
</ul>
1x kebab
3x pierogi
- usunięcie właściwego elementu
data () {
return {
items: [
{ count: 2, label: 'pizza' },
{ count: 1, label: 'kebab' },
{ count: 3, label: 'pierogi' }
]
}
}
HABIT
{
id: Number,
name: String,
color: String,
track: Array,
weeklyTarget: Number
}
Zadanie 3. habit list
{
id: 1234,
name: "wake up early",
color: "#bada55",
track: [],
weeklyTarget: 5
}
- wprowadź dane w postaci listy obiektów
- wyświetl listę obiektów z nazwami
data:image/s3,"s3://crabby-images/10471/1047141188a0f8613d14d77bb1ff60be71bb2720" alt=""
Dodatkowe:
* pokoloruj tekst
UI komponenty i layout
INSTALOWANIE ZALEŻNOŚCI
Element UI
http://element.eleme.io/
data:image/s3,"s3://crabby-images/b9e95/b9e95ce1c09644871edf66419cc54c5d48f4641c" alt=""
data:image/s3,"s3://crabby-images/d1425/d14258c70a6862fef1b9f5d3b2e0c70108dff64b" alt=""
data:image/s3,"s3://crabby-images/4631a/4631ab41c460a3f7807be15073c1234006957782" alt=""
Element UI
http://element.eleme.io/
$ npm install element-ui
Element UI
data:image/s3,"s3://crabby-images/8ed0e/8ed0e47cac87f1fd28437b9d509089d9be3f434d" alt=""
- http://element.eleme.io/#/en-US/component/quickstart
- http://element.eleme.io/#/en-US/component/i18n
CSS załączamy jak JS
- dodaj el-checkbox na każdy dzień tygodnia
- dodaj el-progress
- wyznacz progress przez wyrażenie w html (ile %)
- dodaj guzik do usuwania elementu
zadanie 4. przypięcie danych
data:image/s3,"s3://crabby-images/8ed0e/8ed0e47cac87f1fd28437b9d509089d9be3f434d" alt=""
data:image/s3,"s3://crabby-images/e277d/e277de8327b992b41525d28e53a2abf62495a415" alt=""
tip:
<el-checkbox-group v-model="habit.track">
$ npm install element-ui
Wydzielamy komponent
- Więcej niż jeden plik *.vue
- rejestracja komponentu
- props i :bind
Wydzielamy komponent
data:image/s3,"s3://crabby-images/c29e8/c29e87125c7a2a80ea98a7c4c852b4a0bcfe2dc3" alt=""
import HabitCard from './HabitCard'
export default {
components: {
HabitCard
}
}
Well - styled component
<Well>
<span>something something...</span>
</Well>
<template>
<div class="well">
<slot/>
</div>
</template>
<style scoped>
.well {
// moje style dla well
}
</style>
- slot - miejsce w które wpadnie "wnętrze" komponentu
- scoped style - działa tylko w swoim komponencie
Well.vue
Habit card - props
<HabitCard :habit="objectFromList"/>
<template>
<span>{{ habit.name }}</span>
</template>
<script>
export default {
props: {
habit: Object
}
}
</script>
HabitCard.vue
App.vue
odbieranie danych z zewnątrz
Habit card - props
definiowanie prop
// przekazanie zmiennej someData
// przekazanie ciągu 'someData'
// przekazanie ciągu 'someData'
<div
:prop="someData"
prop="someData"
:prop="'someData'"
>
<HabitCard :habit="objectFromList"/>
App.vue
Przygotuj oddzielne komponenty z props i actions:
- <Well>: komponent wyłącznie ze stylem ramki
- tylko template i style
- <HabitCard>: element listy, który
- używa <Well>
- przyjmuje obiekt habit (prop)
Zadanie 5. Komponenty
data:image/s3,"s3://crabby-images/c29e8/c29e87125c7a2a80ea98a7c4c852b4a0bcfe2dc3" alt=""
import HabitCard from './HabitCard'
export default {
components: {
HabitCard
}
}
<HabitCard :habit="objectFromList"/>
robimy Formularz
- @zdarzenie
- metody
- v-model
ZDARZENIA - V-ON (@)
<button @click="doSomething"/>
// ...
methods: {
doSomething (evt) {
console.log('clicked!', evt)
}
}
// clicked! MouseEvent { ... }
zdarzenia - v-on (@)
<button @click="doSomething($event, 'hello')"/>
// ...
methods: {
doSomething (evt, extraParam) {
console.log(extraParam, evt)
}
}
// hello MouseEvent { ... }
specjalna nazwa
zdarzenia - $emit
<button @click="emitRemove">remove</button>
// ...
methods: {
emitRemove() {
this.$emit('remove')
}
}
<MyComponent @remove="handleRemove"/>
MyComponent.vue
inny komponent
zdarzenia
<el-form @submit.native.prevent="onSubmit">
...
</el-form>
Modyfikatory
- native - chodzi nam tylko o event z html, nie z $emit
- prevent - nie rób prawdziwego submit ( preventDefault() )
v-model
<input v-model="someData" />
<input
:value="someData"
@input="someData = $event.target.value"
/>
Formularz
data:image/s3,"s3://crabby-images/4631a/4631ab41c460a3f7807be15073c1234006957782" alt=""
Formularz
Nasz formularz potrzebuje:
- nazwę trenowanej czynności
- tygodniowy cel wykonania czynności
- opcjonalnie: kolor
{
id: Number,
name: String,
color: String,
track: Array,
weeklyTarget: Number
}
-
utwórz komponent z formularzem
-
dodaj edytowany element do listy na akcję @submit
-
reset formularza po dodaniu
zadanie 6. Formularz
data:image/s3,"s3://crabby-images/fcbcb/fcbcba5c384c321c2be2507ebcb07c95c7a4f3cd" alt=""
-
Zadania z *:
-
usuwanie elementu listy
-
color picker
-
animacja dodawania
<el-form @submit.native.prevent="onSubmit">
...
</el-form>
vue-router
- przenosimy formularz do osobnej strony
- wydzielamy osobny moduł na dane
vue-router
$ npm install vue-router
data:image/s3,"s3://crabby-images/c35e5/c35e5c8543cf4e5c728f0b5398940707b39b2dc0" alt=""
vue-router
data:image/s3,"s3://crabby-images/5d2fe/5d2fe49ec79ec00d308b0b197f34346ff36b7ada" alt=""
jak "slot" dla zawartości podstrony
vue-router
data:image/s3,"s3://crabby-images/c35e5/c35e5c8543cf4e5c728f0b5398940707b39b2dc0" alt=""
vue-router
import HabitList from './components/HabitList.vue'
import NewHabit from './components/NewHabit.vue'
import EditHabit from './components/EditHabit.vue'
Problemy:
- jak współdzielić dane między stronami?
- jak ograniczyć operacje na danych do kilku przewidywalnych akcji?
Centralny zbiór danych
jak współdzielić dane między stronami?
Centralny zbiór danych
data:image/s3,"s3://crabby-images/d2ea4/d2ea433678e0f4c7a89cd767ea7d62acee6788a6" alt=""
data:image/s3,"s3://crabby-images/80f62/80f62d03b9741a31306687ad8f83f33c2f81852d" alt=""
jak ograniczyć operacje na danych?
Centralny zbiór danych
data:image/s3,"s3://crabby-images/cd23b/cd23b883fd3c7b96101f52f7e3952d7bd78c67c1" alt=""
data:image/s3,"s3://crabby-images/2a7b0/2a7b0eb8fee851c6bdc1f6b633df79432e84f11c" alt=""
Zadania z *:
- przygotuj stronę do edycji istniejącego habitu
- animuj przejścia między stronami
- zapisuj i odczytuj dane z localstorage (persystencja)
Zadanie 7. Router
- oddziel dane do osobnego pliku store.js
- dodaj router i zdefiniuj "routes"
- Przygotuj osobną stronę do dodawania habitu
data:image/s3,"s3://crabby-images/56ca7/56ca764e2d709a7cc4b13e8d73b544b17e4639a6" alt=""
<router-view/>
Zadania z *:
- przygotuj stronę do edycji istniejącego habitu
- animuj przejścia między stronami
- zapisuj i odczytuj dane z localstorage (persystencja)
Zadanie 7. Router
- oddziel dane do osobnego pliku store.js
- dodaj router i zdefiniuj "routes"
- Przygotuj osobną stronę do dodawania habitu
data:image/s3,"s3://crabby-images/56ca7/56ca764e2d709a7cc4b13e8d73b544b17e4639a6" alt=""
<router-view/>
Next steps: VUEX
monterail academy vue
By Pawel Grabarz
monterail academy vue
- 1,254