zoukankan      html  css  js  c++  java
  • Vue简易博客总结

    项目结构:

    首先,编写博客的导航栏组件BlogHeader.vue:

     1 <template>
     2     <nav>
     3         <ul>
     4             <li>
     5                 <router-link to="/" exact>博客</router-link>
     6                 <router-link to="/add" exact>写博客</router-link>
     7             </li>
     8         </ul>
     9     </nav>
    10 
    11 </template>
    12 
    13 <script>
    14     export default{
    15         name:"blog-header"
    16     }
    17 
    18 </script>
    19 
    20 <style scoped>
    21     ul{
    22         list-style-type: none;
    23         text-align: center;
    24         margin: 0px;
    25     }
    26 
    27     li{
    28         display: inline-block;
    29         margin: 0 10px;
    30     }
    31 
    32     a{
    33         color: #fff;
    34         text-decoration: none;
    35         padding: 12px;
    36         border-radius: 5px;
    37     }
    38 
    39     nav{
    40         background: crimson;
    41         padding: 30px 0;
    42         margin:10px;
    43     }
    44 
    45     .router-link-active{
    46         background: rgba(255,255,255,0.8);
    47         color: #444;
    48     }
    49 
    50 </style>
    View Code

    如图所示:


    然后,编写展示博客的组件showBlog.vue

    用到的知识点有axios访问api:更多axios知识点请访问:https://www.npmjs.com/package/axios

     1 new Vue({
     2   el: '#app',
     3   data () {
     4     return {
     5       info: null
     6     }
     7   },
     8   created () {
     9     axios
    10       .get('https://api.coindesk.com/v1/bpi/currentprice.json')
    11       .then(response => (this.info = response))
    12   }
    13 })
    14 <div id="app">
    15   {{ info }}
    16 </div>
    View Code

    错误处理:

     1 <!--
     2 很多时候我们可能并没有从 API 获取想要的数据。这可能是由于很多种因素引起的,比如 axios 调用可能由于多种原因而失败,包括但不限于:
     3 API 不工作了;
     4 请求发错了;
     5 API 没有按我们预期的格式返回信息。
     6 -->
     7 axios
     8   .get('https://api.coindesk.com/v1/bpi/currentprice.json')
     9   .then(response => (this.info = response.data.bpi))
    10   .catch(error => console.log(error))
    View Code

    自定义指令,详见:https://cn.vuejs.org/v2/guide/custom-directive.html

     1 // 注册一个全局自定义指令 `v-focus`
     2 Vue.directive('focus', {
     3   // 当被绑定的元素插入到 DOM 中时……
     4   inserted: function (el) {
     5     // 聚焦元素
     6     el.focus()
     7   }
     8 })
     9 
    10 //如果想注册局部指令,组件中也接受一个 directives 的选项:
    11 directives: {
    12   focus: {
    13     // 指令的定义
    14     inserted: function (el) {
    15       el.focus()
    16     }
    17   }
    18 }
    19 
    20 //然后你可以在模板中任何元素上使用新的 v-focus 属性,如下:
    21 <input v-focus>
    View Code

    过滤器:https://cn.vuejs.org/v2/guide/filters.html

    showBlog.vue代码:

     1 <template>
     2   <div id="show-blog" v-theme:column="'narrow'">
     3     <h1 style="text-align: center;padding: 15px 0px 0px 0px">博客总览</h1>
     4     <input type="text" v-model="serach" placeholder="搜索" />
     5     <div v-for="blog in blogs" class="single-blog">
     6         <!-- | pipe管道过滤器的标识 -->
     7         <router-link v-bind:to="'/blog/' + blog.id"><h2 v-rainbow>{{blog.name}} </h2></router-link>
     8         <article>
     9             <!-- {{blog.price | snippet}} -->
    10              {{blog.detail | snippet}}
    11         </article>
    12     </div>
    13   </div>
    14 </template>
    15 
    16 <script>
    17 export default {
    18   name: 'show-blog',
    19   data(){
    20       return{
    21           blogs:[
    22 
    23           ],
    24           serach:''
    25       }
    26   },
    27   created(){
    28       /*this.$http.get('http://jsonplaceholder.typicode.com/posts').then(function(data){
    29               this.blogs = data.body.slice(0,30);
    30       })*/
    31       this.$axios.get('/api/items').then((data)=>{
    32   //        console.log(data.body);
    33          this.blogs = data.data;
    34 
    35       })
    36   },
    37 /*  computed:{
    38       filteredBlogs:function(){
    39           return this.blogs.filter((blog)=>{
    40               return blog.title.match(this.serach);
    41           })
    42       }*/
    43  // },
    44   //过滤器局部实现方法
    45   filters:{
    46       "to-uppercase":function(data){
    47           return data.toUpperCase();
    48       }
    49   },
    50   //自定义指令的局部实现方式
    51   directive:{
    52 
    53   }
    54 }
    55 </script>
    56 
    57 <style>
    58 
    59 #show-blog{
    60     max-width: 800px;
    61     margin:0 auto;
    62 }
    63 
    64 .single-blog{
    65     padding: 20px;
    66     margin: 20px auto;
    67     box-sizing: border-box;
    68     background: #ccc;
    69     border:1px dotted #aaa;
    70 }
    71 
    72 #show-blog a{
    73     color: #444;
    74     text-decoration: none;
    75 }
    76 
    77 input[type="text"]{
    78     padding: 8px;
    79     width:100%;
    80     box-shadow: border-box;
    81 }
    82 </style>
    View Code

     界面如图所示:

    博客详情页代码:

     1 <template>
     2     <div id="single-blog">
     3         <h1>{{blog.title}}</h1>
     4         <article>{{blog.body}}</article>
     5     </div>
     6 </template>
     7 
     8 <script>
     9     export default{
    10         name:"singleblog",
    11         data(){
    12             return{
    13                 id:this.$route.params.id,
    14                 blog:{}
    15             }
    16         },
    17         created(){
    18             this.$http.get('http://jsonplaceholder.typicode.com/posts/'+this.id).then(function(data){
    19                   this.blog = data.body;
    20               })
    21         }
    22     }
    23 
    24 </script>
    25 
    26 <style>
    27     #single-blog{
    28         max-width: 800px;
    29         margin: 0 auto;
    30         padding: 20px;
    31         background: #eee;
    32         border:1px dotted #aaa;
    33     }
    34 </style>
    View Code


    最后编写添加博客页代码:

      1 <template>
      2   <div id="add-blog" >
      3   <h2>添加博客</h2>
      4   <form action="" v-if="!submmited">
      5       <label>博客标题</label>
      6       <input type="text" v-model="blog.title" required="" />
      7       <label for="">博客内容</label>
      8       <textarea name="" id="" cols="30" rows="10" v-model="blog.content"></textarea>
      9     <div id="checkboxes">
     10       <label for="">Vue.js</label>
     11       <input type="checkbox" value="Vue.js" v-model="blog.categories" />
     12       <label for="">Node.js</label>
     13       <input type="checkbox" value="Node.js"  v-model="blog.categories"/>
     14       <label for="">React.js</label>
     15       <input type="checkbox" value="React.js"  v-model="blog.categories"/>
     16       <label for="">Angular</label>
     17       <input type="checkbox" value="Angular"  v-model="blog.categories"/>
     18       <label for="">作者:</label>
     19       <select v-model="blog.author">
     20         <option v-for="author in authors">
     21           {{author}}
     22         </option>
     23       </select>
     24       <button v-on:click.prevent="post">添加博客</button>
     25     </div>
     26   </form>
     27 
     28   <div>
     29     <h3 v-if="submmited">您的博客发布成功!</h3>
     30   </div>
     31   <hr>
     32   <div id="preview">
     33       <h3>博客总览</h3>
     34       <p>博客标题:{{blog.title}}</p>
     35       <p>博客内容:{{blog.content}}</p>
     36     <p>博客分类:</p>
     37     <ul>
     38       <li v-for="category in blog.categories">
     39         {{category}}
     40       </li>
     41     </ul>
     42     <p>作者:{{blog.author}}</p>
     43   </div>
     44   </div>
     45 </template>
     46 
     47 <script>
     48   import axios from 'axios' 
     49 export default {
     50   name: 'addBlog',
     51   data () {
     52     return {
     53         blog:{
     54             title:"",
     55             content:"",
     56         categories:[],
     57         author:""
     58         },
     59       authors:["lianmin","wnagdalu","zhoujielun"],
     60       submmited:false
     61     }
     62   },
     63   methods:{
     64     post:function(){
     65       /*this.$http.post("http://jsonplaceholder.typicode.com/posts",{
     66         title:this.blog.title,
     67         body:this.blog.content,
     68         userId:1
     69       }).then(function(data){
     70         console.log(data.body);
     71         this.submmited=true;
     72       });*/
     73       var _this = this;
     74      axios.post("http://jsonplaceholder.typicode.com/posts",{
     75         title:this.blog.title,
     76         body:this.blog.content,
     77         userId:1
     78       }).then((data)=>{
     79         console.log(data.body);
     80         _this.submmited=true;
     81       });
     82     }
     83   }
     84 }
     85 </script>
     86 
     87 <!-- Add "scoped" attribute to limit CSS to this component only -->
     88 <style scoped>
     89 #add-blog *{
     90   box-sizing:border-box ;
     91 }
     92 #add-blog{
     93   margin: 20px auto;
     94   max-width: 600px;
     95   padding: 20px;
     96 }
     97 
     98 label{
     99   display: block;
    100   margin:20px 0 10px;
    101 }
    102 
    103 input[type="text"],textarea,select{
    104   display: block;
    105   width: 100%;
    106   padding: 8px;
    107 }
    108 
    109 textarea{
    110   height: 200px;
    111 }
    112 
    113 #checkboxes label{
    114   display: inline-block;
    115   margin-top: 0;
    116 }
    117 
    118 #checkboxes input{
    119   display: inline-block;
    120   margin-right: 10px;
    121 }
    122 
    123 button{
    124   display: block;
    125   margin:20px 0;
    126   background: crimson;
    127   color: #fff;
    128   border: 0;
    129   padding: 14px;
    130   border-radius: 4px;
    131   font-size: 18px;
    132   cursor: pointer;
    133 }
    134 
    135 #preview{
    136   padding: 10px 20px;
    137   border:1px dotted #ccc;
    138   margin:30px 0;
    139 }
    140 
    141 h3{
    142   margin-top: 10px;
    143 }
    144 </style>
    View Code


    最后进行路由表的相关配置:

     1 import Vue from 'vue'
     2 import Router from 'vue-router'
     3 import HelloWorld from '@/components/HelloWorld'
     4 import showblog from '@/components/showblog'
     5 import addblog from '@/components/AddBlog'
     6 import singleBlog from '@/components/SingleBlog'
     7 
     8 Vue.use(Router)
     9 
    10 export default new Router({
    11   routes: [
    12     {
    13       path: '/',
    14       name: 'showblog',
    15       component: showblog
    16     },
    17     {
    18       path: '/add',
    19       name: 'addblog',
    20       component: addblog
    21     },
    22     {
    23       path: '/blog/:id', /*路由参数*/
    24       component: singleBlog
    25     },
    26   ],
    27    mode:"history"  //不用显示#号
    28 })
    View Code

    还要在App.vue中进行一些编写,主要是使用写好的路由:

     1 <template>
     2   <div id="app">
     3       <blog-header></blog-header>
     4    <!--  <add-blog></add-blog> -->
     5     <!-- <show-blog></show-blog> -->
     6     <router-view/>
     7   </div>
     8 </template>
     9 
    10 <script>
    11 import AddBlog from './components/AddBlog'
    12 import showblog from './components/showblog'
    13 import blogHeader from './components/BlogHeader'
    14 export default {
    15   name: 'App',
    16   components: {
    17     'add-blog': AddBlog,
    18     'show-blog':showblog,
    19     "blog-header":blogHeader
    20   }
    21 }
    22 </script>
    23 
    24 <style>
    25 
    26 </style>
    View Code

    另外,还可以在main.js中进行一些全局属性的配置:

     1 // The Vue build version to load with the `import` command
     2 // (runtime-only or standalone) has been set in webpack.base.conf with an alias.
     3 import Vue from 'vue'
     4 import App from './App'
     5 import router from './router'
     6 import VueResource from 'vue-resource'
     7 import axios from 'axios'
     8 
     9 
    10 Vue.prototype.$axios = axios
    11 //全局配置axios
    12 //axios.defaults.baseURL='http://localhost:8080'
    13 //请求头配置
    14 /*
    15 axios.defaults.headers.common['Authorization'] = 'Token'
    16 axios.defaults.headers.post['Content-type'] = ''
    17 axios.defaults.headers.get['Accepts']='application/json'
    18 */
    19 Vue.config.productionTip = false
    20 Vue.use(VueResource)
    21 
    22 //自定义指令
    23 Vue.directive('rainbow',{
    24     bind(el,binding,vnode){
    25         el.style.color = "#" + Math.random().toString(16).slice(2,8)
    26     }
    27 })
    28 
    29 Vue.directive('theme',{
    30     bind(el,binding,vnode){
    31         if(binding.value=='wide'){
    32             el.style.maxWidth = "1260px"
    33         }else if(binding.value='narrow'){
    34             el.style.maxWidth="600px"
    35         }
    36 
    37         if(binding.arg=='column'){
    38             el.style.background = "#F4A460";
    39             el.style.margin = "10px auto"
    40         }
    41     }
    42 })
    43 
    44 //自定义过滤器  全局的实现方式
    45 /*Vue.filter("to-uppercase",function(value){
    46     return value.toUpperCase();
    47 })*/
    48 
    49 Vue.filter("snippet",function(value){
    50     return value.slice(0,10) + "...";
    51 })
    52 /* eslint-disable no-new */
    53 new Vue({
    54   el: '#app',
    55   router,
    56   components: { App },
    57   template: '<App/>'
    58 })
    View Code

  • 相关阅读:
    谈谈WPF中的CollectionView与CollectionViewSource (1)
    XPath语法备忘
    WPF中,如何将绑定源设置到单件实例
    避免让WPF资源字典变得杂乱臃肿
    自定义WPF面板
    [WPF疑难] 继承自定义窗口
    WPF高手:站出来,Show出来
    Windows Presentation Foundation Tools and Controls
    放送Ifttt邀请四枚
    使用Amazon S3 Service时报403错误的解决方法
  • 原文地址:https://www.cnblogs.com/ustc-anmin/p/10588620.html
Copyright © 2011-2022 走看看