@opendesign-plus/components 0.0.1-rc.2 → 0.0.1-rc.20

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.
Files changed (203) hide show
  1. package/dist/chunk-OElCookieNotice.cjs.js +1 -0
  2. package/dist/chunk-OElCookieNotice.es.js +839 -0
  3. package/dist/components/OBanner.vue.d.ts +10 -1
  4. package/dist/components/OCookieNotice.vue.d.ts +6 -5
  5. package/dist/components/OFooter.vue.d.ts +46 -0
  6. package/dist/components/OHeaderSearch.vue.d.ts +4 -4
  7. package/dist/components/OHeaderUser.vue.d.ts +40 -0
  8. package/dist/components/OSourceCode.vue.d.ts +20 -0
  9. package/dist/components/activity/OActivityApproval.vue.d.ts +277 -0
  10. package/dist/components/activity/OActivityForm.vue.d.ts +140 -0
  11. package/dist/components/activity/OMyActivityCalendar.vue.d.ts +578 -0
  12. package/dist/components/activity/composables/useActivityConfig.d.ts +17 -0
  13. package/dist/components/activity/config.d.ts +1 -0
  14. package/dist/components/activity/index.d.ts +623 -0
  15. package/dist/components/activity/types.d.ts +81 -0
  16. package/dist/components/element-plus/OElCookieNotice.vue.d.ts +34 -0
  17. package/dist/components/element-plus/index.d.ts +2 -0
  18. package/dist/components/events/OEventsApply.vue.d.ts +16 -0
  19. package/dist/components/events/OEventsCalendar.vue.d.ts +5 -0
  20. package/dist/components/events/OEventsList.vue.d.ts +26 -0
  21. package/dist/components/events/config.d.ts +27 -0
  22. package/dist/components/events/index.d.ts +78 -0
  23. package/dist/components/events/types.d.ts +66 -0
  24. package/dist/components/events/utils.d.ts +7 -0
  25. package/dist/components/header/OHeader.vue.d.ts +30 -0
  26. package/dist/components/header/OHeaderMobile.vue.d.ts +171 -0
  27. package/dist/components/header/components/HeaderContent.vue.d.ts +13 -0
  28. package/dist/components/header/components/HeaderNav.vue.d.ts +19 -0
  29. package/dist/components/header/components/HeaderNavMobile.vue.d.ts +33 -0
  30. package/dist/components/header/index.d.ts +145 -0
  31. package/dist/components/header/types.d.ts +80 -0
  32. package/dist/components/meeting/OMeetingCalendar.vue.d.ts +299 -0
  33. package/dist/components/meeting/OMeetingForm.vue.d.ts +145 -0
  34. package/dist/components/meeting/OMyMeetingCalendar.vue.d.ts +586 -0
  35. package/dist/components/meeting/OSigMeetingCalendar.vue.d.ts +24 -0
  36. package/dist/components/meeting/components/OMeetingCalendarList.vue.d.ts +28 -0
  37. package/dist/components/meeting/components/OMeetingCalendarSelector.vue.d.ts +664 -0
  38. package/dist/components/meeting/components/OMeetingDetail.vue.d.ts +13 -0
  39. package/dist/components/meeting/components/OMeetingPlaybackSubtitles.vue.d.ts +5 -0
  40. package/dist/components/meeting/components/OMeetingPlaybackVideo.vue.d.ts +17 -0
  41. package/dist/components/meeting/components/OSigMeetingAside.vue.d.ts +16 -0
  42. package/dist/components/meeting/composables/useMeetingConfig.d.ts +14 -0
  43. package/dist/components/meeting/config.d.ts +12 -0
  44. package/dist/components/meeting/types.d.ts +168 -0
  45. package/dist/components/meeting/utils.d.ts +8 -0
  46. package/dist/components.cjs.js +224 -1
  47. package/dist/components.css +1 -1
  48. package/dist/components.element.cjs.js +1 -0
  49. package/dist/components.element.es.js +4 -0
  50. package/dist/components.es.js +44348 -785
  51. package/dist/index.d.ts +9 -2
  52. package/docs/design.md +27 -27
  53. package/docs/design_banner.md +41 -41
  54. package/docs/design_section.md +27 -27
  55. package/package.json +12 -3
  56. package/scripts/generate-components-index.js +103 -80
  57. package/src/assets/events/svg-icons/icon-checked.svg +3 -0
  58. package/src/assets/events/svg-icons/icon-competition.svg +7 -0
  59. package/src/assets/events/svg-icons/icon-events.svg +4 -0
  60. package/src/assets/events/svg-icons/icon-release.svg +4 -0
  61. package/src/assets/events/svg-icons/icon-summit.svg +4 -0
  62. package/src/assets/meeting/svg-icons/icon-all.svg +3 -0
  63. package/src/assets/meeting/svg-icons/icon-backward.svg +4 -0
  64. package/src/assets/meeting/svg-icons/icon-calendar.svg +3 -0
  65. package/src/assets/meeting/svg-icons/icon-cancel.svg +4 -0
  66. package/src/assets/meeting/svg-icons/icon-captions.svg +4 -0
  67. package/src/assets/meeting/svg-icons/icon-close-captions.svg +6 -0
  68. package/src/assets/meeting/svg-icons/icon-close-fullscreen.svg +6 -0
  69. package/src/assets/meeting/svg-icons/icon-copy.svg +3 -0
  70. package/src/assets/meeting/svg-icons/icon-create.svg +5 -0
  71. package/src/assets/meeting/svg-icons/icon-delete.svg +7 -0
  72. package/src/assets/meeting/svg-icons/icon-empty.svg +31 -0
  73. package/src/assets/meeting/svg-icons/icon-empty_dark.svg +49 -0
  74. package/src/assets/meeting/svg-icons/icon-event.svg +3 -0
  75. package/src/assets/meeting/svg-icons/icon-export.svg +3 -0
  76. package/src/assets/meeting/svg-icons/icon-forward.svg +4 -0
  77. package/src/assets/meeting/svg-icons/icon-fullscreen.svg +6 -0
  78. package/src/assets/meeting/svg-icons/icon-help.svg +3 -0
  79. package/src/assets/meeting/svg-icons/icon-important.svg +4 -0
  80. package/src/assets/meeting/svg-icons/icon-info.svg +3 -0
  81. package/src/assets/meeting/svg-icons/icon-meet.svg +3 -0
  82. package/src/assets/meeting/svg-icons/icon-meeting-message.svg +5 -0
  83. package/src/assets/meeting/svg-icons/icon-meeting.svg +4 -0
  84. package/src/assets/meeting/svg-icons/icon-play.svg +5 -0
  85. package/src/assets/meeting/svg-icons/icon-playing-tip.svg +7 -0
  86. package/src/assets/meeting/svg-icons/icon-playing.svg +5 -0
  87. package/src/assets/meeting/svg-icons/icon-question.svg +4 -0
  88. package/src/assets/meeting/svg-icons/icon-sound.svg +5 -0
  89. package/src/assets/meeting/svg-icons/icon-speaker.svg +3 -0
  90. package/src/assets/meeting/svg-icons/icon-summit.svg +3 -0
  91. package/src/assets/meeting/svg-icons/icon-telligent.svg +3 -0
  92. package/src/assets/meeting/svg-icons/icon-tip.svg +3 -0
  93. package/src/assets/meeting/svg-icons/icon-todo.svg +4 -0
  94. package/src/assets/meeting/transparent.png +0 -0
  95. package/src/assets/svg-icons/icon-arrow-left.svg +3 -0
  96. package/src/assets/svg-icons/icon-avatar-line.svg +3 -0
  97. package/src/assets/svg-icons/icon-caret-left.svg +3 -0
  98. package/src/assets/svg-icons/icon-caret-right.svg +3 -0
  99. package/src/assets/svg-icons/icon-chevron-down.svg +3 -0
  100. package/src/assets/svg-icons/icon-chevron-right.svg +3 -3
  101. package/src/assets/svg-icons/icon-chevron-up.svg +3 -0
  102. package/src/assets/svg-icons/icon-close.svg +3 -3
  103. package/src/assets/svg-icons/icon-delete.svg +3 -3
  104. package/src/assets/svg-icons/icon-filter.svg +3 -0
  105. package/src/assets/svg-icons/icon-header-back.svg +3 -3
  106. package/src/assets/svg-icons/icon-header-delete.svg +3 -3
  107. package/src/assets/svg-icons/icon-header-menu.svg +3 -0
  108. package/src/assets/svg-icons/icon-header-person.svg +3 -0
  109. package/src/assets/svg-icons/icon-header-search.svg +4 -4
  110. package/src/assets/svg-icons/icon-loading.svg +4 -0
  111. package/src/assets/svg-icons/icon-locale.svg +3 -0
  112. package/src/assets/svg-icons/icon-log-off.svg +3 -0
  113. package/src/assets/svg-icons/icon-message.svg +3 -0
  114. package/src/assets/svg-icons/icon-moon.svg +3 -3
  115. package/src/assets/svg-icons/icon-outlink.svg +3 -0
  116. package/src/assets/svg-icons/icon-overview.svg +3 -0
  117. package/src/assets/svg-icons/icon-search.svg +3 -0
  118. package/src/assets/svg-icons/icon-setting.svg +3 -0
  119. package/src/assets/svg-icons/icon-sun.svg +3 -3
  120. package/src/assets/svg-icons/icon-tips.svg +3 -0
  121. package/src/components/OBanner.vue +398 -390
  122. package/src/components/OCookieNotice.vue +575 -424
  123. package/src/components/OFooter.vue +576 -0
  124. package/src/components/OHeaderSearch.vue +601 -601
  125. package/src/components/OHeaderUser.vue +237 -0
  126. package/src/components/OPlusConfigProvider.vue +32 -32
  127. package/src/components/OSection.vue +178 -178
  128. package/src/components/OSourceCode.vue +153 -0
  129. package/src/components/OThemeSwitcher.vue +108 -108
  130. package/src/components/activity/OActivityApproval.vue +864 -0
  131. package/src/components/activity/OActivityForm.vue +542 -0
  132. package/src/components/activity/OMyActivityCalendar.vue +1486 -0
  133. package/src/components/activity/composables/useActivityConfig.ts +141 -0
  134. package/src/components/activity/config.ts +1 -0
  135. package/src/components/activity/index.ts +24 -0
  136. package/src/components/activity/types.ts +88 -0
  137. package/src/components/common/AppAvatar.vue +83 -0
  138. package/src/components/common/ClientOnlyWrapper.ts +21 -21
  139. package/src/components/common/ContentWrapper.vue +85 -85
  140. package/src/components/common/MoreText.vue +124 -0
  141. package/src/components/common/ThFilter.vue +330 -0
  142. package/src/components/element-plus/OElCookieNotice.vue +603 -0
  143. package/src/components/element-plus/index.ts +3 -0
  144. package/src/components/events/OEventsApply.vue +418 -0
  145. package/src/components/events/OEventsCalendar.vue +598 -0
  146. package/src/components/events/OEventsList.vue +362 -0
  147. package/src/components/events/config.ts +35 -0
  148. package/src/components/events/index.ts +24 -0
  149. package/src/components/events/types.ts +80 -0
  150. package/src/components/events/utils.ts +9 -0
  151. package/src/components/header/OHeader.vue +157 -0
  152. package/src/components/header/OHeaderMobile.vue +177 -0
  153. package/src/components/header/components/HeaderContent.vue +1121 -0
  154. package/src/components/header/components/HeaderNav.vue +278 -0
  155. package/src/components/header/components/HeaderNavMobile.vue +377 -0
  156. package/src/components/header/index.ts +16 -0
  157. package/src/components/header/types.ts +91 -0
  158. package/src/components/meeting/OMeetingCalendar.vue +878 -0
  159. package/src/components/meeting/OMeetingForm.vue +1053 -0
  160. package/src/components/meeting/OMeetingPlayback.vue +439 -0
  161. package/src/components/meeting/OMyMeetingCalendar.vue +1497 -0
  162. package/src/components/meeting/OSigMeetingCalendar.vue +410 -0
  163. package/src/components/meeting/components/OMeetingCalendarList.vue +510 -0
  164. package/src/components/meeting/components/OMeetingCalendarSelector.vue +206 -0
  165. package/src/components/meeting/components/OMeetingDetail.vue +229 -0
  166. package/src/components/meeting/components/OMeetingPlaybackSubtitles.vue +611 -0
  167. package/src/components/meeting/components/OMeetingPlaybackVideo.vue +741 -0
  168. package/src/components/meeting/components/OSigMeetingAside.vue +200 -0
  169. package/src/components/meeting/composables/useMeetingConfig.ts +108 -0
  170. package/src/components/meeting/config.ts +48 -0
  171. package/src/components/meeting/index.ts +45 -0
  172. package/src/components/meeting/types.ts +195 -0
  173. package/src/components/meeting/utils.ts +70 -0
  174. package/src/draft/Banner.vue +265 -265
  175. package/src/draft/ButtonCards.vue +105 -105
  176. package/src/draft/Feature.vue +133 -133
  177. package/src/draft/Footer.vue +512 -512
  178. package/src/draft/HorizontalAnchor.vue +165 -165
  179. package/src/draft/ItemSwiper.vue +133 -133
  180. package/src/draft/Logo.vue +141 -141
  181. package/src/draft/LogoCard.vue +74 -74
  182. package/src/draft/LogoV2.vue +19 -19
  183. package/src/draft/MainCard.vue +38 -38
  184. package/src/draft/MultiCard.vue +94 -94
  185. package/src/draft/MultiIconCard.vue +73 -73
  186. package/src/draft/OInfoCard.vue +176 -176
  187. package/src/draft/Process.vue +81 -81
  188. package/src/draft/Section.vue +167 -167
  189. package/src/draft/SingleTabCard.vue +84 -84
  190. package/src/draft/SliderCard.vue +110 -110
  191. package/src/env.d.ts +16 -1
  192. package/src/i18n/en.ts +264 -20
  193. package/src/i18n/index.ts +56 -42
  194. package/src/i18n/zh.ts +253 -9
  195. package/src/index.ts +14 -3
  196. package/src/shared/provide.ts +6 -6
  197. package/src/shims-vue-dompurify-html.d.ts +17 -0
  198. package/src/vue.d.ts +9 -9
  199. package/tsconfig.json +37 -33
  200. package/vite.config.ts +118 -94
  201. package/dist/components/OCookieNoticeEl.vue.d.ts +0 -29
  202. package/dist/components.umd.js +0 -1
  203. package/src/components/OCookieNoticeEl.vue +0 -404
@@ -0,0 +1,362 @@
1
+ <script setup lang="ts">
2
+ import {
3
+ OCard,
4
+ ODivider,
5
+ OIcon,
6
+ OIconSearch,
7
+ OInput, OPagination,
8
+ ORadio,
9
+ ORadioGroup,
10
+ OTag,
11
+ OToggle,
12
+ } from '@opensig/opendesign';
13
+ import { computed, ref } from 'vue';
14
+ import { type EventsCardItemT, EventsStatusT, type EventsListPropsT, SearchParamsT } from './types';
15
+ import { compareDate } from './utils.ts';
16
+ import { useI18n } from '@/i18n';
17
+
18
+ import { CITY_MAP, DEFAULT_COVER } from './config.ts';
19
+ import { useScreen } from '@opendesign-plus/composables';
20
+
21
+ const { t } = useI18n();
22
+ const { lePadV } = useScreen();
23
+
24
+ const props = withDefaults(
25
+ defineProps<EventsListPropsT>(),
26
+ {
27
+ data: () => [],
28
+ },
29
+ );
30
+
31
+ interface StatusOptionT {
32
+ label: string;
33
+ value: EventsStatusT;
34
+ }
35
+
36
+ const statusOptions: StatusOptionT[] = [
37
+ {
38
+ label: t('events.statusAll'),
39
+ value: EventsStatusT.ALL,
40
+ },
41
+ {
42
+ label: t('events.statusIng'),
43
+ value: EventsStatusT.ING,
44
+ },
45
+ {
46
+ label: t('events.statusFinish'),
47
+ value: EventsStatusT.FINISH,
48
+ },
49
+ ];
50
+
51
+ defineSlots<{
52
+ empty?: () => any;
53
+ }>();
54
+
55
+ const emits = defineEmits<{
56
+ (e: 'search', params: SearchParamsT): void;
57
+ }>();
58
+ const status = ref<EventsStatusT>(EventsStatusT.ALL);
59
+ const input = ref('');
60
+ const keyword = ref('');
61
+
62
+ const emitSearch = (params = {}) => {
63
+ emits(
64
+ 'search',
65
+ Object.assign(
66
+ {
67
+ page: props.page,
68
+ pageSize: props.pageSize,
69
+ status: status.value,
70
+ keyword: keyword.value,
71
+ },
72
+ params,
73
+ ),
74
+ );
75
+ };
76
+
77
+ const changeStatus = () => {
78
+ emitSearch();
79
+ };
80
+ const changeKeyword = () => {
81
+ keyword.value = input.value;
82
+ emitSearch();
83
+ };
84
+ const list = computed(() => {
85
+ return props.data
86
+ .map((v: EventsCardItemT) => {
87
+ return {
88
+ ...v,
89
+ status: !compareDate(new Date(), v.date as string) ? EventsStatusT.ING : EventsStatusT.FINISH,
90
+ };
91
+ })
92
+ .sort((a: EventsCardItemT, b: EventsCardItemT) => (compareDate(a.date as string, b.date as string) ? -1 : 1))
93
+ .map((v) => {
94
+ const city = v.city?.replace('市', '')?.replace('中国', '')?.replaceAll(' ', '');
95
+ let cover = null;
96
+
97
+ if (Array.isArray(CITY_MAP[city])) {
98
+ cover = CITY_MAP[city][v.name.length % CITY_MAP[city].length];
99
+ } else {
100
+ cover = CITY_MAP[city];
101
+ }
102
+ if (!cover) {
103
+ cover = DEFAULT_COVER;
104
+ }
105
+ return {
106
+ ...v,
107
+ cover,
108
+ };
109
+ });
110
+ });
111
+ </script>
112
+
113
+ <template>
114
+ <div class="o-events-list">
115
+ <div class="filter-wrapper">
116
+ <div class="filter-left">
117
+ <span>{{ t('events.status') }}</span>
118
+ <ORadioGroup v-model="status" :style="{ '--radio-group-gap': '8px' }" @change="changeStatus">
119
+ <ORadio v-for="t in statusOptions" :key="t.value" :value="t.value">
120
+ <template #radio="{ checked }">
121
+ <OToggle :checked="checked">{{ t.label }}</OToggle>
122
+ </template>
123
+ </ORadio>
124
+ </ORadioGroup>
125
+ </div>
126
+ <OInput
127
+ v-model="input"
128
+ :placeholder="t('events.searchPlaceholder')"
129
+ @pressEnter="changeKeyword"
130
+ @clear="changeKeyword"
131
+ clearable>
132
+ <template #prefix>
133
+ <OIcon class="input-icon">
134
+ <OIconSearch />
135
+ </OIcon>
136
+ </template>
137
+ </OInput>
138
+ </div>
139
+ <div class="list-content">
140
+ <OCard v-for="item in list" :key="item.name" :cover="item.cover" :href="item.link">
141
+ <OTag class="event-status">
142
+ {{ item.status === EventsStatusT.ING ? t('events.statusIng') : t('events.statusFinish') }}
143
+ </OTag>
144
+ <div class="event-name">{{ item.name }}</div>
145
+ <div class="event-desc">
146
+ <div class="event-date">{{ item.date }}</div>
147
+ <ODivider direction="v" />
148
+ <div class="event-location" v-if="item.city">{{ item.city }}</div>
149
+ </div>
150
+ </OCard>
151
+ </div>
152
+ <template v-if="!list.length">
153
+ <slot name="empty"></slot>
154
+ </template>
155
+ <OPagination
156
+ v-if="total > 8"
157
+ :page="page"
158
+ :simple="lePadV"
159
+ :show-total="true"
160
+ :pageSize="pageSize"
161
+ :total="total"
162
+ @change="emitSearch"
163
+ :pageSizes="[4, 12, 24, 36, 48]"
164
+ />
165
+ </div>
166
+ </template>
167
+
168
+ <style lang="scss">
169
+ .o-events-list {
170
+ .filter-wrapper {
171
+ display: flex;
172
+ align-items: center;
173
+ flex-wrap: wrap;
174
+ justify-content: space-between;
175
+ background-color: var(--o-color-fill2);
176
+ border-radius: var(--o-radius-xs);
177
+ padding: var(--o-gap-5) var(--o-gap-6);
178
+ gap: var(--o-gap-3) var(--o-gap-2);
179
+ @include text1;
180
+ @include respond-to('<=pad_v') {
181
+ padding: var(--o-gap-3) var(--o-gap-4);
182
+ }
183
+
184
+ .filter-left {
185
+ display: flex;
186
+ align-items: center;
187
+ gap: var(--o-gap-2);
188
+ flex-wrap: wrap;
189
+
190
+ span:first-child {
191
+ flex-shrink: 0;
192
+ margin-right: var(--o-gap-5);
193
+ font-weight: 500;
194
+ @include respond-to('pad_v') {
195
+ margin-right: var(--o-gap-4);
196
+ }
197
+ @include respond-to('phone') {
198
+ margin-right: var(--o-gap-4);
199
+ }
200
+ }
201
+
202
+ .o-toggle {
203
+ --toggle-size: 32px;
204
+ --toggle-padding: 3px 15px;
205
+ --toggle-radius: 4px;
206
+ max-height: 32px;
207
+ --toggle-bg-color: var(--o-color-fill1);
208
+ --toggle-bg-color-hover: var(--o-color-control2-light);
209
+ --toggle-color: var(--o-color-info1);
210
+ --toggle-color-hover: var(--o-color-info1);
211
+ --toggle-color-active: var(--o-color-primary1);
212
+ --toggle-color-disabled: var(--o-color-info4);
213
+ @include text1;
214
+ }
215
+ }
216
+
217
+ .o-input {
218
+ .o-icon-search {
219
+ font-size: 24px;
220
+ @include respond-to('<=pad_v') {
221
+ font-size: 16px;
222
+ }
223
+ }
224
+ }
225
+ }
226
+ .list-content {
227
+ display: grid;
228
+ grid-template-columns: repeat(4, 1fr);
229
+ gap: var(--o-gap-6);
230
+ margin-top: var(--o-gap-6);
231
+ @include respond-to('<=pad_v') {
232
+ grid-template-columns: repeat(1, 1fr);
233
+ container-type: inline-size;
234
+ }
235
+ @include respond-to('pad_v') {
236
+ margin-top: var(--o-gap-4);
237
+ gap: var(--o-gap-3);
238
+ }
239
+ @include respond-to('phone') {
240
+ margin-top: var(--o-gap-4);
241
+ gap: var(--o-gap-3);
242
+ }
243
+
244
+ .o-card {
245
+ aspect-ratio: 1 / 1;
246
+ position: relative;
247
+ @include respond-to('<=pad_v') {
248
+ aspect-ratio: 1 / 0.4;
249
+ }
250
+ @container (max-width: 300px) {
251
+ height: 140px;
252
+ aspect-ratio: unset;
253
+ }
254
+
255
+ .o-card-cover {
256
+ padding: 0;
257
+ }
258
+
259
+ .o-card-content {
260
+ display: flex;
261
+ flex-direction: column;
262
+ align-items: center;
263
+ justify-content: center;
264
+ position: absolute;
265
+ overflow: hidden; // 防止内容溢出
266
+ box-sizing: border-box; // 确保 padding 计入总尺寸
267
+ left: 0;
268
+ top: 0;
269
+ width: 100%;
270
+ height: 100%;
271
+ color: var(--o-color-white);
272
+ text-align: center;
273
+ cursor: pointer;
274
+ padding: var(--o-gap-6);
275
+ z-index: 10;
276
+
277
+ @include respond-to('pad_v') {
278
+ margin-top: var(--o-gap-4);
279
+ }
280
+ @include respond-to('phone') {
281
+ margin-top: var(--o-gap-4);
282
+ }
283
+
284
+ .event-status {
285
+ --tag-color: rgb(var(--o-white));
286
+ --tag-bg-color: rgba(var(--o-black), 0.25);
287
+ --tag-padding: var(--o-gap-1) var(--o-gap-3);
288
+ border: none;
289
+ position: absolute;
290
+ top: var(--o-gap-3);
291
+ right: var(--o-gap-3);
292
+ border-radius: var(--o-radius-xs);
293
+ @include respond-to('<=pad_v') {
294
+ --tag-padding: var(--o-gap-1);
295
+ }
296
+ }
297
+
298
+ .event-name {
299
+ font-weight: 500;
300
+ text-align: center;
301
+ height: 3lh; // 3倍行高
302
+ @include h2;
303
+ @include text-truncate(3);
304
+ @include respond-to('pad_v') {
305
+ @include h1;
306
+ }
307
+ @include respond-to('phone') {
308
+ height: 2lh;
309
+ @include text-truncate(2);
310
+ }
311
+ }
312
+
313
+ .event-desc {
314
+ text-align: center;
315
+ font-weight: 500;
316
+ margin-top: var(--o-gap-5);
317
+
318
+ .o-divider {
319
+ --o-divider-color: var(--o-color-info1-inverse);
320
+ --o-divider-bd-color: var(--o-color-info1-inverse);
321
+ display: none;
322
+ }
323
+
324
+ @include text1;
325
+ @include respond-to('pad_v') {
326
+ margin-top: var(--o-gap-2);
327
+ }
328
+ @include respond-to('phone') {
329
+ margin-top: var(--o-gap-2);
330
+ }
331
+ @include respond-to('<=pad_v') {
332
+ display: flex;
333
+ align-items: center;
334
+ .o-divider {
335
+ display: block;
336
+ }
337
+ @include tip1;
338
+ }
339
+ }
340
+ }
341
+ }
342
+
343
+ .event-card {
344
+ border-radius: var(--o-radius-xs);
345
+ color: rgb(var(--o-white));
346
+ background-size: cover;
347
+ background-repeat: no-repeat;
348
+ }
349
+ }
350
+
351
+ .o-pagination {
352
+ margin-top: var(--o-gap-6);
353
+ --pagination-item-bg-color-current: var(--o-color-primary1);
354
+ --pagination-item-color-current: #fff;
355
+
356
+ .o-pagination-wrap {
357
+ justify-content: flex-end;
358
+ }
359
+ }
360
+
361
+ }
362
+ </style>
@@ -0,0 +1,35 @@
1
+ import IconSummit from '~icons/events/icon-summit.svg';
2
+ import IconEvents from '~icons/events/icon-events.svg';
3
+ import IconCompetition from '~icons/events/icon-competition.svg';
4
+ import IconRelease from '~icons/events/icon-release.svg';
5
+
6
+ export const EVENTS_COLOR_MAP = {
7
+ summit: '--o-orange-6',
8
+ events: '--o-cyan-6',
9
+ competition: '--o-blue-6',
10
+ release: '--o-purple-6',
11
+ };
12
+
13
+ export const EVENTS_ICON_MAP = {
14
+ events: IconEvents,
15
+ competition: IconCompetition,
16
+ release: IconRelease,
17
+ summit: IconSummit,
18
+ };
19
+
20
+ export const CITY_MAP = {
21
+ 上海: 'https://infrastructure-website.osinfra.cn/picture/city/shanghai.jpg',
22
+ 北京: 'https://infrastructure-website.osinfra.cn/picture/city/beijing.jpg',
23
+ 南京: 'https://infrastructure-website.osinfra.cn/picture/city/nanjing.jpg',
24
+ 天津: 'https://infrastructure-website.osinfra.cn/picture/city/tianjin.jpg',
25
+ 成都: 'https://infrastructure-website.osinfra.cn/picture/city/chengdu.jpg',
26
+ 无锡: 'https://infrastructure-website.osinfra.cn/picture/city/wuxi.jpg',
27
+ 杭州: 'https://infrastructure-website.osinfra.cn/picture/city/hangzhou.jpg',
28
+ 深圳: ['https://infrastructure-website.osinfra.cn/picture/city/shenzhen1.jpg', 'https://infrastructure-website.osinfra.cn/picture/city/shenzhen2.jpg'],
29
+ 苏州: 'https://infrastructure-website.osinfra.cn/picture/city/suzhou.jpg',
30
+ 西安: 'https://infrastructure-website.osinfra.cn/picture/city/xian.jpg',
31
+ 郑州: 'https://infrastructure-website.osinfra.cn/picture/city/zhengzhou.jpg',
32
+ 武汉: 'https://infrastructure-website.osinfra.cn/picture/city/wuhan.jpg',
33
+ };
34
+
35
+ export const DEFAULT_COVER = 'https://infrastructure-website.osinfra.cn/picture/city/default-cover.jpg';
@@ -0,0 +1,24 @@
1
+ import _OEventsCalendar from './OEventsCalendar.vue';
2
+ import _OEventsList from './OEventsList.vue';
3
+ import _OEventsApply from './OEventsApply.vue';
4
+ import type { App } from 'vue';
5
+
6
+
7
+ const OEventsCalendar = Object.assign(_OEventsCalendar, {
8
+ install(app: App) {
9
+ app.component('OEventsCalendar', _OEventsCalendar);
10
+ },
11
+ });
12
+ const OEventsList = Object.assign(_OEventsList, {
13
+ install(app: App) {
14
+ app.component('OEventsList', _OEventsList);
15
+ },
16
+ });
17
+ const OEventsApply = Object.assign(_OEventsApply, {
18
+ install(app: App) {
19
+ app.component('OEventsApply', _OEventsApply);
20
+ },
21
+ });
22
+
23
+ export { OEventsCalendar, OEventsList, OEventsApply };
24
+ export * from './types';
@@ -0,0 +1,80 @@
1
+ type EventsTextAlignT = 'center' | 'left';
2
+
3
+ export enum EventsStatusT {
4
+ FINISH = 'finish',
5
+ ING = 'ing',
6
+ ALL = ''
7
+ }
8
+
9
+
10
+ export interface EventsItemT {
11
+ name: string; // 事件名称
12
+ date: string | [string, string]; // 事件日期, "YYYY/MM/DD" 或 ["YYYY/MM/DD", "YYYY/MM/DD"]
13
+ location?: string; // 事件地点
14
+ link?: string; // 跳转链接
15
+ linkProps?: {
16
+ [key: string]: string | boolean;
17
+ },
18
+ align?: EventsTextAlignT; // 文本居中方式
19
+ cover?: string; // 事件背景,作为列表卡片形式时使用
20
+ status?: EventsStatusT; // 事件状态,作为列表卡片形式时使用,不传递,组件内部通过date计算得出
21
+ }
22
+
23
+ export type EventsTypeT = 'summit' | 'events' | 'competition' | 'release';
24
+
25
+ export interface CalendarItemT {
26
+ name: string; // 分组标题
27
+ desc?: string; // 分组描述,可选,支持传递html片段
28
+ type: EventsTypeT; // 分组类型,决定颜色和图标
29
+ color?: string; // 分组颜色,可选,如果没有传递由type决定, RGB格式: "255, 0, 0",包括图标背景和事件背景
30
+ icon?: string; // 分组图标,可选,如果没有传递由type决定,白色线性,背景由颜色决定
31
+ data: EventsItemT[];
32
+ isSpanMonth?: boolean; // 是否包含跨月事件,为true时date需要是[string, string]
33
+ }
34
+
35
+ export interface StepItemT {
36
+ title: string; // 标题
37
+ desc?: string; // 描述,支持HTML片段
38
+ img?: string; // 插图
39
+ imgPhone?: string; // 移动端插图
40
+ listTitle?: string; // 列表标题
41
+ list: Array<{
42
+ title: string; // 列表项标题
43
+ desc?: string; // 列表项描述
44
+ }>;
45
+ }
46
+
47
+ // 活动列表
48
+ export interface EventsCardItemT {
49
+ name: string; // 事件名称
50
+ date: string; // 事件日期
51
+ city: string; // 事件城市
52
+ cover?: string; // 卡片封面,如果不传递,组件内部通过date计算得出
53
+ status?: EventsStatusT; // 事件状态,不传递,组件内部通过date计算得出
54
+ link: string; // 跳转链接
55
+ }
56
+
57
+
58
+ // 组件 Props 类型定义
59
+ export interface EventsApplyPropsT {
60
+ steps: StepItemT[];
61
+ }
62
+
63
+ export interface EventsListPropsT {
64
+ data: EventsCardItemT[]; // 活动列表数据
65
+ page: number; // 当前页码
66
+ pageSize: number; // 当前每页个数
67
+ total: number; // 活动总计个数
68
+ }
69
+
70
+ export interface EventsCalendarPropsT {
71
+ data: CalendarItemT[];
72
+ }
73
+
74
+ // OEventList 组件搜索事件参数
75
+ export interface SearchParamsT {
76
+ page: number; // 页码
77
+ pageSize: number; // 每页数量
78
+ status: EventsStatusT; // 状态
79
+ keyword: string; // 关键词
80
+ }
@@ -0,0 +1,9 @@
1
+ /**
2
+ * 比较两个日期字符串的先后顺序
3
+ * @param dateA - 第一个日期字符串
4
+ * @param dateB - 第二个日期字符串
5
+ * @returns 如果 dateA 晚于 dateB 返回 true,否则返回 false
6
+ */
7
+ export const compareDate = (dateA: string | Date, dateB: string | Date): boolean => {
8
+ return new Date(dateA).getTime() > new Date(dateB).getTime();
9
+ };
@@ -0,0 +1,157 @@
1
+ <script setup lang="ts">
2
+ import { ref } from 'vue';
3
+
4
+ import ContentWrapper from '../common/ContentWrapper.vue';
5
+ import HeaderNav from './components/HeaderNav.vue';
6
+ import HeaderContent from './components/HeaderContent.vue';
7
+
8
+ import { type NavT } from './types.ts';
9
+
10
+ import { useTheme } from '@opendesign-plus/composables';
11
+
12
+ const { isDark } = useTheme();
13
+
14
+ export interface OHeaderT {
15
+ (e: 'go-home'): void;
16
+ (e: 'handle-click', val: any): void;
17
+ }
18
+
19
+ const emit = defineEmits<OHeaderT>();
20
+
21
+ const props = defineProps<NavT>();
22
+
23
+ const hoverIndex = ref();
24
+ const hoverId = ref();
25
+ const itemData = ref();
26
+ const itemVisible = ref(false);
27
+
28
+ const goHome = () => {
29
+ emit('go-home');
30
+ };
31
+ const handleMouseenter = (val: any) => {
32
+ itemData.value = val;
33
+ hoverIndex.value = val.id;
34
+ if (val?.children?.length) {
35
+ itemVisible.value = true;
36
+ }
37
+ };
38
+ const handleMouseleave = (val: any) => {
39
+ itemVisible.value = false;
40
+ };
41
+ const handleClick = (val: any) => {
42
+ emit('handle-click', val);
43
+ };
44
+
45
+ const handleMouseenterSub = (val: string) => {
46
+ hoverId.value = val;
47
+ };
48
+ const handleMouseleaveSub = (val: string) => {
49
+ hoverId.value = val;
50
+ };
51
+ </script>
52
+
53
+ <template>
54
+ <div class="header">
55
+ <div class="header-wrap" :class="{ 'header-wrap-dark': isDark }">
56
+ <ContentWrapper class="header-content">
57
+ <!-- logo -->
58
+ <div class="header-logo" @click="goHome">
59
+ <slot name="logo">
60
+ <img v-if="props.logo" class="logo" alt="logo" :src="props.logo" />
61
+ </slot>
62
+ </div>
63
+ <!-- nav -->
64
+ <div class="header-nav">
65
+ <slot name="nav">
66
+ <HeaderNav
67
+ v-if="props.navData"
68
+ :nav-data="props.navData"
69
+ :hoverId="hoverId"
70
+ :active-index="props.activeIndex"
71
+ @handle-mouseenter="handleMouseenter"
72
+ @handle-mouseleave="handleMouseleave"
73
+ @handle-click="handleClick"
74
+ ></HeaderNav>
75
+ </slot>
76
+ </div>
77
+ <!-- toolbar -->
78
+ <div v-if="$slots.toolbar" class="header-toolbar">
79
+ <slot name="toolbar"></slot>
80
+ </div>
81
+ </ContentWrapper>
82
+ </div>
83
+ <div class="header-nav-content">
84
+ <HeaderContent
85
+ :itemData="itemData"
86
+ :item-visible="itemVisible"
87
+ :bg-left="props.bgLeft"
88
+ :bg-right="props.bgRight"
89
+ :community="props.community"
90
+ :hover-index="hoverIndex"
91
+ @handle-mouseenter-sub="handleMouseenterSub"
92
+ @handle-mouseleave-sub="handleMouseleaveSub"
93
+ ></HeaderContent>
94
+ </div>
95
+ </div>
96
+ </template>
97
+
98
+ <style lang="scss" scoped>
99
+ .header {
100
+ --o-header-height: 72px;
101
+ }
102
+ .header-wrap {
103
+ background-color: var(--o-color-fill2);
104
+ position: fixed;
105
+ left: 0;
106
+ right: 0;
107
+ top: 0;
108
+ z-index: 999;
109
+ box-shadow: 0 3px 9px 0 rgba(0, 18, 85, 0.08);
110
+ backdrop-filter: blur(5px);
111
+ height: var(--o-header-height);
112
+ }
113
+ .header-content {
114
+ display: flex;
115
+ align-items: center;
116
+ height: 100%;
117
+ }
118
+ .header-logo {
119
+ height: 100%;
120
+ display: flex;
121
+ align-items: center;
122
+ cursor: pointer;
123
+ margin-right: 24px;
124
+ flex-shrink: 0;
125
+ }
126
+ .logo {
127
+ height: 32px;
128
+ }
129
+ .header-nav {
130
+ flex: 1;
131
+ height: 100%;
132
+ min-width: 0;
133
+ overflow: hidden;
134
+ position: relative;
135
+ }
136
+ .header-toolbar {
137
+ height: 100%;
138
+ position: relative;
139
+ }
140
+ .header-nav-content {
141
+ position: fixed;
142
+ top: var(--o-header-height);
143
+ left: 0;
144
+ right: 0;
145
+ background: var(--o-color-fill2);
146
+ z-index: 998;
147
+ box-shadow: 0 3px 9px 0 rgba(0, 18, 85, 0.08);
148
+ overflow: hidden;
149
+ }
150
+
151
+ .header-wrap-dark {
152
+ box-shadow: 0 3px 9px 0 rgba(var(--o-mixedgray-4), 0.08);
153
+ .header-nav-content {
154
+ box-shadow: 0 3px 9px 0 rgba(var(--o-mixedgray-4), 0.08);
155
+ }
156
+ }
157
+ </style>