The thing that really stands out to me about Vue3, apart from a bunch of piecematic optimizations, is the composition component and its corresponding functional writing.
First, let’s look at the use of the Composition component. The new hook setup() is used to register the function and return it
Specific writing
export default {
setup() {
A reactive object is generated by a ref and its value is assigned to trigger a dependency collection access
const count = ref(0)
// Equivalent to computed attributes
const plusOne = computed(() = > count.value + 1)
// is the same as method
const increment = () = > { count.value++ }
//
watch(() = > count.value * 2.val= > {
console.log(`count * 2 is ${val}`)})// Specifies the mounted life cycle
onMounted(() = > {
console.log(`mounted`)})// Returns the options that need to be registered in the page
return {
count,
plusOne,
increment
}
}
}
Copy the code
Writing a composition component is not difficult for a skilled framer, but it takes a little digging into the new features to feel comfortable using them.
To understand a new feature first, look at why it’s there and how it changes the current development ecosystem.
In VUe2, the development is called optional development. The same function logic may need to be written in data option and changed in methods, and the life cycle may be written in two sentences, which may not feel any difference during development. With headphones listening to the song, the face roll keyboard business code casually write.
But! Yard brick for a while, maintenance crematorium. Especially if you’re not familiar with the business of taking on someone else’s code. I remember taking over my last colleague’s code when I first joined the company. A function block that writes the component’s options in bits and pieces in seven or eight files and references them back and forth. A small function, half an hour just didn’t find the full code block.
That project took a huge toll on me, and ever since. I have been thinking about how to be block-oriented development, packaging an independent complete business function into a separate code block, so that it is easy to maintain. At the same time, encapsulate the same logic to encapsulate references, which greatly reduces the workload, reduces copy and paste, as well as errors in the process of copy and paste.
I wasn’t the only one, and Yuxi shared my confusion,
The separation of options obscures the underlying logical concerns. In addition, when working on a single logical concern, we have to constantly “jump” around option blocks for the relevant code.
As shown in the figure below, in later maintenance of large VUE projects, it often takes only 3 minutes to change the code and 30 minutes to find the code to change.
So our goal is to make it like this, to have the same color function code next to each other. This is convenient for maintenance and logical extractionBefore composition, implement code modularization extraction and insertion, usually withmixin. But while it’s good to find mixins in my vast collection of business code bricks, it’s also pretty damn fucked up from a business development experience alone.
1. The problem of the same name: Those who have used mixin to extract business should have a deep experience of this, which is very unfriendly to me as a patient with naming difficulties.
2. The problem of invisible dependency: When the project is small, this problem is actually not felt. Once the project is large, dozens of mixin modules are mixed into a page, and then data parameters and method are called each other before each mixin. But we know that mixins are referenced horizontally on the same page. That is, who uses whose parameters, without showing references at all. In fact, there is a big hidden danger in it. Cause you to finish writing things, late is not quite dare to move. One day your colleague changes a data attribute and a whole mixin module crashes.
Since mixins work but do have their drawbacks, let’s see how Composition can improve on them.
First of all, when you use composition you can do everything that mixin can do. And because it is a display reference, mixins don’t have the trouble of naming names and implicit dependencies. There is also the mixin no matter how to extract it is still option configuration programming in the final analysis. With the spread of business, it will also face too broken files with unclear semantics, too large and too bloated blocks.
But composition is functional programming, and its benefits are:
1. First of all, the same business no longer needs to suffer from the separation of data and method. Business blocks can be written together and then separated.
Secondly, it is much more convenient to pass and return parameters. Such as:
A lot of my business is a classic management page of one form above and one form below. So when the business scenarios are similar, I usually pull out the business scenarios, but each page table has a different interface to get the data, and if you use mixins, you have to define another property in the data of each page component. Similarly, if the business becomes bloated, as more and more mixin modules and interfaces are requested, more and more attributes will be defined in data. And when other pages refer to the module, the data attribute name in the page has to be consistent, so it is rigid and rigid.
With composition component extraction, we simply pass in the request interface name as a parameter to the component, instead of defining a data property in the page component
Get the page data component
import { ref, onMounted } from 'vue';
import api from '@/config/api/jsonConfig.js'
import commonData from '@/config/general/commonData.js'
/* Fcn: request table data interface name autoRequest: whether to automatically load data after the mount is complete */
export default (Fcn,autoRequest) => {
// Table binding data
const tableData = ref([])
// Table page number data
const paginationOpt = ref({
pageNumber: 1.pageSize: 20.totalRecords: 0
})
// Get table data
const getTableData = async (params) => {
let d = await api[Fcn]({ params: { ...params, organizationUID: commonData.currentOrg.organizationUID } })
if(! d)return
tableData.value =d.data.pagination? d.data.records:d.data
paginationOpt.value = d.data.pagination
}
/ / page
const handleSizeChange = (v) = > {
paginationOpt.value.pageSize = v
getTableData(paginationOpt.value)
}
const handleCurrentChange = (v) = > {
paginationOpt.value.pageNumber = v
getTableData(paginationOpt.value)
}
// Mount the requested data
onMounted(() = > {
if(autoRequest) getTableData()
})
return { tableData, paginationOpt, getTableData, handleSizeChange, handleCurrentChange }
}
Copy the code
Page component: Just pass in the method name
export default defineComponent({
setup() {
return {
...useTable("examSourcegetList".true),}; }});Copy the code
3. Pull out the hierarchy for better control. In previous mixins we mixed a module into the page, which meant that it accepted all the configuration of the module. So if pages A and B need all of the business options of the module, page C only needs some data and methods. It’s very difficult to deal with. In composition components, however, the component needs a return in setup() to register with the page component, so we can safely pull out by business function blocks. In C pages that do not need all business blocks, we can return only the property methods it needs.
For example, I put all the binding data of the select drop-down fill and the method of getting the data in one component for easy management. On different pages I just need to return the drop-down population for their business needs. You can’t do that in mixins
// Drop down fill
import { ref } from 'vue'
import api from '@/config/api/jsonConfig.js'
export default() = > {/ / RMCO institutions
let RMCOorgList = ref([])
// Sign service center
let ServiceCenterList = ref([])
// Department list drop-down box
let DepartmentList = ref([])
// Check the type
let ServiceSectList = ref([])
// Check the status
let ExamStatusList = ref([])
// Type of visit
let PatientClassList = ref([])
// Upload rules
let UploadRuleList = ref([])
/ / the data source
let ExamSourceList = ref([])
// Image configuration
let PACSSourceList = ref([])
const getRMCOrganizationList = async (params) => {
...
}
const getSignedServiceCenterList = async (params) => {
...
}
const getDepartmentList = async (params) => {
...
}
const getServiceSectList = async (params) => {
...
}
const getExamStatusList = async(params) => { ... }...return{ RMCOorgList, ServiceCenterList, DepartmentList, ServiceSectList, ExamStatusList, PatientClassList, gainTypeList, dbTypeList, .... }}Copy the code
The page component simply introduces the option to return the corresponding data
import userDropOptions from "@/components/composition/dropOptions";
setup(props,{emit}) {
// Get the drop-down correlation
const { depGainTypeList, dbTypeList } = userDropOptions();
}
return {
depGainTypeList,
dbTypeList,
};
Copy the code