-
zhangwei
2025-03-24 2d43a1df3f5ba42710e6d21c27d1e13bdb8dfd56
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
<template>
    <view
        class="u-col"
        ref="u-col"
        :class="[
            'u-col-' + span
        ]"
        :style="[colStyle]"
        @tap="clickHandler"
    >
        <slot></slot>
    </view>
</template>
 
<script>
    import { props } from './props';
    import { mpMixin } from '../../libs/mixin/mpMixin';
    import { mixin } from '../../libs/mixin/mixin';
    import { addStyle, addUnit, deepMerge, getPx } from '../../libs/function/index';
    /**
     * CodeInput 栅格系统的列 
     * @description 该组件一般用于Layout 布局 通过基础的 12 分栏,迅速简便地创建布局
     * @tutorial https://ijry.github.io/uview-plus/components/Layout.html
     * @property {String | Number}    span        栅格占据的列数,总12等份 (默认 12 ) 
     * @property {String | Number}    offset        分栏左边偏移,计算方式与span相同 (默认 0 ) 
     * @property {String}            justify        水平排列方式,可选值为`start`(或`flex-start`)、`end`(或`flex-end`)、`center`、`around`(或`space-around`)、`between`(或`space-between`)  (默认 'start' ) 
     * @property {String}            align        垂直对齐方式,可选值为top、center、bottom、stretch (默认 'stretch' ) 
     * @property {String}            textAlign    文字水平对齐方式 (默认 'left' ) 
     * @property {Object}            customStyle    定义需要用到的外部样式
     * @event {Function}    click    col被点击,会阻止事件冒泡到row
     * @example     <u-col  span="3" offset="3" > <view class="demo-layout bg-purple"></view> </u-col>
     */
    export default {
        name: 'u-col',
        mixins: [mpMixin, mixin, props],
        data() {
            return {
                width: 0,
                parentData: {
                    gutter: 0
                },
                gridNum: 12
            }
        },
        //  微信小程序中 options 选项
        options: {
            virtualHost: true // 将自定义节点设置成虚拟的,更加接近Vue组件的表现。我们不希望自定义组件的这个节点本身可以设置样式、响应 flex 布局等
        },
        computed: {
            uJustify() {
                if (this.justify == 'end' || this.justify == 'start') return 'flex-' + this.justify
                else if (this.justify == 'around' || this.justify == 'between') return 'space-' + this.justify
                else return this.justify
            },
            uAlignItem() {
                if (this.align == 'top') return 'flex-start'
                if (this.align == 'bottom') return 'flex-end'
                else return this.align
            },
            colStyle() {
                const style = {
                    // 这里写成"padding: 0 10px"的形式是因为nvue的需要
                    paddingLeft: addUnit(getPx(this.parentData.gutter)/2),
                    paddingRight: addUnit(getPx(this.parentData.gutter)/2),
                    alignItems: this.uAlignItem,
                    justifyContent: this.uJustify,
                    textAlign: this.textAlign,
                    // #ifndef APP-NVUE
                    // 在非nvue上,使用百分比形式
                    flex: `0 0 ${100 / this.gridNum * this.span}%`,
                    marginLeft: 100 / 12 * this.offset + '%',
                    // #endif
                    // #ifdef APP-NVUE
                    // 在nvue上,由于无法使用百分比单位,这里需要获取父组件的宽度,再计算得出该有对应的百分比尺寸
                    width: addUnit(Math.floor(this.width / this.gridNum * Number(this.span))),
                    marginLeft: addUnit(Math.floor(this.width / this.gridNum * Number(this.offset))),
                    // #endif
                }
                return deepMerge(style, addStyle(this.customStyle))
            }
        },
        mounted() {
            this.init()
        },
        emits: ["click"],
        methods: {
            async init() {
                // 支付宝小程序不支持provide/inject,所以使用这个方法获取整个父组件,在created定义,避免循环引用
                this.updateParentData()
                this.width = await this.parent.getComponentWidth()
            },
            updateParentData() {
                this.getParentData('u-row')
            },
            clickHandler(e) {
                this.$emit('click');
            }
        },
    }
</script>
 
<style lang="scss" scoped>
    @import "../../libs/css/components.scss";
 
    .u-col {
        padding: 0;
        /* #ifndef APP-NVUE */
        box-sizing:border-box;
        /* #endif */
        /* #ifdef MP */
        display: block;
        /* #endif */
    }
 
    // nvue下百分比无效
    /* #ifndef APP-NVUE */
    .u-col-0 {
        width: 0;
    }
 
    .u-col-1 {
        width: calc(100%/12);
    }
 
    .u-col-2 {
        width: calc(100%/12 * 2);
    }
 
    .u-col-3 {
        width: calc(100%/12 * 3);
    }
 
    .u-col-4 {
        width: calc(100%/12 * 4);
    }
 
    .u-col-5 {
        width: calc(100%/12 * 5);
    }
 
    .u-col-6 {
        width: calc(100%/12 * 6);
    }
 
    .u-col-7 {
        width: calc(100%/12 * 7);
    }
 
    .u-col-8 {
        width: calc(100%/12 * 8);
    }
 
    .u-col-9 {
        width: calc(100%/12 * 9);
    }
 
    .u-col-10 {
        width: calc(100%/12 * 10);
    }
 
    .u-col-11 {
        width: calc(100%/12 * 11);
    }
 
    .u-col-12 {
        width: calc(100%/12 * 12);
    }
 
    /* #endif */
</style>