Bladeren bron

feat: 新文章页.

peter 2 maanden geleden
bovenliggende
commit
194767dc92

BIN
assets/img/article/About.png


BIN
assets/img/article/Advantage.png


BIN
assets/img/article/Capability.png


BIN
assets/img/article/Compliance.png


+ 1 - 2
components/PcFooter.vue

@@ -119,8 +119,7 @@
                 </nuxt-link>
                 <nuxt-link
                   :to="{
-                    name: 'news-newsName',
-                    params: { newsName: 'About-Us' },
+                    path: '/article/About-Us',
                   }"
                   >About Us
                 </nuxt-link>

+ 1 - 2
components/PcHeader.vue

@@ -459,8 +459,7 @@
           <div class="nav-list-title">
             <nuxt-link
               :to="{
-                name: 'news-newsName',
-                params: { newsName: 'compliance', id: complianceArticle.id },
+                path: 'article/Compliance',
               }"
               target="_self">
               Compliance

+ 486 - 0
pages/article/_type.vue

@@ -0,0 +1,486 @@
+<template>
+  <div class="index com-width-1400 com-margin-auto">
+    <div class="btn-article-menu-switch">
+      <div
+        class="btn flex column center"
+        @click="navMenuVisible = !navMenuVisible">
+        <div class="icon el-icon-notebook-2"></div>
+        <div>Article Menu</div>
+      </div>
+    </div>
+    <section
+      class="nav"
+      v-show="navMenuVisible">
+      <ul class="nav-l">
+        <li
+          v-for="(opt, i) in articleLists"
+          :key="i"
+          @click="getList(i)"
+          :class="{ active: type == i }">
+          <img :src="opt.src" />
+          <p>{{ opt.n }}</p>
+        </li>
+      </ul>
+      <ul class="nav-r">
+        <!-- 大标题,二级目录 -->
+        <li
+          v-for="items in articleLists[type]?.data"
+          :key="items.id">
+          <a
+            class="nav-r-title com-singe-line"
+            :class="{ active: items.hyphensTitle == articleLists[type]?.nav }"
+            :href="`#${items.hyphensTitle}`"
+            @click="smoothScroll($event, items.hyphensTitle)">
+            <i
+              :class="
+                items.show ? 'el-icon-caret-bottom' : 'el-icon-caret-right'
+              "
+              @click.stop="toggle(items)"></i>
+            {{ items.title }}
+          </a>
+          <a
+            v-show="items.show"
+            class="nav-r-anchor com-singe-line"
+            :class="{ active: item == articleLists[type]?.nav }"
+            v-for="(item, i) in items.label_ids"
+            :key="i"
+            :href="`#${item}`"
+            @click="smoothScroll($event, item)"
+            >{{ replaceHyphensWithSpaces(item) }}</a
+          >
+        </li>
+      </ul>
+    </section>
+    <article v-loading="listLoading">
+      <ul>
+        <li
+          v-for="items in articleLists[type]?.data"
+          :key="items.id">
+          <div class="root">
+            <img
+              v-if="items.logo"
+              :src="items.logo" />
+            <div>
+              <a
+                class="root-title"
+                :id="replaceSpacesWithHyphens(items.title)"
+                >{{ items.title }}</a
+              >
+              <p class="root-des">{{ items.description }}</p>
+            </div>
+          </div>
+          <div
+            class="content"
+            v-html="items.content"></div>
+        </li>
+      </ul>
+      <div v-if="!articleLists.length">
+        <el-empty description="No Data"></el-empty>
+      </div>
+    </article>
+  </div>
+</template>
+
+<script>
+import throttle from 'lodash.throttle'
+export default {
+  data() {
+    return {
+      articleLists: [
+        {
+          src: require('@/assets/img/article/About.png'),
+          n: 'About Us',
+          nav: '',
+        },
+        {
+          src: require('@/assets/img/article/Advantage.png'),
+          n: 'Our Advantage',
+          nav: '',
+        },
+        {
+          src: require('@/assets/img/article/Capability.png'),
+          n: 'Capability',
+          nav: '',
+        },
+        {
+          src: require('@/assets/img/article/Compliance.png'),
+          n: 'Compliance',
+          nav: '',
+        },
+      ],
+      type: 1,
+      listLoading: true,
+      navMenuVisible: true,
+      isScrollingToTarget: false,
+    }
+  },
+  watch: {
+    type(newValue, oldValue) {
+      if (process.client && newValue !== oldValue) {
+        window.scrollTo({
+          top: 0,
+          behavior: 'smooth',
+        })
+      }
+    },
+  },
+  created() {
+    let a
+    switch (this.$route.params.type) {
+      case 'About-Us':
+        a = 0
+        break
+
+      case 'Our-Advantage':
+        a = 1
+        break
+
+      case 'Capability':
+        a = 2
+        break
+
+      case 'Compliance':
+        a = 3
+        break
+      default:
+        a = 0
+    }
+    this.getList(a)
+  },
+  mounted() {
+    window.addEventListener('resize', this.$_resizeHandler)
+    window.addEventListener('scroll', this.$_handleScroll) // 监听滚动事件
+  },
+  beforeDestroy() {
+    window.removeEventListener('resize', this.$_resizeHandler)
+    window.removeEventListener('scroll', this.$_handleScroll) // 解绑滚动事件
+  },
+  methods: {
+    getList(type) {
+      this.type = type
+      if (this.articleLists[type].data) {
+        this.articleLists[type].nav =
+          this.articleLists[type].data[0].hyphensTitle
+        window.scrollTo({
+          top: 0,
+          behavior: 'smooth',
+        })
+        return
+      }
+      this.listLoading = true
+
+      this.$axios
+        .post('/uk-api/home/articleList', {
+          type: this.type + 1,
+          page: 1,
+          limit: 1000, // 旧接口迭代,后端不去掉分页
+        })
+        .then(res => {
+          if (res.code === 1) {
+            this.$set(this.articleLists[type], 'data', [])
+            this.articleLists[type].data = res.result.data?.map(item => {
+              this.$set(item, 'show', true)
+              item.hyphensTitle = this.replaceSpacesWithHyphens(item.title)
+              item.content = item.content.replace(
+                /<img([^>]*)>/g,
+                (match, attributes) => {
+                  let newAttributes = attributes
+                  // 如果已有 style 属性
+                  if (attributes.includes('style=')) {
+                    // 提取原有的 style 属性值
+                    const style =
+                      (attributes.match(/style="([^"]*)"/) || [])[1] || ''
+                    // 检查是否已经包含 max-width
+                    if (!style.includes('max-width')) {
+                      // 如果没有 max-width,则添加 max-width: 100%;
+                      newAttributes = attributes.replace(
+                        /style="([^"]*)"/,
+                        `style="$1;max-width:100%;"`
+                      )
+                    }
+                  } else {
+                    // 如果没有 style 属性,则添加 style="max-width:100%;"
+                    newAttributes = `${attributes} style="max-width:100%;"`
+                  }
+                  return `<img${newAttributes}>`
+                }
+              )
+              return item
+            })
+            this.articleLists[type].nav =
+              this.articleLists[type].data[0].hyphensTitle
+            this.listLoading = false
+          }
+        })
+        .catch(() => {
+          this.listLoading = false
+        })
+    },
+    smoothScroll(event, target) {
+      if (process.client) {
+        event && event.preventDefault()
+        const escapedSelector = CSS.escape(target)
+        const element = document.querySelector(`#${escapedSelector}`)
+
+        if (element) {
+          this.isScrollingToTarget = true // 设置标志
+          this.selNav(target) // 立即更新选中状态
+
+          const elementPosition =
+            element.getBoundingClientRect().top + window.scrollY
+          const offsetPosition = elementPosition - 230
+
+          window.scrollTo({
+            top: offsetPosition,
+            behavior: 'smooth',
+          })
+
+          // 等待滚动动画完成后重置标志
+          setTimeout(() => {
+            this.isScrollingToTarget = false
+          }, 1000) // 设置稍长于滚动动画的时间
+        }
+      }
+    },
+    selNav(str) {
+      this.articleLists[this.type].nav = str
+    },
+    toggle(items) {
+      this.$set(items, 'show', !items.show)
+    },
+    replaceSpacesWithHyphens(str) {
+      return str.replace(/\s+/g, '-')
+    },
+    replaceHyphensWithSpaces(str) {
+      return str.replace(/-/g, ' ')
+    },
+    $_resizeHandler: throttle(function () {
+      const width = window.document.body.clientWidth
+      this.navMenuVisible = width >= 1200
+    }, 500),
+
+    $_handleScroll: throttle(function () {
+      // 如果是点击导航触发的滚动,不处理滚动监听
+      if (this.isScrollingToTarget) return
+
+      const anchors = document.querySelectorAll('article a[id]')
+      const scrollTop = window.scrollY
+      const buffer = 230
+
+      let currentAnchor = null
+
+      for (const anchor of anchors) {
+        const rect = anchor.getBoundingClientRect()
+        const elementTop = rect.top + window.scrollY
+
+        if (elementTop <= scrollTop + buffer) {
+          currentAnchor = anchor
+        } else {
+          break
+        }
+      }
+
+      if (currentAnchor) {
+        const anchorId = currentAnchor.getAttribute('id')
+        this.selNav(anchorId)
+      }
+    }, 200),
+  },
+}
+</script>
+
+<style lang="scss" scoped>
+.index {
+  display: flex;
+  flex-direction: row;
+  position: relative;
+
+  .nav {
+    position: sticky;
+    top: 148px;
+    height: fit-content;
+    font-size: 18px;
+
+    ul {
+      display: inline-block;
+    }
+
+    .nav-l {
+      text-align: center;
+
+      li {
+        padding: 10px 0;
+        transition: background-color 0.7s;
+        cursor: pointer;
+
+        &:nth-child(1) {
+          color: #007ad8;
+        }
+
+        &:nth-child(2) {
+          color: #c806ff;
+        }
+
+        &:nth-child(3) {
+          color: #df0000;
+        }
+
+        &:nth-child(4) {
+          color: #ffc300;
+        }
+      }
+
+      .active {
+        &:nth-child(1) {
+          background-color: rgba(0, 122, 216, 0.2);
+        }
+
+        &:nth-child(2) {
+          background-color: rgba(200, 6, 255, 0.2);
+        }
+
+        &:nth-child(3) {
+          background-color: rgba(223, 0, 0, 0.2);
+        }
+
+        &:nth-child(4) {
+          background-color: rgba(255, 195, 0, 0.2);
+        }
+      }
+    }
+
+    .nav-r {
+      width: 180px;
+      background-color: #00213b;
+      vertical-align: top;
+      padding: 10px;
+      min-height: 463px;
+      max-height: calc(100vh - 168px);
+      overflow-y: scroll;
+
+      li {
+        a {
+          color: #fff;
+          display: block;
+          padding: 10px 0;
+        }
+
+        .nav-r-title {
+          font-weight: 700;
+          font-size: 16px;
+        }
+
+        .nav-r-anchor {
+          color: #ddd;
+          padding-left: 20px;
+          padding-right: 10px;
+        }
+
+        .active {
+          background-color: #4da9f2;
+          border-top-right-radius: 20px;
+          border-bottom-right-radius: 20px;
+        }
+      }
+    }
+  }
+
+  article {
+    flex: 1;
+    padding: 80px 20px 20px;
+    max-width: 100%;
+    overflow: auto;
+
+    ul {
+      li {
+        .root {
+          display: flex;
+          flex-direction: row;
+          align-items: center;
+
+          img {
+            width: 100px;
+            height: 100px;
+            margin-right: 20px;
+          }
+
+          .root-title {
+            font-weight: 700;
+            font-size: 20px;
+          }
+
+          .root-des {
+            margin-top: 10px;
+            font-size: 16px;
+            color: #434649;
+          }
+        }
+
+        .content {
+          padding: 10px 0;
+
+          &::after {
+            content: '';
+            display: block;
+            clear: both;
+          }
+        }
+      }
+    }
+  }
+}
+
+.btn-article-menu-switch {
+  position: absolute;
+  left: 2px;
+  top: 12px;
+  display: none;
+
+  @media screen and (max-width: 1200px) {
+    display: block;
+  }
+
+  .btn {
+    cursor: pointer;
+    box-shadow: 1px 1px 3px 2px rgba(#eee, 0.5);
+    background-color: rgba(#fff, 0.7);
+    font-size: 14px;
+    border-radius: 8px;
+    width: 100px;
+    height: 52px;
+    color: #777;
+    border: 1px solid #ccc;
+
+    &:hover {
+      background-color: rgba(#fff, 1);
+    }
+  }
+
+  .icon {
+    font-size: 24px;
+  }
+}
+
+.index section.nav {
+  @media screen and (max-width: 1200px) {
+    box-shadow: 1px 1px 3px 2px rgba(#777, 0.5);
+    z-index: 2;
+    background-color: #fff;
+    position: absolute;
+    top: 72px;
+    left: 0;
+  }
+}
+
+::v-deep .content {
+  * {
+    word-break: break-word !important;
+    white-space: pre-wrap !important;
+  }
+
+  p {
+    font-size: 16px;
+    line-height: 2;
+  }
+}
+</style>