MrShi
2025-03-26 6f1203d1cf895efc473fade3ab049fc30d92c618
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
<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.js';
    /**
     * CodeInput 栅格系统的列 
     * @description 该组件一般用于Layout 布局 通过基础的 12 分栏,迅速简便地创建布局
     * @tutorial https://www.uviewui.com/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: [uni.$u.mpMixin, uni.$u.mixin, props],
        data() {
            return {
                width: 0,
                parentData: {
                    gutter: 0
                },
                gridNum: 12
            }
        },
        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: uni.$u.addUnit(uni.$u.getPx(this.parentData.gutter)/2),
                    paddingRight: uni.$u.addUnit(uni.$u.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: uni.$u.addUnit(Math.floor(this.width / this.gridNum * Number(this.span))),
                    marginLeft: uni.$u.addUnit(Math.floor(this.width / this.gridNum * Number(this.offset))),
                    // #endif
                }
                return uni.$u.deepMerge(style, uni.$u.addStyle(this.customStyle))
            }
        },
        mounted() {
            this.init()
        },
        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>