lxui-uni 0.0.2 → 0.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,6 @@
1
+ # lxui-uni
2
+
3
+ ### npm安装
4
+ ```sh
5
+ npm install lxui-uni -S
6
+ ```
@@ -0,0 +1,250 @@
1
+ <script lang="ts" setup>
2
+ /*
3
+ * @description: 头部组件
4
+ * @fileName: Header.vue
5
+ * @params
6
+ * @author: lxx
7
+ * @date: 2023-07-16 09:32:09
8
+ * @version: V1.0.2
9
+ */
10
+ import { goToPage } from '../../util'
11
+ import { type CSSProperties, computed, onMounted, ref, watchEffect } from 'vue'
12
+
13
+ type headerInt = {
14
+ // 头部高度 默认为44px (微信小程序不可用)
15
+ headerHeight?: number
16
+ // 是否显示左侧内容
17
+ leftIconShow?: boolean
18
+ // 样式部分
19
+ backgroundColor?: string
20
+ backgroundColor2?: string
21
+ textColor?: string
22
+ textFontSize?: number
23
+ title: string
24
+ // 是否需要生成和头部高度相同的盒子
25
+ isShowHeaderBox?: boolean
26
+ positionState?: string
27
+ isShowShadow?: boolean
28
+ isBlackIcon?: boolean
29
+ // 头部没有右侧盒子,true:左中右结构 false:左中结构(小程序可以为左中右结构但是没有右侧盒子)
30
+ isHaveRightBox?: boolean
31
+ }
32
+
33
+ const props = withDefaults(defineProps<headerInt>(), {
34
+ // 头部高度 默认为44px (微信小程序不可用)
35
+ headerHeight: 44,
36
+ // 是否显示左侧内容
37
+ leftIconShow: true,
38
+ // 样式部分
39
+ backgroundColor: 'linear-gradient(90deg, rgba(10, 207, 254, 1) 0%, rgba(74, 92, 255, 1) 100%)', //linear-gradient(90deg, rgba(10, 207, 254, 1) 0%, rgba(74, 92, 255, 1) 100%)
40
+ backgroundColor2: '#ffffff',
41
+ textColor: '#000',
42
+ textFontSize: 34,
43
+ title: '标题',
44
+ // 是否需要生成和头部高度相同的盒子
45
+ isShowHeaderBox: true,
46
+ positionState: 'fixed',
47
+ isShowShadow: false,
48
+ isBlackIcon: true, // 是否为黑色图标
49
+ isHaveRightBox: true
50
+ })
51
+
52
+ let { statusBarHeight } = uni.getSystemInfoSync()
53
+ // #ifdef MP-WEIXIN
54
+ // 胶囊状态
55
+ let menuButton = uni.getMenuButtonBoundingClientRect()
56
+ // 微信头部宽度
57
+ let wxHeaderWidth = menuButton.left - 10
58
+ // 上边距
59
+ statusBarHeight = menuButton.top
60
+ // #endif
61
+
62
+ // padding的高度(防止头部塌陷)
63
+ const fillBoxHeight = ref(statusBarHeight + 3)
64
+
65
+ // 设置header的高度
66
+ const headerHeightRef = ref(0)
67
+ watchEffect(() => {
68
+ headerHeightRef.value = props.headerHeight
69
+ // #ifdef MP-WEIXIN
70
+ // 中间高度
71
+ headerHeightRef.value = menuButton.height
72
+ // #endif
73
+ })
74
+ // console.log('statusBarHeight', statusBarHeight)
75
+
76
+ const style = computed(() => {
77
+ return {
78
+ boxShadow: props.isShowShadow ? '0 0 8rpx -3rpx #333' : '0 0 0 0 #333',
79
+ background: props.backgroundColor,
80
+ color: props.textColor,
81
+ position: props.positionState
82
+ } as CSSProperties
83
+ })
84
+ // 返回上一页(如没有页面返回首页)
85
+ const goBack = () => {
86
+ if (getCurrentPages().length <= 1) {
87
+ goToPage({
88
+ url: 'pages/index/index',
89
+ mode: 'redirectTo'
90
+ })
91
+ } else {
92
+ uni.navigateBack()
93
+ }
94
+ }
95
+
96
+ const backIcon = ''
97
+ const backIconW = ''
98
+ </script>
99
+
100
+ <template>
101
+ <!-- #ifndef MP-TOUTIAO -->
102
+ <view class="header_box">
103
+ <view class="header_main" :style="style">
104
+ <view class="status_bar" :style="{ height: statusBarHeight + 'px' }"></view>
105
+ <!-- 标准的左中右结构 -->
106
+ <view v-if="isHaveRightBox" class="header flex-center-between" :style="{ height: headerHeightRef + 'px' }">
107
+ <view class="header_left flex-center-between">
108
+ <slot name="left" v-if="leftIconShow">
109
+ <view class="icon flex" @click="goBack">
110
+ <image
111
+ :src="isBlackIcon ? backIcon : backIconW"
112
+ mode="widthFix"></image>
113
+ </view>
114
+ </slot>
115
+ </view>
116
+ <view class="header_center">
117
+ <slot name="center">
118
+ <view class="title" :style="{ fontSize: textFontSize + 'rpx' }">
119
+ {{ title }}
120
+ </view>
121
+ </slot>
122
+ </view>
123
+ <view class="header_right flex">
124
+ <!-- #ifndef MP-WEIXIN -->
125
+ <slot name="right"></slot>
126
+ <!-- #endif -->
127
+ </view>
128
+ </view>
129
+ <!-- 左右结构 -->
130
+ <view v-else class="wx_header flex"
131
+ :style="{ height: headerHeightRef + 'px', width: wxHeaderWidth + 'px' }">
132
+ <view class="wx_header_left flex">
133
+ <slot name="left">
134
+ <view class="icon flex" @click="goBack" v-if="leftIconShow">
135
+ <image
136
+ :src="isBlackIcon ? backIcon : backIconW"
137
+ mode="widthFix"></image>
138
+ </view>
139
+ </slot>
140
+ </view>
141
+ <view class="wx_header_txt flex">
142
+ <slot name="center">
143
+ <view class="title" :style="{ fontSize: textFontSize + 'rpx' }">
144
+ {{ title }}
145
+ </view>
146
+ </slot>
147
+ </view>
148
+ </view>
149
+ </view>
150
+ <!-- 填充头部防止塌陷 新加(v-if="isShowHeaderBox") -->
151
+ <view class="status_bar" v-if="isShowHeaderBox" :style="{ height: fillBoxHeight + 'px' }"></view>
152
+ <!-- #ifdef MP-WEIXIN -->
153
+ <view v-if="isShowHeaderBox" :style="{ height: headerHeightRef + 4 + 'px', background: backgroundColor2 }">
154
+ </view>
155
+ <!-- #endif -->
156
+ <!-- 待测试 -->
157
+ <!-- #ifndef MP-WEIXIN -->
158
+ <view v-if="isShowHeaderBox" :style="{ height: headerHeightRef + 'px', background: backgroundColor2 }"></view>
159
+ <!-- #endif -->
160
+ </view>
161
+ <!-- #endif -->
162
+ </template>
163
+
164
+ <style lang="scss" scoped>
165
+ .header_box {
166
+ // padding: 0 20rpx 5rpx;
167
+ // background-color: #fff;
168
+ }
169
+
170
+ .header_main {
171
+ width: 100%;
172
+ z-index: 9999;
173
+ top: 0;
174
+ left: 0;
175
+ /* #ifdef MP-WEIXIN */
176
+ padding: 0 20rpx 15rpx;
177
+ /* #endif */
178
+ /* #ifndef MP-WEIXIN */
179
+ padding: 5rpx 20rpx;
180
+ /* #endif */
181
+ box-sizing: border-box;
182
+ .img {
183
+ width: 48rpx;
184
+ height: 48rpx;
185
+ }
186
+ .header {
187
+ // padding: 0 16rpx;
188
+ box-sizing: border-box;
189
+
190
+ .header_left {
191
+ width: 20%;
192
+
193
+ .icon {
194
+ width: 48rpx;
195
+ }
196
+
197
+ .left_txt {
198
+ font-size: 22rpx;
199
+ line-height: 22rpx;
200
+ }
201
+ }
202
+
203
+ .header_center {
204
+ width: 60%;
205
+ text-align: center;
206
+ font-size: 28rpx;
207
+
208
+ .title {
209
+ overflow: hidden;
210
+ text-overflow: ellipsis;
211
+ white-space: nowrap;
212
+ }
213
+ }
214
+
215
+ .header_right {
216
+ width: 20%;
217
+ }
218
+ }
219
+
220
+ .wx_header {
221
+ .wx_header_left {
222
+ height: 100%;
223
+
224
+ .icon {
225
+ width: 48rpx;
226
+
227
+ image {
228
+ vertical-align: middle;
229
+ width: 100%;
230
+ }
231
+ }
232
+ }
233
+
234
+ .wx_header_txt {
235
+ flex: 1;
236
+ height: 100%;
237
+ padding-left: 26rpx;
238
+
239
+ .title {
240
+ width: 100%;
241
+ line-height: 1;
242
+ overflow: hidden;
243
+ text-overflow: ellipsis;
244
+ white-space: nowrap;
245
+ font-size: 28rpx;
246
+ }
247
+ }
248
+ }
249
+ }
250
+ </style>
@@ -1,4 +1,9 @@
1
1
  <script setup lang="ts">
2
+ /**
3
+ * @description:
4
+ * @param {*} msg 显示字符串
5
+ * @return {*}
6
+ */
2
7
  defineProps<{
3
8
  msg: string
4
9
  }>()
@@ -0,0 +1,3 @@
1
+ export const defauIcon = ''
2
+
3
+ export const nomoreIcon = ''
@@ -0,0 +1,102 @@
1
+ <script setup lang="ts">
2
+ import { ref, reactive, computed, watchEffect } from 'vue';
3
+ import { defauIcon, nomoreIcon } from './index'
4
+
5
+ /*
6
+ * @description: 图片组件
7
+ * @fileName: lx-image.vue
8
+ * @params src : 图片路径
9
+ * @params mode : 图片模式 默认:aspectFill 与uni相同
10
+ * @params width : 宽 默认:200rpx
11
+ * @params height : 高 默认:200rpx
12
+ * @params type : 区分分页和icon类型 默认:list
13
+ * @author: lxx
14
+ * @date: 2023-12-18"
15
+ * @version: V1.0.0
16
+ */
17
+ const props = withDefaults(
18
+ defineProps<{
19
+ src: string
20
+ mode?: string
21
+ width?: number | string
22
+ height?: number | string
23
+ type?: 'list' | 'icon'
24
+ radius?: string
25
+ isNeedPreview?: boolean
26
+ }>(),
27
+ {
28
+ src: '',
29
+ mode: 'aspectFill',
30
+ width: '200rpx',
31
+ height: '200rpx',
32
+ type: 'list',
33
+ isNeedPreview: true
34
+ }
35
+ )
36
+ const url = ref(props.src)
37
+ const isCorrect = ref(true)
38
+ const correctStyle = reactive({
39
+ width: props.width,
40
+ height: props.height,
41
+ borderRadius: props.radius,
42
+ overflow: 'hidden'
43
+ })
44
+
45
+ const errorStyle = reactive({
46
+ width: props.width,
47
+ height: props.height,
48
+ padding: '14rpx',
49
+ backgroundColor: '#dedede',
50
+ borderRadius: '4px'
51
+ })
52
+ const isImgError = ref(false)
53
+
54
+ const defaultPic = computed(() => {
55
+ // return `../../static/${props.type === 'list' ? 'nomore' : 'defauIcon'}.png`
56
+ return props.type === 'list' ? nomoreIcon : defauIcon
57
+ })
58
+ const handleImageError = () => {
59
+ isImgError.value = true
60
+ if (props.type !== 'list') {
61
+ isCorrect.value = false
62
+ }
63
+ url.value = defaultPic.value
64
+ }
65
+
66
+ const previewImage = () => {
67
+ if (!props.isNeedPreview || isImgError.value === true) return
68
+ uni.previewImage({
69
+ urls: [url.value]
70
+ })
71
+ }
72
+
73
+ watchEffect(() => {
74
+ if (!isImgError.value) {
75
+ url.value = props.src
76
+ if (!url.value) {
77
+ handleImageError()
78
+ return
79
+ }
80
+ // 判断图片地址是否有http
81
+ if (url.value.indexOf('http') === -1) {
82
+ url.value = import.meta.env.VITE_APP_BASE_URL + url.value
83
+ }
84
+ }
85
+ })
86
+ </script>
87
+
88
+ <template>
89
+ <view class="images_lxx" :style="isCorrect ? correctStyle : errorStyle" @click="previewImage()">
90
+ <image :src="url" @error="handleImageError" :mode="mode" />
91
+ </view>
92
+ </template>
93
+ <style lang="scss" scoped>
94
+ .images_lxx {
95
+ box-sizing: border-box;
96
+
97
+ image {
98
+ width: 100%;
99
+ height: 100%;
100
+ }
101
+ }
102
+ </style>
@@ -0,0 +1,162 @@
1
+ <script setup lang="ts">
2
+ /*
3
+ * @description: 分页组件
4
+ * @fileName: List.vue
5
+ * @params {Function} api : 数据请求的接口API
6
+ * @params {Function} afterLoadData : 数据请求完毕前置处理方法
7
+ * @params {Boolean} isNeedSearch : 是否需要搜索---(暂不可用,待修改)
8
+ * @params {Boolean} isFixedSearch : 是否需要固定定位搜索框(仅listType:'default'有效)
9
+ * @params {object: any} options : 数据请求的额外参数
10
+ * @params {'default' | 'scrollView'} listType : scrollView为scroll-view包裹的分页
11
+ * @params {number} scrollTopThreshold : 用于指定滚动条距离顶部的阈值->触发输入框固定 (仅listType:'scrollView'有效)
12
+ * @params {string} searchPlaceholder : 搜索框默认提示
13
+ * @ref setListParams {Function} : 使用==>ref.value.setListParams(obj, isClear) obj:请求额外的参数 isClear是否清空请求参数
14
+ * @author: lxx
15
+ * @date: 2023-07-28 11:36:23
16
+ * @update: 2023-10-13 10:34:03
17
+ * @version: V1.0.1
18
+ */
19
+ import { LoadData } from '../../util/useListLoadClass'
20
+ import { debounce } from '../../util'
21
+ import { toRefs, ref } from 'vue';
22
+ import lxListState from '../lx-list-state/lx-list-state.vue'
23
+ interface listPropsInt {
24
+ api: Function
25
+ afterLoadData?: Function
26
+ isNeedSearch?: boolean
27
+ isFixedSearch?: boolean
28
+ options?: any
29
+ scrollTopThreshold?: number
30
+ listType?: 'default' | 'scrollView'
31
+ searchPlaceholder?: string
32
+ }
33
+
34
+ // 接收传值
35
+ // 3.2.x defineProps泛型类型参数仅限于类型文字或对本地接口的引用
36
+ const props = withDefaults(defineProps<listPropsInt>(), {
37
+ api: () => ({}),
38
+ isNeedSearch: true,
39
+ isFixedSearch: true,
40
+ // eslint-disable-next-line @typescript-eslint/no-empty-function
41
+ options: {},
42
+ scrollTopThreshold: 50,
43
+ listType: 'default',
44
+ searchPlaceholder: '请输入搜索内容'
45
+ })
46
+ let { options, api, afterLoadData } = toRefs(props)
47
+ console.log('props', props.options, options.value)
48
+
49
+ // 分页相关
50
+ let { list, isLoading, isEmpty, isNoData, setParams, LoadMore } = LoadData({
51
+ api: api.value,
52
+ afterLoadData: afterLoadData?.value,
53
+ options: options.value
54
+ })
55
+ // 搜索相关
56
+ const inputTxt = ref('')
57
+ const inputChange = debounce(() => {
58
+ console.log('input change')
59
+ setParams({ search: inputTxt.value })
60
+ })
61
+
62
+ // 用于父组件设置额外的参数(选项卡切换的时候使用)--> isClear true 清空列表需测试
63
+ const setListParams = (obj: any, isClear = true) => {
64
+ const param = ref<{
65
+ search?: string
66
+ [key: string]: any
67
+ }>({})
68
+ if (props.isNeedSearch) {
69
+ param.value.search = inputTxt.value
70
+ }
71
+ param.value = { ...obj }
72
+ setParams({ ...param.value }, isClear)
73
+ }
74
+ defineExpose({
75
+ list,
76
+ setListParams
77
+ })
78
+ // 页面滚动相关
79
+ const scrollTop = ref(0)
80
+ const scroll = (e: any) => {
81
+ scrollTop.value = e.detail.scrollTop
82
+ }
83
+
84
+ const emit = defineEmits(['scrollTolower'])
85
+ const scrolltolower = () => {
86
+ console.log('触底啦!')
87
+ LoadMore()
88
+ emit('scrollTolower')
89
+ }
90
+ </script>
91
+ <template>
92
+ <template v-if="listType == 'default'">
93
+ <view class="lxx_list_box">
94
+ <view class="search_box" v-if="isNeedSearch" :style="{ position: isFixedSearch ? 'fixed' : 'unset' }">
95
+ <!-- <u-search :placeholder="searchPlaceholder" v-model="inputTxt" @change="inputChange" :show-action="false"></u-search> -->
96
+ </view>
97
+ <view class="lxx_list_box_content" :style="{ paddingTop: isNeedSearch && isFixedSearch ? '100rpx' : 'unset' }">
98
+ <slot name="list_title"></slot>
99
+ <template v-for="(item, index) in list" :key="index">
100
+ <slot v-bind:item="item" v-bind:index="index"></slot>
101
+ </template>
102
+ <slot name="list_bottom"></slot>
103
+ </view>
104
+ <lxListState :is-empty="isEmpty" :is-loading="isLoading" :is-no-data="isNoData"></lxListState>
105
+ </view>
106
+ </template>
107
+
108
+ <template v-else>
109
+ <scroll-view scroll-y="true" class="scroll-Y" @scroll="scroll" @scrolltolower="scrolltolower">
110
+ <view
111
+ class="search_box"
112
+ v-if="isNeedSearch"
113
+ :class="{ lxx_scroll_fixed: isNeedSearch && scrollTop > scrollTopThreshold }"
114
+ :style="{ top: scrollTop + 'px' }"
115
+ >
116
+ <!-- <u-search :placeholder="searchPlaceholder" v-model="inputTxt" @change="inputChange" :show-action="false"></u-search> -->
117
+ </view>
118
+ <!-- 防止页面上顶 -->
119
+ <view class="search_box" v-if="isNeedSearch && scrollTop > scrollTopThreshold">
120
+ <!-- <u-search :placeholder="searchPlaceholder" v-model="inputTxt" @change="inputChange" :show-action="false"></u-search> -->
121
+ </view>
122
+ <view class="lxx_list_box">
123
+ <slot name="list_title"></slot>
124
+ <view v-for="(item, index) in list" :key="index">
125
+ <slot v-bind:item="item" v-bind:index="index"></slot>
126
+ </view>
127
+ <slot name="list_bottom"></slot>
128
+ </view>
129
+ <lxListState :is-empty="isEmpty" :is-loading="isLoading" :is-no-data="isNoData"></lxListState>
130
+ </scroll-view>
131
+ </template>
132
+ </template>
133
+ <style lang="scss" scoped>
134
+ .lxx_list_box {
135
+ position: relative;
136
+
137
+ .search_box {
138
+ padding: 10rpx 20rpx;
139
+ width: 100%;
140
+ height: 90rpx;
141
+ background-color: #fff;
142
+ box-sizing: border-box;
143
+ }
144
+ }
145
+
146
+ .scroll-Y {
147
+ height: 100%;
148
+ position: relative;
149
+
150
+ .search_box {
151
+ padding: 10rpx 20rpx;
152
+ width: 100%;
153
+ box-sizing: border-box;
154
+ background-color: #ffffff;
155
+ z-index: 2;
156
+ }
157
+
158
+ .lxx_scroll_fixed {
159
+ position: absolute;
160
+ }
161
+ }
162
+ </style>