2 Commits 4f45335c00 ... b008be561b

Auteur SHA1 Bericht Datum
  AAA123 b008be561b init: web3-1234567 0.1 4 maanden geleden
  AAA123 1c0a435727 init: web3-1234567 0.1 4 maanden geleden
5 gewijzigde bestanden met toevoegingen van 173 en 275 verwijderingen
  1. 1 1
      package-lock.json
  2. 1 1
      package.json
  3. 12 0
      src/app/tab1/tab1.page.html
  4. 80 146
      src/app/tab1/tab1.page.scss
  5. 79 127
      src/app/tab1/tab1.page.ts

+ 1 - 1
package-lock.json

@@ -34,7 +34,7 @@
         "@angular-eslint/eslint-plugin-template": "^18.0.0",
         "@angular-eslint/schematics": "^18.0.0",
         "@angular-eslint/template-parser": "^18.0.0",
-        "@angular/cli": "^18.0.0",
+        "@angular/cli": "18.2.12",
         "@angular/compiler-cli": "^18.0.0",
         "@angular/language-service": "^18.0.0",
         "@capacitor/cli": "6.2.0",

+ 1 - 1
package.json

@@ -39,7 +39,7 @@
     "@angular-eslint/eslint-plugin-template": "^18.0.0",
     "@angular-eslint/schematics": "^18.0.0",
     "@angular-eslint/template-parser": "^18.0.0",
-    "@angular/cli": "^18.0.0",
+    "@angular/cli": "18.2.12",
     "@angular/compiler-cli": "^18.0.0",
     "@angular/language-service": "^18.0.0",
     "@capacitor/cli": "6.2.0",

+ 12 - 0
src/app/tab1/tab1.page.html

@@ -1,3 +1,8 @@
+<<<<<<< HEAD
+<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no">
+
+=======
+>>>>>>> 4f45335c00c8e925b67f803d52e4c36b1fd571e3
 <ion-header>
   <ion-toolbar color="primary">
     <ion-buttons slot="start">
@@ -59,7 +64,11 @@
     <div class="title">今日推荐医生</div>
     <div class="subtitle">每日更新</div>
   </div>
+<<<<<<< HEAD
+  <div class="scroll-container" id="carousel" (touchstart)="onTouchStart($event)" (touchmove)="onTouchMove($event)" (touchend)="onTouchEnd($event)" [ngStyle]="{ transform: 'translateX(' + currentTranslate + 'px)' }">
+=======
   <div class="scroll-container" id="carousel" (touchstart)="onTouchStart($event)" (touchmove)="onTouchMove($event)" (touchend)="onTouchEnd($event)" [ngStyle]="{ transform: 'translateX(' + currentTranslate + '%)' }">
+>>>>>>> 4f45335c00c8e925b67f803d52e4c36b1fd571e3
     <ng-container *ngFor="let doctor of doctors; let i = index">
       <div class="card">
         <div class="tag">专家</div>
@@ -98,6 +107,8 @@
       <div class="label">{{ item.label }}</div>
     </div>
   </div>
+<<<<<<< HEAD
+=======
 
   <!-- 轮播图区域 -->
   <div class="promotion-carousel" id="promotionCarousel" (touchstart)="onPromotionTouchStart($event)" (touchmove)="onPromotionTouchMove($event)" (touchend)="onPromotionTouchEnd($event)" [ngStyle]="{ transform: 'translateX(' + promotionCurrentTranslate + '%)' }">
@@ -108,6 +119,7 @@
       </div>
     </ng-container>
   </div>
+>>>>>>> 4f45335c00c8e925b67f803d52e4c36b1fd571e3
 </div>
 
 <!-- 浮动操作按钮 -->

+ 80 - 146
src/app/tab1/tab1.page.scss

@@ -167,86 +167,83 @@ ion-toolbar {
   justify-content: space-between;
   align-items: center;
   padding: 16px 0;
+}
 
-  .title {
-    font-size: 16px;
-    font-family: Roboto, "Helvetica Neue", sans-serif;
-    color: #000000;
-    line-height: 21px; /* 确保高度一致 */
-  }
+.header .title {
+  font-size: 16px;
+  font-family: Roboto, "Helvetica Neue", sans-serif;
+  color: #000000;
+  line-height: 21px; /* 确保高度一致 */
+}
 
-  .subtitle {
-    font-size: 14px;
-    color: #999;
-  }
+.header .subtitle {
+  font-size: 14px;
+  color: #999;
 }
 
 .scroll-container {
   display: flex;
-  overflow: hidden;
+  overflow-x: auto;
+  scroll-snap-type: x mandatory;
   position: relative;
-  width: 100%;
+  width: 100%; /* 宽度为100%,确保容器不会过大 */
   min-height: 200px; /* 设置最小高度 */
-  // 如果需要固定高度,可以取消注释下面的行
-  // height: 200px;
+  white-space: nowrap; /* 防止换行 */
+}
 
-  .card {
-    display: flex;
-    flex-direction: column;
-    align-items: center;
-    background: #f8f8f8;
-    border-radius: 10px;
-    padding: 16px;
-    text-align: center;
-    position: absolute;
-    top: 0;
-    left: 0;
-    width: 100%;
-    transition: transform 0.5s ease-in-out;
-    min-height: 200px; /* 设置最小高度 */
-    // 如果需要固定高度,可以取消注释下面的行
-    // height: 200px;
-
-    .tag {
-      font-size: 12px;
-      background: #ff4d4d;
-      color: white;
-      padding: 4px 8px;
-      border-radius: 4px;
-      position: absolute;
-      top: 16px;
-      left: 16px;
-    }
+.card {
+  display: inline-flex; /* 使用inline-flex以允许水平排列 */
+  flex-direction: column;
+  align-items: center;
+  background: #f8f8f8;
+  border-radius: 10px;
+  padding: 16px;
+  text-align: center;
+  scroll-snap-align: start;
+  min-width: 200px; /* 确保卡片有一定的宽度 */
+  transition: transform 0.5s ease-in-out;
+  margin-right: 16px; /* 卡片间距 */
+  position: relative;
+}
 
-    .avatar {
-      width: 80px;
-      height: 80px;
-      border-radius: 50%;
-      margin-bottom: 8px;
-    }
+.card .tag {
+  font-size: 12px;
+  background: #ff4d4d;
+  color: white;
+  padding: 4px 8px;
+  border-radius: 4px;
+  position: absolute;
+  top: 16px;
+  left: 16px;
+}
 
-    .name {
-      font-size: 16px;
-      font-weight: bold;
-      margin-bottom: 4px;
-    }
+.card .avatar {
+  width: 80px;
+  height: 80px;
+  border-radius: 50%;
+  margin-bottom: 8px;
+}
 
-    .specialty {
-      font-size: 14px;
-      color: #666;
-      margin-bottom: 8px;
-    }
+.card .name {
+  font-size: 16px;
+  font-weight: bold;
+  margin-bottom: 4px;
+}
 
-    ion-button {
-      width: 100%; /* 占满卡片的宽度 */
-      margin-top: 8px;
-    }
-  }
+.card .specialty {
+  font-size: 14px;
+  color: #666;
+  margin-bottom: 8px;
+}
+
+.card ion-button {
+  width: 100%; /* 占满卡片的宽度 */
+  margin-top: 8px;
 }
 
 .controls {
   display: flex;
-  justify-content: center;
+  justify-content: space-between;
   margin-top: 16px;
 
   button {
@@ -266,30 +263,32 @@ ion-toolbar {
   }
 }
 
+/* 边界提示 */
+.scroll-container::before,
+.scroll-container::after {
+  content: '';
+  position: absolute;
+  top: 0;
+  bottom: 0;
+  width: 50px; /* 边界提示区域的宽度 */
+  z-index: 1;
+  pointer-events: none; /* 使提示区域不影响触摸事件 */
+}
 
-// 功能区
-.controls {
-  display: flex;
-  justify-content: center;
-  margin-top: 16px;
 
-  button {
-    margin: 0 8px;
-    padding: 8px 16px;
-    border: none;
-    border-radius: 4px;
-    cursor: pointer;
-    background-color: #007bff;
-    color: white;
-    font-size: 14px;
-    transition: background-color 0.3s;
+/* 媒体查询 - 适应小屏幕 */
+@media (max-width: 600px) {
+  .card {
+    min-width: 150px; /* 减小卡片宽度以适应小屏幕 */
+  }
 
-    &:hover {
-      background-color: #0056b3;
-    }
+  .scroll-container::before,
+  .scroll-container::after {
+    width: 30px; /* 减小边界提示区域的宽度 */
   }
 }
 
+// 功能区
 .function-area {
   margin: 16px 0;
   padding: 0 16px;
@@ -330,71 +329,6 @@ ion-toolbar {
   }
 }
 
-.promotion-carousel {
-  display: flex;
-  overflow: hidden;
-  position: relative;
-  width: 100%;
-
-  .poster {
-    display: flex;
-    flex-direction: column;
-    align-items: center;
-    background: #f8f8f8;
-    border-radius: 10px;
-    padding: 16px;
-    text-align: center;
-    position: absolute;
-    top: 0;
-    left: 0;
-    width: 100%;
-    transition: transform 0.5s ease-in-out;
-    min-height: 200px; /* 设置最小高度 */
-
-    img {
-      width: 100%;
-      height: auto;
-      border-radius: 10px;
-      margin-bottom: 8px;
-    }
-
-    .description {
-      font-size: 14px;
-      color: #333;
-      text-align: center;
-    }
-  }
-}
-
-.function-row {
-  display: flex; // 使用 Flexbox 布局
-  justify-content: space-between; // 按空间均匀分布
-  align-items: center; // 垂直居中
-  overflow-x: auto; // 允许水平滚动
-  scroll-snap-type: x mandatory; // 启用滚动捕捉
-  -webkit-overflow-scrolling: touch; // 平滑滚动
-  padding: 16px 0;
-
-  .function-item {
-    flex: 1 0 20%; // 每个项目占 20% 宽度
-    min-width: 0; // 确保项目可以缩小
-    display: flex;
-    flex-direction: column;
-    align-items: center;
-    text-align: center;
-    scroll-snap-align: start; // 滚动时对齐
-
-    .icon {
-      font-size: 24px;
-      margin-bottom: 8px;
-    }
-
-    .label {
-      font-size: 14px;
-      color: #333;
-    }
-  }
-}
 
 .floating-action-button {
   position: fixed;

+ 79 - 127
src/app/tab1/tab1.page.ts

@@ -36,12 +36,6 @@ export class Tab1Page implements AfterViewInit {
     }
   ];
 
-  currentIndex = 0;
-  startX: number = 0; // 初始化 startX
-  endX: number = 0;   // 初始化 endX
-  isDragging: boolean = false; // 标记是否正在拖动
-  currentTranslate: number = 0; // 当前的平移值
-
   // 功能按钮数据
   functionItems1 = [
     { label: '我的病历', icon: 'document-text', route: '/medical-records' },
@@ -59,171 +53,129 @@ export class Tab1Page implements AfterViewInit {
     { label: '健康社区', icon: 'people', route: '/health-community' }
   ];
 
-  // 轮播图数据
-  promotionPosters = [
-    { image: '../../assets/images/promotion1.jpg', description: '健康推广1' },
-    { image: '../../assets/images/promotion2.jpg', description: '健康推广2' },
-    { image: '../../assets/images/promotion3.jpg', description: '健康推广3' },
-    { image: '../../assets/images/promotion4.jpg', description: '健康推广4' },
-    { image: '../../assets/images/promotion5.jpg', description: '健康推广5' }
-  ];
-
-  promotionCurrentIndex = 0;
-  promotionStartX: number = 0; // 初始化 startX
-  promotionEndX: number = 0;   // 初始化 endX
-  isPromotionDragging: boolean = false; // 标记是否正在拖动
-  promotionCurrentTranslate: number = 0; // 当前的平移值
+  currentIndex = 0;
+  startX: number = 0; // 初始化 startX
+  endX: number = 0;   // 初始化 endX
+  isDragging: boolean = false; // 标记是否正在拖动
+  currentTranslate: number = 0; // 当前的平移值
+  cardWidth = 216; // 每个卡片的宽度加上间距 (200 + 16)
+  isCyclic: boolean = true; // 是否启用循环滑动
 
   constructor(private router: Router) {}
 
   ngAfterViewInit() {
-    // 初始化时调用一次以确保初始状态正确
-    this.updateCarousel();
-    this.updatePromotionCarousel();
+    this.updateCarousel(); // 确保初始状态正确
+    this.calculateCardWidth(); // 动态计算卡片宽度
   }
 
-  // 上一张
-  prevSlide() {
-    if (this.currentIndex > 0) {
-      this.currentIndex--;
-    } else {
-      this.currentIndex = this.doctors.length - 1;
+  // 计算卡片宽度
+  calculateCardWidth() {
+    const container = document.getElementById('carousel');
+    if (container && container.children.length > 0) {
+      const firstCard = container.children[0] as HTMLElement;
+      this.cardWidth = firstCard.offsetWidth + 16; // 加上间距
     }
-    this.updateCarousel();
-  }
-
-  // 下一张
-  nextSlide() {
-    if (this.currentIndex < this.doctors.length - 1) {
-      this.currentIndex++;
-    } else {
-      this.currentIndex = 0;
-    }
-    this.updateCarousel();
   }
 
   // 更新轮播位置
-updateCarousel() {
-  this.currentTranslate = -this.currentIndex * 100;
-  this.updateCarouselPosition();
-}
-
-// 更新轮播位置
-updateCarouselPosition() {
-  const container = document.getElementById('carousel');
-  if (container) {
-    container.style.transform = `translateX(${this.currentTranslate}%)`;
-  }
-}
-
-// 触摸开始
-onTouchStart(event: TouchEvent) {
-  this.startX = event.touches[0].clientX;
-  this.isDragging = true;
-}
-
-// 触摸移动
-onTouchMove(event: TouchEvent) {
-  if (this.isDragging) {
-    const touchX = event.touches[0].clientX;
-    const deltaX = touchX - this.startX;
-    this.currentTranslate = -this.currentIndex * 100 + deltaX;
+  updateCarousel() {
+    this.currentTranslate = -this.currentIndex * this.cardWidth;
     this.updateCarouselPosition();
   }
-}
-
-// 触摸结束
-onTouchEnd(event: TouchEvent) {
-  this.endX = event.changedTouches[0].clientX;
-  this.isDragging = false;
-
-  const swipeThreshold = 50; // 滑动阈值
-  const deltaX = this.startX - this.endX;
-
-  if (deltaX > swipeThreshold) {
-    // 向左滑动
-    this.nextSlide();
-  } else if (deltaX < -swipeThreshold) {
-    // 向右滑动
-    this.prevSlide();
-  } else {
-    // 如果滑动距离不够,则恢复到原来的位置
-    this.updateCarousel();
-  }
-}
-  // 导航到指定路由
-  navigateTo(route: string) {
-    this.router.navigate([route]);
-  }
 
-  // 更新轮播图位置
-  updatePromotionCarousel() {
-    this.promotionCurrentTranslate = -this.promotionCurrentIndex * 100;
-    this.updatePromotionCarouselPosition();
+  // 更新轮播位置
+  updateCarouselPosition() {
+    const container = document.getElementById('carousel');
+    if (container) {
+      container.style.transform = `translateX(${this.currentTranslate}px)`;
+    }
   }
 
   // 触摸开始
-  onPromotionTouchStart(event: TouchEvent) {
-    this.promotionStartX = event.touches[0].clientX;
-    this.isPromotionDragging = true;
+  onTouchStart(event: TouchEvent) {
+    // event.preventDefault(); // 防止默认行为
+    this.startX = event.touches[0].clientX;
+    this.isDragging = true;
   }
 
   // 触摸移动
-  onPromotionTouchMove(event: TouchEvent) {
-    if (this.isPromotionDragging) {
+  onTouchMove(event: TouchEvent) {
+    if (this.isDragging) {
+      // event.preventDefault(); // 防止默认行为
       const touchX = event.touches[0].clientX;
-      const deltaX = touchX - this.promotionStartX;
-      this.promotionCurrentTranslate = -this.promotionCurrentIndex * 100 + deltaX;
-      this.updatePromotionCarouselPosition();
+      const deltaX = touchX - this.startX;
+      this.currentTranslate = -this.currentIndex * this.cardWidth + deltaX; // 动态更新平移值
+
+      // 限制 currentTranslate 的值
+      const maxTranslate = -(this.doctors.length - 1) * this.cardWidth;
+      this.currentTranslate = Math.max(Math.min(this.currentTranslate, 0), maxTranslate);
+
+      this.updateCarouselPosition();
     }
   }
 
   // 触摸结束
-  onPromotionTouchEnd(event: TouchEvent) {
-    this.promotionEndX = event.changedTouches[0].clientX;
-    this.isPromotionDragging = false;
+  onTouchEnd(event: TouchEvent) {
+    this.endX = event.changedTouches[0].clientX;
+    this.isDragging = false;
 
     const swipeThreshold = 50; // 滑动阈值
-    const deltaX = this.promotionStartX - this.promotionEndX;
+    const deltaX = this.startX - this.endX;
 
     if (deltaX > swipeThreshold) {
       // 向左滑动
-      this.nextPromotionSlide();
+      this.nextSlide();
     } else if (deltaX < -swipeThreshold) {
       // 向右滑动
-      this.prevPromotionSlide();
+      this.prevSlide();
     } else {
       // 如果滑动距离不够,则恢复到原来的位置
-      this.updatePromotionCarousel();
-    }
-  }
-
-  // 更新轮播图位置
-  updatePromotionCarouselPosition() {
-    const container = document.getElementById('promotionCarousel');
-    if (container) {
-      container.style.transform = `translateX(${this.promotionCurrentTranslate}%)`;
+      this.snapToNearestCard();
     }
   }
 
   // 上一张
-  prevPromotionSlide() {
-    if (this.promotionCurrentIndex > 0) {
-      this.promotionCurrentIndex--;
+  prevSlide() {
+    if (this.isCyclic) {
+      if (this.currentIndex === 0) {
+        this.currentIndex = this.doctors.length - 1;
+      } else {
+        this.currentIndex--;
+      }
     } else {
-      this.promotionCurrentIndex = this.promotionPosters.length - 1;
+      if (this.currentIndex > 0) {
+        this.currentIndex--;
+      }
     }
-    this.updatePromotionCarousel();
+    this.updateCarousel();
   }
 
   // 下一张
-  nextPromotionSlide() {
-    if (this.promotionCurrentIndex < this.promotionPosters.length - 1) {
-      this.promotionCurrentIndex++;
+  nextSlide() {
+    if (this.isCyclic) {
+      if (this.currentIndex === this.doctors.length - 1) {
+        this.currentIndex = 0;
+      } else {
+        this.currentIndex++;
+      }
     } else {
-      this.promotionCurrentIndex = 0;
+      if (this.currentIndex < this.doctors.length - 1) {
+        this.currentIndex++;
+      }
     }
-    this.updatePromotionCarousel();
+    this.updateCarousel();
+  }
+
+  // 快照到最近的卡片
+  snapToNearestCard() {
+    const cardIndex = Math.round(-this.currentTranslate / this.cardWidth);
+    this.currentIndex = Math.max(0, Math.min(cardIndex, this.doctors.length - 1));
+    this.updateCarousel();
+  }
+
+  // 导航到指定路由
+  navigateTo(route: string) {
+    this.router.navigate([route]);
   }
 
   // 发布求医信息