preface

This article is just a summary of my experience in my daily practice. It is a little bit of skill, not very clever technology. If it is helpful to you, I will be honored.

This article does not cover the use of rare apis, etc., and most of it is based on some practice with VUE. Due to suspected speculation, may bring some non-standard side effects, please use according to the requirements of the project.

  1. Multiple pages are used to the method, invue.prototypeIt would come in handy

    I did a stupid thing when I first started vUE, because I wrapped an asynchronous request interface POST, put it in a post.js file, and introduced it in every page that needed to use asynchronous requests

    import port from './xxxx/xxxx/post'
    Copy the code

    If that’s all, we can write a page and copy it later, ensuring that each page has the above statement. But what if each file resides at a different directory level?

    // If it is normal
    import port from '.. /xxxx/xxxx/post'
    // If the directory is deepened one level, it looks like this
    import port from '.. /.. /xxxx/xxxx/post'
    // Make it one level deeper
    import port from '.. /.. /.. /xxxx/xxxx/post'
    Copy the code

    Of course, at this point, we could use the @/ XXXX /post alias, but we still want to make sure that every page references. How easy is it to use Vue. prototype? First, you need to set the following parameters in the vue entry file (vue-cli-generated projects default to/SRC /main.js)

     import port from './xxxx/xxxx/post'
    
     vue.prototype.$post = post   
    Copy the code

    This way, we can use the this.post() method in all vue components (pages), just like vue’s own son

    When attaching methods to Prototype, use a $prefix to avoid collisions with other variables

    Til again: Don’t mount too many methods to Prototype, just a few that are used very frequently

  2. The data to be responded to is set when the interface data is obtained

    We have very often encounter such a situation, in the circulation list, we need to give a list item control display properties, such as whether to delete, is chosen, and so on, and then end interface generally will not return to the field, because it belongs to pure front-end display, with back-end do not have what relation, such as the back-end data to are as follows

    [{name: 'abc'.age: 18},
      {name: 'def'.age: 20},
      {name: 'ghi'.age: 22},]Copy the code

    Let’s assume that the above data is a list of students

    Then we need to render the list, behind every show a select button, if the user box, this button is green, by default, this button is gray at this time, the table is not satisfy this rendering condition data, and if we at the time of user box, to add the data, then the normal practice is not response in time.

    We can solve this problem by marking each item of the array with a check mark when we get the data. Let’s say we get res.list

    res.list.map(item= >{item. IsTicked =false
    })
    Copy the code

    The principle is that vUE cannot respond to non-existent attributes, so when we get the data, we add the required attributes first, and then assign the value to Data, so that when data receives the data, the attribute already exists, so it will respond. Of course there are other ways to do it. However, for an OCD, I still prefer this approach

  3. Encapsulation global basedpromiseThe asynchronous request method of

    Looking at the source code of many projects, I found that the majority of asynchronous requests use methods like AXIos directly, as follows

    axios({
      method: 'post'.url: '/user/12345'.data: {
        firstName: 'Fred'.lastName: 'Flintstone'
      }
    })
     .then(function (response) {
        console.log(response);
      })
      .catch(function (error) {
        console.log(error);
      });
    Copy the code

    If you have cross-domain or need to set HTTP and so on, you need to add more configurations, and these configurations, for the same project, are basically the same, only the URL and parameters are different, so why don’t I encapsulate it as a method?

    function post (url,param) {
        return axios({
          method: 'post'.url: url,
          data: param ... Other configurations of AXIOS})}Copy the code

    I used a layer of promise that was too much for a simple need. Pointed out that

    Combined with the first point, we can use this in any VUE instance

    let param = {
      firstName: 'Fred'.lastName: 'Flintstone'
    }
    this.post('/user/12345',param)
    .then(...)
    .catch(...)
    Copy the code

    Is it much simpler than the original? You can also use this if your project supports async await

    let param = {
      firstName: 'Fred'.lastName: 'Flintstone'
    }
    let res  = await this.post('/user/12345',param)
    console.log(res) // res is the data returned asynchronously
    
    Copy the code

    The tip: await keyword must be used inside async decorated functions

  4. If you think that sometimes you really need parent and child components to share a value, try passing a reference type

    There are several ways to pass values to vue parent components, and I won’t list them here, but today we’re going to take advantage of javascript’s reference type features to achieve another purpose of passing values

    Let’s say the parent component needs to send 3 values to the child component, and then immediately respond to the child component’s changes. We would normally emit events via this.$emit, and then the parent component would listen for the corresponding events. If you send too much data, it’s exhausting. We might as well pass the data package into an object/array and then pass it to the child component

    <subComponent :subData="subData"></subComponent>
    Copy the code
    data () {
      return {
        subData: {
          filed1: 'field1'.filed2: 'field2'.filed3: 'field3'.filed4: 'field4'.filed5: 'field5',}}}Copy the code

    This way, if we change the content of subData in the child component, the parent component can respond directly without this.$emit or vuex, and if there are other sibling components, the subData in the sibling component can also respond as long as the sibling component is bound to the subData

    Tip: First of all, I personally feel that this is a bit out of the norm. If there is not a lot of data, use this.$emit.

  5. The parameters of the asynchronous request are indataInside the structure, with an object wrapped, will be a lot easier

    For those of you who have worked on erP-type systems, you have come across a scenario where you have a list with N filter criteria, and this is how we bind it

     <input type="text" v-model="field1">
     <input type="text" v-model="field2">
     <input type="text" v-model="field3">.<input type="text" v-model="fieldn">
    Copy the code
    data () {
     return {
       field1: 'value1'.field2: 'value2'.field3: 'value3'. fieldn:'valuen'}}Copy the code

    When submitting the data, it looks like this:

     var param = {
       backend_field1: this.field1,
       backend_field2: this.field2,
       backend_field3: this.field3,
       ...
       backend_fieldn: this.fieldn
     }
     this.post(url,param)
    Copy the code

    As you can see, every time you submit an interface, you have to construct parameters, and it’s easy to miss. Instead, go to the interface documentation and look at the name of the field that you need at the back end

        <input type="text" v-model="queryParam.backend_field1">
        <input type="text" v-model="queryParam.backend_field2">
        <input type="text" v-model="queryParam.backend_field3">.<input type="text" v-model="queryParam.backend_fieldn">``` ```javascript data () { return { queryParam:{ backend_field1: 'value1' backend_field2: 'value2' backend_field3: 'value3' ... backend_fieldn: 'valuen'}}} ' 'then commit data like this: Javascript this.post(URL,this.queryParam) yes, there are limitations to doing this, for example if you share the same data in two places, for example if the front-end component binds to an array and you need to submit two strings to the back end (example: 'Element UI' time control), but it's easier to deal with some special issues than to build a new parameter, isn't it?Copy the code
  6. dataWhen there is a lot of data in it, add a note to each data so that you can look back clearly

    Continue on a bit, the data there are many data, could you write is quite clear, after all, is what you wrote, but after ten days half a month, or other people look at your code, and believe me, either you own, or others, are confused about (memory than anyone except), so we might as well add a note to each data behind

    data () {
     return {
       field1: 'value1'.// control the display of XXX
       field2: 'value2'.// Page loading status
       field3: [],        // List of users. fieldn:'valuen'   // XXXXXXXX}}Copy the code
  7. Logically complex content, as far as possible into components

    Suppose we have a scenario like this:

    <div>
       <div>Name: {{user1. Name}}</div>
       <div>Gender: {{user1. Sex}}</div>
       <div>Age: {{user1. Age}}</div>. Omitted 999 fields...<div>His next door neighbor's aunt's dog name: {{user1.petName}}</div>
    </div>
    <--Of course, we're not so stupid as not to use the displayv-forLet's assume that this case doesn't workv-for -->
    <div>
        <div>Name: {{user2. Name}}</div>
        <div>Gender: {{user2. Sex}}</div>
        <div>Age: {{user2. Age}}</div>. Omitted 999 fields...<div>His next door neighbor's aunt's dog name: {{user2.petname}}</div>
    </div>
    Copy the code

    In this case, we might as well extract [user] code into a component: assume the following code in comUserinfo.vue

    < the template > < div > < div > name: {{user. The name}} < / div > < div > gender: {{user. Sex}} < / div > < div > age: {{user. Age}} < / div >... Omitted 999 fields... <div> The name of his next door neighbor's aunt's dog: {{user.petName}}</div> </div> </template> <script > export default { props:{ user:{ type:Object, default: () => {} } } } </script>Copy the code

    The original page can then be changed to look like this (omitting the import and registration components, assuming the registration name is comUserInfo) :

    <comUserInfo :user="user1"/>
    <comUserInfo :user="user2"/>
    Copy the code

    Isn’t that much clearer? Without looking at the comment, you can guess that there are two user information modules. Another advantage of this is that when an error occurs, you can more easily locate the error.

  8. If you only change a parent value in a child component, try it$emit('input')Will change directlyv-model

    Our normal parent-child communication is that the parent component passes props to the child component, and the child component uses this.$emit(‘eventName’,value) to notify the parent component of the method bound to @eventName to do the corresponding processing. There is an exception here, however. By default, VUE listens for component input events and assigns values from child components to the values currently bound to the V-Model

    Normal usage – Parent component

    <template>
      <subComponent :data="param" @dataChange="dataChangeHandler"></subComponent>
    </template>
    
    <script >
      export default {
        data () {
          return {
            param:'xxxxxx'
          }
        },
        methods:{
          dataChangeHandler (newParam) {
            this.param = newParam
          }
        }
      }
    </script>
    Copy the code

    Normal usage – Child components

    <script >
      export default {
        methods:{
          updateData (newParam) {
            this.$emit('dataChange',newParam)
          }
        }
      }
    </script>
    Copy the code

    Use the defaultinputEvent – Parent component

    <template>
      <subComponent  v-model="param"></subComponent>
    </template>
    Copy the code

    Use the defaultinputEvent – Child component

    <script >
      export default {
        methods:{
          updateData (newParam) {
            this.$emit('input',newParam)
          }
        }
      }
    </script>
    Copy the code

    This way, we can save a line of processing code from the parent component, and VUE will take care of it for you automatically

    Tip: This approach only works when a single value is changed, and the child component simply passes the value to the parent, with no additional action (such as updating the list) required.

    Add a this.$emit(‘update:fidldName’,value) method (thanks @riyue! It is pointed out that:

    The parent component

        <subComponent field1.sync="param1" field2.sync="param2"></subComponent>
    Copy the code

    Child components

    <script >
      export default {
        methods:{
          updateData1 (newValue) {
            this.$emit('update:field1',newValue)
          },
          updateData2 (newValue) {
            this.$emit('update:field2',newValue)
          }
        }
      }
    </script>
    Copy the code

    In my opinion, this method is suitable for the situation that the data to be updated cannot be bound to the V-model, or the data to be two-way communication is more than 1 (1 can also be used, but I personally prefer input, depending on personal preference), but not many cases.

  9. conponentsOn theVue optionsThe top of the

    If you’ve ever had the experience of importing a component and then using it in a page, ok, error, why? You forgot to register components. Why do you often forget to register components? Because a normal vUE instance would look something like this:

    import xxx form 'xxx/xxx'
    export default {
      name: 'component-name',
      data () {
        return {
          / /... Depending on the complexity of the business logic, several lines are omitted}},computed: {
        / /... Depending on the complexity of the business logic, several lines are omitted
      },
      created () {
        / /... Depending on the complexity of the business logic, several lines are omitted
      },
      mounted () {
        / /... Depending on the complexity of the business logic, several lines are omitted
      },
      methods () {
        / /... Depending on the complexity of the business logic, several lines are omitted}},Copy the code

    I don’t know where people normally put the components property, but I used to put it at the bottom, and as a result, I often make this mistake.

    And then I moved the components to the first one

    import xxx form 'xxx/xxx'
    export default {
      components: {
        xxx
      },
      // omit other code
    }
    Copy the code

    From now on, mother no longer need to worry about me forgetting to register components, import and registration are in the same location, want to forget all difficult.

  10. In most cases, you don’t have too many lines of code in your life cycle, and you can wrap them up into methods and call them

    I’ve seen a lot of code, including my own, written a couple of hundred lines of code in a lifetime, like this: Created creates code that makes it hard to read and has no idea what you’re doing when a page loads. Instead, we can encapsulate that logic into methods and call them in the lifecycle:

    created () {
      // Get user information
      this.getUserInfo()
      // Obtain system information
      this.getSystemInfo()
      // Get the configuration
      this.getConfigInfo()
    },
    methods: {// Get user informationgetUserInfo () {... },// Obtain system informationgetSystemInfo () {... },// Get the configurationgetConfigInfo () {... }},Copy the code

    Is it easy to see what you’re doing when the page loads?

    Tip: This is a common rule, but I think I read it a lot, and I did it myself when I was a beginner. I hope that newcomers can avoid this problem

  11. To use lesswatchIf you think you need it in a lot of placeswatchNine times out of ten you are rightvueAPINot enough to know

    Vue itself is a data-driven framework, and changes in data can be fed back to the view in real time. If you want to control attempts based on data, normally with computed use you can solve most of the problems, and changes in views can be monitored through events like input change. To achieve the purpose of real-time monitoring, watch is rarely used, at least in the ten or so projects I have been to recently. Of course, this is not to say that Watch is definitely useless. Vue must have its reasons for providing this API, and some requirements really need to be used. It’s just that I think it should be used infrequently, and if you think you need it everywhere, THEN I think in all likelihood you should familiarize yourself with computed and vue’s other apis

The last

The Github address of this article welcomes arbitrary star,follow, and arbitrary issue

In addition, there are other front-end tutorials and components available on Github for those who are interested. Your support is my biggest motivation.