2018-03-29 从MVC到MVVM

关于axios
是一个ajax的库

以前用ajax是这么用的


image.png

可以用axios.ajax()
也可以用axios.post()
也可以用axios.get()
支持更多的API


image.png

1、比jQuery.ajax的功能更多
2、除了ajax功能之外,就没有其他功能,更专注

jQuery的ajax功能用axios替换
jQuery的DOM功能用Vue替换

写个demo


image.png
image.png

这个原始的书本值是死,我们希望是从数据库里去,这个axios支持自己造数据


image.png

用个api,config

image.png

data上面的就是config,config有什么用呢?
image.png

里面有个url method data比较重要
image.png

当路径和method是对的时候,就把响应的数据填上name:tsl
image.png

let config = response.config
let {method,url,data} =config //data是请求的data
上面2句可以用1句代替
let{config:{method,url,data}}=response
image.png

找到占位符,然后进行html替换
问题:点击增加、减少按钮不起作用
因为上面的代码已经把html更新了,
怎么解决?用事件委托
image.png

问题:加减只是在当前页面能显示,即加减是对页面的内容直接更新,我们需要进行发请求
服务器put成功了,才改本地的

let book={
'name':'tsl',
'number':1,
id:1
}

//在真正返回response之前使用
axios.interceptors.response.use(function(response){
let config = response.config
let {method,url,data} =config //data是请求的data

// let {config:{method,url,data}}=response

if(url==='/books/1' && method==='get'){
   response.data=book   //响应的data
}else if(url==='/books/1'&& method ==='put'){

// response.data={
// name:'tsl',
// id:1,
// number = data.number //data是请求的data
// }
Object.assign(book,data) //data是请求的data
response.data=book
}

return response

})

axios.get('/books/1')
.then((response)=>{
let data = response.data
let originalHtml = $('#app').html()
let newHtml = originalHtml.replace('name',data.name)
.replace('number',data.number)
$('#app').html(newHtml)

})

$('#app').on('click','#addOne',function(){
var oldNumber = $('#number').text()//string
var newNumber = oldNumber-0+1
axios.put('/books/1',function(){ //post也可以
number:newNumber
}).then(()=>{ //后台修改成功了再更新页面
$('#number').text(newNumber)
})
})

$('#app').on('click','#minusOne',function(){
var oldNumber = $('#number').text() //string
var newNumber = oldNumber-0-1
axios.put('/books/1',function(){
number:newNumber
}).then(()=>{
$('#number').text(newNumber)
})
})

$('#app').on('click','#reset',function(){
axios.put('/books/1',function(){
number:0
}).then(()=>{
$('#number').text(0)
})
})

image.png

image.png

image.png

接下里用MVC思想来进行构建
fakeDate()

let model = {
data: {
name: '',
number: 0,
id: ''
},
fetch(id) {
return axios.get(/books/${id}).then((response) => {
this.data = response.data
return response
})
},
updata(id, data) {
// console.log(data)
return axios.put(/books/${id}, data).then((response) => {
this.data = response.data
// console.log(response)
return response
})
}
}

let view = {
el: '#app',
template: <div> 书名:《__name__》 数量:<span id="number">__number__</span> </div> <div> <button id="addOne">加1</button> <button id="minusOne">减1</button> <button id="reset">重置</button> </div>,
render(data) {
let html = this.template.replace('name', data.name)
.replace('number', data.number)
$(this.el).html(html)
}
}

let controller = {
init(options) {
let view = options.view
let model = options.model
this.view = view
this.model = model
this.view.render(this.model.data)
this.bindEvents()
// axios.get('/books/1')
this.model.fetch(1)
.then(() => {
// let data = response.data
// let originalHtml = $('#app').html()
// let newHtml = originalHtml.replace('name',data.name)
// .replace('number',data.number)
// $('#app').html(newHtml)
this.view.render(this.model.data)
// view.render(model.data)也可以

        })
},
addOne() {
    var oldNumber = $('#number').text()//string
    var newNumber = oldNumber - 0 + 1
    //   axios.put('/books/1',function(){   //post也可以
    //     number:newNumber
    //   })
    this.model.updata(1, { number: newNumber })
        .then(() => {    //后台修改成功了再更新页面
            //         $('#number').text(newNumber)
            //               console.log(this.model.data)
            this.view.render(this.model.data)
        })
},
minusOne() {
    var oldNumber = $('#number').text() //string
    var newNumber = oldNumber - 0 - 1
    //   axios.put('/books/1',function(){
    //     number:newNumber
    //   })
    this.model.updata(1, { number: newNumber })
        .then(() => {
            //       $('#number').text(newNumber)
            this.view.render(this.model.data)
        })
},
reset() {
    //   axios.put('/books/1',function(){
    //     number:0
    //   })
    this.model.updata(1, { number: 0 })
        .then(() => {
            //       $('#number').text(0)
            this.view.render(this.model.data)
        })
},

bindEvents() {
    $(this.view.el).on('click', '#addOne', this.addOne.bind(this))

    $(this.view.el).on('click', '#minusOne', this.minusOne.bind(this))

    $(this.view.el).on('click', '#reset', this.reset.bind(this))
}

}

controller.init({ view: view, model: model })

function fakeDate() {
let book = {
'name': 'tsl',
'number': 1,
id: 1
}

//在真正返回response之前使用
axios.interceptors.response.use(function (response) {
    let config = response.config
    let { method, url, data } = config //data是请求的data

    //     let {config:{method,url,data}}=response

    if (url === '/books/1' && method === 'get') {
        response.data = book   //响应的data
    } else if (url === '/books/1' && method === 'put') {
        //       response.data={
        //         name:'tsl',
        //         id:1,
        //         number = data.number  //data是请求的data
        //       }
        data = JSON.parse(data)  //字符串转换下
        Object.assign(book, data)  //data是请求的data
        response.data = book
    }

    return response
})

}

注意这里面的this绑定

继续代码构建,如果再写个页面还是需要model、view、controller,那么我们把相同的归下类

用构造函数的方式
fakeDate()

function Model(options) {
this.data = options.data
this.resource = options.resource

}

Model.prototype.fetch = function (id) {
return axios.get(/${this.resource}/${id}).then((response) => {
this.data = response.data
return response
})
}
Model.prototype.updata = function (id, data) {
return axios.put(/${this.resource}/${id}, data).then((response) => {
this.data = response.data
// console.log(response)
return response
})
}

function View({ el, template }) {
this.el = el
this.template = template
}

View.prototype.render = function (data) {
let html = this.template
for (let key in data) {
html = html.replace(__${key}__, data[key])
}
console.log(html)
$(this.el).html(html)
}

//---------------------上面是类,下面是对象,对象是从类派生出来的

let model = new Model({
data: { //data是特有属性
name: '',
number: 0,
id: ''
},
resource: 'books'
}
)

let view = new View({
el: '#app',
template: <div> 书名:《__name__》 数量:<span id="number">__number__</span> </div> <div> <button id="addOne">加1</button> <button id="minusOne">减1</button> <button id="reset">重置</button> </div>
})

let controller = {
init(options) {
let view = options.view
let model = options.model
this.view = view
this.model = model
this.view.render(this.model.data)
this.bindEvents()
// axios.get('/books/1')
this.model.fetch(1)
.then(() => {

            this.view.render(this.model.data)

        })
},
addOne() {
    var oldNumber = $('#number').text()//string
    var newNumber = oldNumber - 0 + 1
    //   axios.put('/books/1',function(){   //post也可以
    //     number:newNumber
    //   })
    this.model.updata(1, { number: newNumber })
        .then(() => {    //后台修改成功了再更新页面
            //         $('#number').text(newNumber)
            //               console.log(this.model.data)
            this.view.render(this.model.data)
        })
},
minusOne() {
    var oldNumber = $('#number').text() //string
    var newNumber = oldNumber - 0 - 1
    //   axios.put('/books/1',function(){
    //     number:newNumber
    //   })
    this.model.updata(1, { number: newNumber })
        .then(() => {
            //       $('#number').text(newNumber)
            this.view.render(this.model.data)
        })
},
reset() {
    //   axios.put('/books/1',function(){
    //     number:0
    //   })
    this.model.updata(1, { number: 0 })
        .then(() => {
            //       $('#number').text(0)
            this.view.render(this.model.data)
        })
},

bindEvents() {
    $(this.view.el).on('click', '#addOne', this.addOne.bind(this))

    $(this.view.el).on('click', '#minusOne', this.minusOne.bind(this))

    $(this.view.el).on('click', '#reset', this.reset.bind(this))
}

}

controller.init({ view: view, model: model })

function fakeDate() {
let book = {
'name': 'tsl',
'number': 1,
id: 1
}

//在真正返回response之前使用
axios.interceptors.response.use(function (response) {
    let config = response.config
    let { method, url, data } = config //data是请求的data

    //     let {config:{method,url,data}}=response

    if (url === '/books/1' && method === 'get') {
        response.data = book   //响应的data
    } else if (url === '/books/1' && method === 'put') {
        //       response.data={
        //         name:'tsl',
        //         id:1,
        //         number = data.number  //data是请求的data
        //       }
        data = JSON.parse(data)  //字符串转换下
        Object.assign(book, data)  //data是请求的data
        response.data = book
    }

    return response
})

}

接下里用vue来重构代码
让MVC的C变的更智能
fakeDate()

function Model(options) {
this.data = options.data
this.resource = options.resource

}

Model.prototype.fetch = function (id) {
return axios.get(/${this.resource}/${id}).then((response) => {
this.data = response.data
return response
})
}
Model.prototype.updata = function (id, data) {
return axios.put(/${this.resource}/${id}, data).then((response) => {
this.data = response.data
// console.log(response)
return response
})
}

//---------------------上面是类,下面是对象,对象是从类派生出来的

let model = new Model({
data: { //data是特有属性
name: '',
number: 0,
id: ''
},
resource: 'books'
}
)

let view = new Vue({
el: '#app',
data: {
book: {
name: '',
number: 0,
id: ''
}
},
template: <div> <div> 书名:《{{book.name}}》 数量:<span id="number">{{book.number}}</span> </div> <div> <button v-on:click="addOne">加1</button> <button v-on:click="minusOne">减1</button> <button v-on:click="reset">重置</button> </div> </div> ,
created() {
model.fetch(1).then(() => {
this.book = model.data
})
},
methods: {
addOne() {
model.updata(1, { number: this.book.number + 1 })
.then(() => {
this.book = model.data
})
},
minusOne() {
model.updata(1, { number: this.book.number - 1 })
.then(() => {
this.book = model.data
})
},
reset() {
model.updata(1, { number: 0 })
.then(() => {
this.book = model.data
})
},
}
})

function fakeDate() {
let book = {
'name': 'tsl',
'number': 1,
id: 1
}

//在真正返回response之前使用
axios.interceptors.response.use(function (response) {
    let config = response.config
    let { method, url, data } = config //data是请求的data

    //     let {config:{method,url,data}}=response

    if (url === '/books/1' && method === 'get') {
        response.data = book   //响应的data
    } else if (url === '/books/1' && method === 'put') {
        //       response.data={
        //         name:'tsl',
        //         id:1,
        //         number = data.number  //data是请求的data
        //       }
        data = JSON.parse(data)  //字符串转换下
        Object.assign(book, data)  //data是请求的data
        response.data = book
    }

    return response
})

}

加一个可以增减n的


image.png
image.png

单项绑定

image.png

双向绑定在input里可以

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容

  • 哈哈哈,太开心了,正二八经的第一个漆甲。皮叔32037kills,大概3769把。还差一个项链,加油
    这次可以咕咕阅读 499评论 0 0
  • 在漆黑的夜里我用一样漆黑的眼睛来寻找漆黑中的光明哪怕是一丝灯光又或是莹虫的忽亮还是星星的闪烁这已够了,够了只要有光...
    西门可情阅读 192评论 0 1
  • 小子凌晨做噩梦,嚷着“妈妈不要去玉环”。 醒来还强调一次,懒妈安慰:“妈妈和宝贝一起,不去玉环”。 小子想了下,又...
    欧元小姨阅读 137评论 1 1