ProductsView.vue 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981
  1. <script setup>
  2. import PageHero from "../components/PageHero.vue";
  3. import { Icon } from "@iconify/vue";
  4. import { onBeforeUnmount, onMounted, ref } from "vue";
  5. import platformCover from "../assets/images/pbg.png?url";
  6. import equipmentCover from "../assets/images/case2.png?url";
  7. import productionCover from "../assets/images/sence2.jpg?url";
  8. import qhseCover from "../assets/images/sence3.png?url";
  9. const activeTab = ref("platform");
  10. const tabBarRef = ref(null);
  11. const prefersReducedMotion = () =>
  12. typeof window !== "undefined" &&
  13. window.matchMedia &&
  14. window.matchMedia("(prefers-reduced-motion: reduce)").matches;
  15. const pillars = [
  16. {
  17. key: "platform",
  18. title: "工业互联网平台",
  19. lead: "油气专属底座,打通数采—数据—系统—AI,为全场景应用落地提供稳定支撑。",
  20. icon: "lucide:layers-3",
  21. cover: platformCover,
  22. items: [
  23. {
  24. title: "油气专属底座 · 适配全场景需求",
  25. desc: "聚焦油气领域专属场景,深度适配数采、大模型相关各类应用需求,可灵活承载勘探、开发等全环节数字化场景,贴合行业业务特性。",
  26. image: platformCover,
  27. },
  28. {
  29. title: "高兼容架构 · 打破系统壁垒",
  30. desc: "采用高兼容技术架构,可无缝对接油气领域各类数采设备、现有系统及大模型应用,实现数据互通、系统联动,降低对接成本。",
  31. image: new URL("../assets/images/news3.png", import.meta.url).href,
  32. },
  33. {
  34. title: "灵活可扩展 · 承载多元需求",
  35. desc: "平台底座具备高度灵活性与可扩展性,可根据油气领域不同场景、不同需求,灵活承载数采、大模型相关各类数字化应用落地。",
  36. image: new URL("../assets/images/case7.png", import.meta.url).href,
  37. },
  38. ],
  39. },
  40. {
  41. key: "equipment",
  42. title: "设备管理",
  43. lead: "围绕设备全生命周期,实现台账、巡检维保、预测维护与实时监控预警的闭环管理。",
  44. icon: "lucide:cpu",
  45. cover: equipmentCover,
  46. items: [
  47. {
  48. title: "全维度设备管理 · 筑牢管控基础",
  49. desc: "依托平台底座,承载油气领域设备全生命周期管理场景,兼容各类数采设备,实现设备台账、参数信息统一管控,适配不同型号、不同环节油气设备,高效统筹设备资源。",
  50. image: equipmentCover,
  51. },
  52. {
  53. title: "智慧维保巡检 · 降低运维成本",
  54. desc: "联动巡检、维保全流程,支持自定义巡检计划与维保流程,无缝对接现场数采数据,实现维保巡检标准化、规范化,减少人工冗余,提高运维效率。",
  55. image: new URL("../assets/images/a.png", import.meta.url).href,
  56. },
  57. {
  58. title: "预测性维护 · 防范设备风险",
  59. desc: "借助平台大模型与数采技术支撑,实时分析设备运行数据,精准预测潜在故障并提前触发维护提醒,规避停机风险,降低故障损失。",
  60. image: new URL("../assets/images/b.png", import.meta.url).href,
  61. },
  62. {
  63. title: "实时监控预警 · 保障作业安全",
  64. desc: "实时采集运行参数,实现设备状态可视化监控;异常情况快速预警、精准推送,适配油气现场复杂作业环境,筑牢安全生产防线。",
  65. image: new URL("../assets/images/c.png", import.meta.url).href,
  66. },
  67. ],
  68. },
  69. {
  70. key: "production",
  71. title: "生产管理",
  72. lead: "面向油气生产项目,贯通任务、计划、数据与报表,提升协同效率与过程可控性。",
  73. icon: "lucide:clipboard-check",
  74. cover: productionCover,
  75. items: [
  76. {
  77. title: "项目任务管理 · 联动平台高效统筹",
  78. desc: "承接油气领域生产项目全流程任务管理,无缝对接平台数据体系,实现任务分配、进度跟踪、闭环管理,统筹项目各环节,确保生产任务有序推进。",
  79. image: new URL("../assets/images/d.png", import.meta.url).href,
  80. },
  81. {
  82. title: "计划与实际联动 · 精准把控生产节奏",
  83. desc: "基于平台数采能力,科学制定生产计划;同步对接生产实际数据,自动匹配计划与实际差异,实时调整生产安排,保障流程高效衔接。",
  84. image: new URL("../assets/images/e.png", import.meta.url).href,
  85. },
  86. {
  87. title: "自动报表生成 · 沉淀核心数据资产",
  88. desc: "自动汇总数采数据、生产进度与任务完成情况,一键生成日报、月报等报表,减少人工录入,实现生产数据规范化保存与资产沉淀。",
  89. image: new URL("../assets/images/f.png", import.meta.url).href,
  90. },
  91. {
  92. title: "全流程数智赋能 · 提升生产整体效能",
  93. desc: "深度联动平台数采与数据管理能力,打造生产计划、任务执行、数据统计全链条,减少人工冗余,提升协同效率,实现数字化、智能化转型。",
  94. image: new URL("../assets/images/g.png", import.meta.url).href,
  95. },
  96. ],
  97. },
  98. {
  99. key: "qhse",
  100. title: "QHSE",
  101. lead: "以合规为底线,以数据为驱动,实现风险隐患闭环与作业全过程可控,助力管理升级。",
  102. icon: "lucide:shield-check",
  103. cover: qhseCover,
  104. items: [
  105. {
  106. title: "合规适配 QHSE 体系 · 筑牢安全底线",
  107. desc: "全面贴合 QHSE 管理体系标准,覆盖风险隐患、作业许可、环境管理等核心模块,确保管理流程合规达标,契合油气行业安全环保监管要求,规避合规风险。",
  108. image: new URL("../assets/images/value5.jpg", import.meta.url).href,
  109. },
  110. {
  111. title: "多系统集成联动 · 实现协同管控",
  112. desc: "深度集成设备管理、生产管理等系统,实现数据互通、流程联动,同步设备状态、作业计划等数据,打破系统壁垒,构建安全 + 生产 + 设备一体化管控体系。",
  113. image: new URL("../assets/images/h.png", import.meta.url).href,
  114. },
  115. {
  116. title: "全维度风险防控 · 保障作业安全",
  117. desc: "整合风险隐患管理、应急与健康管理、作业许可与过程监督等功能,借助数采与数据处理能力,实现风险精准识别、隐患闭环治理、作业全过程可控。",
  118. image: new URL("../assets/images/m.png", import.meta.url).href,
  119. },
  120. {
  121. title: "体系化支撑 · 赋能管理升级",
  122. desc: "涵盖目标管理、培训与体系支持等模块,联动平台数据资产,实现目标拆解、培训落地与体系落地,推动从被动应对到主动防控转型,提升管理效能。",
  123. image: new URL("../assets/images/k.png", import.meta.url).href,
  124. },
  125. ],
  126. },
  127. ];
  128. const deliver = [
  129. { title: "SaaS", desc: "开箱即用,按需订阅,快速试点与复制。" },
  130. { title: "私有化", desc: "数据本地化部署,适配企业安全与合规要求。" },
  131. { title: "混合云", desc: "核心数据本地,弹性算力上云,兼顾成本与性能。" },
  132. ];
  133. function headerOffsetPx() {
  134. const rootStyle = getComputedStyle(document.documentElement);
  135. const headerVar = rootStyle.getPropertyValue("--header-h")?.trim();
  136. const headerH = Number.parseFloat(headerVar || "72") || 72;
  137. const tabH = tabBarRef.value?.getBoundingClientRect?.().height || 0;
  138. return Math.round(headerH + tabH + 16);
  139. }
  140. function scrollToTab(key) {
  141. const el = document.getElementById(`product-${key}`);
  142. if (!el) return;
  143. activeTab.value = key;
  144. const offset = headerOffsetPx();
  145. const y = el.getBoundingClientRect().top + window.scrollY - offset;
  146. window.scrollTo({
  147. top: Math.max(0, y),
  148. behavior: prefersReducedMotion() ? "auto" : "smooth",
  149. });
  150. }
  151. let observer = null;
  152. const currentImageIndex = ref({
  153. platform: 0,
  154. equipment: 0,
  155. production: 0,
  156. qhse: 0,
  157. });
  158. const imageTimers = ref({});
  159. const getActiveIndex = (key, totalItems) => {
  160. const rawIndex = currentImageIndex.value[key];
  161. const index = Number.isFinite(rawIndex) ? rawIndex : 0;
  162. if (!totalItems || totalItems <= 0) return 0;
  163. return ((index % totalItems) + totalItems) % totalItems;
  164. };
  165. const getActiveItem = (pillar) => {
  166. const totalItems = pillar?.items?.length || 0;
  167. const index = getActiveIndex(pillar?.key, totalItems);
  168. return pillar?.items?.[index] || null;
  169. };
  170. const getActiveImageSrc = (pillar) =>
  171. getActiveItem(pillar)?.image || pillar.cover;
  172. const getActiveImageAlt = (pillar) =>
  173. getActiveItem(pillar)?.title || pillar.title;
  174. const startImageCarousel = (key, totalItems) => {
  175. if (prefersReducedMotion()) return;
  176. if (!totalItems || totalItems < 2) return;
  177. stopImageCarousel(key);
  178. imageTimers.value[key] = setInterval(() => {
  179. currentImageIndex.value[key] =
  180. (currentImageIndex.value[key] + 1) % totalItems;
  181. }, 4000);
  182. };
  183. const stopImageCarousel = (key) => {
  184. if (imageTimers.value[key]) {
  185. clearInterval(imageTimers.value[key]);
  186. imageTimers.value[key] = null;
  187. }
  188. };
  189. const goToImageSlide = (key, index, totalItems) => {
  190. currentImageIndex.value[key] = index;
  191. startImageCarousel(key, totalItems);
  192. };
  193. onMounted(() => {
  194. const offset = headerOffsetPx();
  195. observer = new IntersectionObserver(
  196. (entries) => {
  197. const visible = entries
  198. .filter((e) => e.isIntersecting)
  199. .sort(
  200. (a, b) => (b.intersectionRatio || 0) - (a.intersectionRatio || 0),
  201. )[0];
  202. if (visible?.target?.id)
  203. activeTab.value = visible.target.id.replace("product-", "");
  204. },
  205. {
  206. root: null,
  207. threshold: [0.25, 0.45, 0.6],
  208. rootMargin: `-${offset}px 0px -55% 0px`,
  209. },
  210. );
  211. for (const p of pillars) {
  212. const el = document.getElementById(`product-${p.key}`);
  213. if (el) observer.observe(el);
  214. }
  215. for (const p of pillars) {
  216. if (currentImageIndex.value[p.key] === undefined)
  217. currentImageIndex.value[p.key] = 0;
  218. startImageCarousel(p.key, p.items?.length || 0);
  219. }
  220. });
  221. onBeforeUnmount(() => {
  222. observer?.disconnect?.();
  223. observer = null;
  224. for (const key of Object.keys(imageTimers.value || {})) {
  225. stopImageCarousel(key);
  226. }
  227. });
  228. </script>
  229. <template>
  230. <div class="productsView">
  231. <PageHero
  232. kicker=""
  233. title="油气行业数智产品矩阵"
  234. subtitle="以工业互联网平台为底座,支撑从勘探开发到生产运营的全场景数字化。"
  235. >
  236. <template #actions>
  237. <RouterLink
  238. class="btn btn-primary"
  239. style="border-radius: 0"
  240. to="/contact"
  241. >获取产品资料</RouterLink
  242. >
  243. </template>
  244. </PageHero>
  245. <section class="section" style="padding-top: 0; margin-top: -10px">
  246. <div>
  247. <div
  248. ref="tabBarRef"
  249. class="tabBar"
  250. role="tablist"
  251. aria-label="产品域导航"
  252. >
  253. <button
  254. v-for="p in pillars"
  255. :key="p.key"
  256. class="tabBtn"
  257. :class="{ 'is-active': activeTab === p.key }"
  258. type="button"
  259. role="tab"
  260. :aria-selected="activeTab === p.key ? 'true' : 'false'"
  261. @click="scrollToTab(p.key)"
  262. >
  263. <span
  264. class="tabText"
  265. style="display: flex; align-items: center; gap: 10px"
  266. ><Icon :icon="p.icon" width="18" height="18" aria-hidden="true" />
  267. <span>{{ p.title }}</span></span
  268. >
  269. </button>
  270. </div>
  271. <div class="productStac container">
  272. <section
  273. v-for="p in pillars"
  274. :id="`product-${p.key}`"
  275. :key="p.key"
  276. class="productSection"
  277. >
  278. <header class="productHead">
  279. <div class="productHeadTop">
  280. <h2 class="productTitle">{{ p.title }}</h2>
  281. </div>
  282. <p class="productLead">{{ p.lead }}</p>
  283. </header>
  284. <div class="tabContentLayout">
  285. <div class="tabContentImage">
  286. <div class="imageCarouselWrapper">
  287. <div class="imageCarouselViewport">
  288. <Transition name="image-carousel-slide" mode="out-in">
  289. <div
  290. v-if="p.items?.length"
  291. :key="`${p.key}-${getActiveIndex(p.key, p.items.length)}`"
  292. class="imageCarouselSlide"
  293. >
  294. <img
  295. :src="getActiveImageSrc(p)"
  296. :alt="getActiveImageAlt(p)"
  297. loading="lazy"
  298. decoding="async"
  299. />
  300. </div>
  301. </Transition>
  302. </div>
  303. </div>
  304. <div class="carouselIndicators">
  305. <button
  306. v-for="(item, idx) in p.items"
  307. :key="idx"
  308. :class="[
  309. 'indicator',
  310. { 'is-active': idx === currentImageIndex[p.key] },
  311. ]"
  312. @click="goToImageSlide(p.key, idx, p.items.length)"
  313. @mouseenter="stopImageCarousel(p.key)"
  314. @mouseleave="startImageCarousel(p.key, p.items.length)"
  315. :aria-label="`第${idx + 1}张图片`"
  316. ></button>
  317. </div>
  318. </div>
  319. <div class="tabContentFeatures">
  320. <div
  321. v-for="(it, idx) in p.items"
  322. :key="it.title"
  323. class="tabFeatureRow"
  324. :class="{
  325. 'is-highlighted': idx === currentImageIndex[p.key],
  326. }"
  327. @mouseenter="
  328. () => {
  329. stopImageCarousel(p.key);
  330. currentImageIndex[p.key] = idx;
  331. }
  332. "
  333. @mouseleave="() => startImageCarousel(p.key, p.items.length)"
  334. >
  335. <div class="tabFeatureMain">
  336. <div class="featureTitle">{{ it.title }}</div>
  337. <div class="muted featureDesc">{{ it.desc }}</div>
  338. </div>
  339. </div>
  340. </div>
  341. </div>
  342. </section>
  343. </div>
  344. </div>
  345. </section>
  346. <section class="section section--soft">
  347. <div class="container">
  348. <div class="sectionHead">
  349. <div>
  350. <h2 class="h2">灵活适配企业 IT 与合规边界</h2>
  351. <div class="muted">
  352. 支持试点快速落地,也支持规模化推广与安全合规的长期运行。
  353. </div>
  354. </div>
  355. </div>
  356. <div class="grid grid-3">
  357. <div
  358. v-for="d in deliver"
  359. :key="d.title"
  360. class="card card-pad deliverCard"
  361. style="border: none; box-shadow: 0px 0px 10px rgba(0, 0, 0, 0.05)"
  362. >
  363. <div class="deliverTitle">{{ d.title }}</div>
  364. <div class="muted">{{ d.desc }}</div>
  365. </div>
  366. </div>
  367. </div>
  368. </section>
  369. </div>
  370. </template>
  371. <style scoped>
  372. .productsView {
  373. --ink: rgba(2, 6, 23, 0.92);
  374. --tab-bg: rgba(255, 255, 255, 0.92);
  375. }
  376. .sectionHead {
  377. display: flex;
  378. align-items: flex-end;
  379. justify-content: space-between;
  380. gap: 18px;
  381. margin-bottom: 16px;
  382. }
  383. .productStac {
  384. display: flex;
  385. width: 100%;
  386. justify-content: center;
  387. align-items: center;
  388. flex-direction: column;
  389. }
  390. :deep(.pageHero) {
  391. background-image: url("../assets/images/bg3.png");
  392. background-size: cover, cover;
  393. background-position: center, center;
  394. background-repeat: no-repeat, no-repeat;
  395. background-blend-mode: overlay;
  396. color: #111;
  397. height: 50vh;
  398. border-bottom: none;
  399. padding: 80px 0;
  400. }
  401. :deep(.pageHero__subtitle) {
  402. color: #4f5055;
  403. }
  404. .tabBar {
  405. position: sticky;
  406. top: var(--header-h);
  407. padding: 0 30px;
  408. z-index: 20;
  409. display: grid;
  410. grid-template-columns: repeat(6, minmax(0, 1fr));
  411. gap: 0px;
  412. background: #eef5f8;
  413. margin: 10px 0 18px;
  414. height: 80px;
  415. border: 0;
  416. border-radius: 0;
  417. }
  418. .tabBtn {
  419. height: 80px;
  420. font-size: 20px;
  421. border: 0;
  422. background: #eef5f8;
  423. box-shadow: var(--tab-shadow);
  424. padding: 0 50px;
  425. display: inline-flex;
  426. align-items: center;
  427. justify-content: center;
  428. gap: 10px;
  429. color: var(--slate-700);
  430. letter-spacing: 0.01em;
  431. cursor: pointer;
  432. transition:
  433. transform 160ms ease,
  434. box-shadow 160ms ease,
  435. color 160ms ease;
  436. }
  437. .tabBtn.is-active {
  438. color: var(--brand-700);
  439. background-color: #fff;
  440. }
  441. .tabBtn:focus-visible {
  442. outline: none;
  443. box-shadow:
  444. 0 0 0 3px rgba(37, 99, 235, 0.24),
  445. var(--tab-shadow-hover);
  446. }
  447. .tabText {
  448. display: inline-block;
  449. white-space: nowrap;
  450. padding-bottom: 10px;
  451. background-image: linear-gradient(currentColor, currentColor);
  452. background-repeat: no-repeat;
  453. background-position: 0 100%;
  454. background-size: 0 2px;
  455. transition: background-size 160ms ease;
  456. }
  457. .tabBtn.is-active .tabText {
  458. background-size: 100% 2px;
  459. }
  460. .productSection {
  461. scroll-margin-top: calc(var(--header-h) + 84px);
  462. border: 0;
  463. border-radius: 0;
  464. background: transparent;
  465. box-shadow: none;
  466. margin-top: 50px;
  467. }
  468. .productHead {
  469. padding: 18px 18px 14px;
  470. text-align: center;
  471. }
  472. .productHeadTop {
  473. display: flex;
  474. align-items: center;
  475. justify-content: center;
  476. gap: 12px;
  477. }
  478. .productTitle {
  479. letter-spacing: -0.02em;
  480. color: var(--ink);
  481. font-size: 30px;
  482. line-height: 1.25;
  483. text-align: center;
  484. }
  485. .productLead {
  486. margin-top: 10px;
  487. max-width: 92ch;
  488. text-align: center;
  489. margin-left: auto;
  490. margin-right: auto;
  491. }
  492. .platformLayout {
  493. margin-top: 40px;
  494. display: grid;
  495. grid-template-columns: 1fr 1fr;
  496. gap: 50px;
  497. align-items: start;
  498. }
  499. .platformImage {
  500. /* border-radius: 16px; */
  501. overflow: hidden;
  502. box-shadow: 0 12px 40px rgba(2, 6, 23, 0.12);
  503. }
  504. .platformImage img {
  505. width: 100%;
  506. height: auto;
  507. object-fit: contain;
  508. display: block;
  509. }
  510. .platformFeatures {
  511. display: grid;
  512. gap: 24px;
  513. }
  514. .platformFeature {
  515. display: grid;
  516. grid-template-columns: 22px 1fr;
  517. align-items: start;
  518. gap: 10px;
  519. padding: 22px;
  520. padding-bottom: 0;
  521. background: #fff;
  522. /* border-radius: 16px; */
  523. box-shadow: 0 8px 24px rgba(2, 6, 23, 0.08);
  524. min-height: 160px;
  525. transition:
  526. transform 180ms ease,
  527. box-shadow 180ms ease;
  528. }
  529. .platformFeature:hover {
  530. transform: translateY(-2px);
  531. box-shadow: 0 12px 32px rgba(2, 6, 23, 0.12);
  532. }
  533. .tabContentLayout {
  534. margin-top: 40px;
  535. display: grid;
  536. grid-template-columns: 1fr 1.2fr;
  537. gap: 50px;
  538. align-items: start;
  539. }
  540. .tabContentImage {
  541. /* border-radius: 16px; */
  542. --carousel-h: 500px;
  543. overflow: hidden;
  544. box-shadow: 0 12px 40px rgba(2, 6, 23, 0.12);
  545. position: sticky;
  546. top: calc(var(--header-h) + 100px);
  547. }
  548. .tabContentImage img {
  549. width: 100%;
  550. height: var(--carousel-h);
  551. object-fit: cover;
  552. display: block;
  553. }
  554. .tabContentFeatures {
  555. display: grid;
  556. gap: 20px;
  557. }
  558. .tabFeatureRow {
  559. padding: 28px;
  560. background: #fff;
  561. /* border-radius: 16px; */
  562. box-shadow: 0 8px 24px rgba(2, 6, 23, 0.08);
  563. min-height: 140px;
  564. transition:
  565. transform 180ms ease,
  566. box-shadow 180ms ease;
  567. }
  568. .tabFeatureRow:hover {
  569. transform: translateY(-2px);
  570. box-shadow: 0 12px 32px rgba(2, 6, 23, 0.12);
  571. }
  572. .featureDot {
  573. width: 18px;
  574. height: 18px;
  575. border-radius: 999px;
  576. background: linear-gradient(180deg, #0766cd, #0d4a9e);
  577. box-shadow:
  578. 0 10px 22px rgba(10, 71, 156, 0.25),
  579. inset 0 0 0 2px rgba(255, 255, 255, 0.85);
  580. margin-top: 4px;
  581. }
  582. .featureMain,
  583. .tabFeatureMain {
  584. min-width: 0;
  585. }
  586. .featureTitle {
  587. letter-spacing: -0.01em;
  588. color: rgba(2, 6, 23, 0.9);
  589. font-size: 22px;
  590. font-weight: 700;
  591. line-height: 1.35;
  592. }
  593. .featureDesc {
  594. margin-top: 10px;
  595. line-height: 1.75;
  596. color: #666666;
  597. text-align: left;
  598. }
  599. .deliverTitle {
  600. /* font-weight: 900; */
  601. margin-bottom: 6px;
  602. }
  603. @media (max-width: 960px) {
  604. .sectionHead {
  605. flex-direction: column;
  606. align-items: flex-start;
  607. gap: 12px;
  608. }
  609. .tabBar {
  610. grid-template-columns: repeat(2, minmax(0, 1fr));
  611. }
  612. .platformLayout,
  613. .tabContentLayout {
  614. grid-template-columns: 1fr;
  615. gap: 30px;
  616. }
  617. .tabContentImage {
  618. position: static;
  619. --carousel-h: 300px;
  620. }
  621. }
  622. @media (max-width: 560px) {
  623. .platformFeature,
  624. .tabFeatureRow {
  625. padding: 20px;
  626. min-height: auto;
  627. }
  628. .featureTitle {
  629. font-size: 18px;
  630. }
  631. .featureDesc {
  632. font-size: 14px;
  633. }
  634. }
  635. @media (prefers-reduced-motion: reduce) {
  636. .tabBtn {
  637. transition: none !important;
  638. transform: none !important;
  639. }
  640. .tabText {
  641. transition: none !important;
  642. }
  643. .platformFeature,
  644. .tabFeatureRow {
  645. transition: none !important;
  646. transform: none !important;
  647. }
  648. }
  649. .imageCarouselWrapper {
  650. position: relative;
  651. display: flex;
  652. align-items: center;
  653. gap: 12px;
  654. }
  655. .imageCarouselViewport {
  656. flex: 1;
  657. overflow: hidden;
  658. position: relative;
  659. height: var(--carousel-h);
  660. }
  661. .imageCarouselSlide {
  662. position: absolute;
  663. top: 0;
  664. left: 0;
  665. width: 100%;
  666. height: 100%;
  667. }
  668. .imageCarouselSlide img {
  669. width: 100%;
  670. height: 100%;
  671. object-fit: cover;
  672. display: block;
  673. }
  674. .imageOverlay {
  675. position: absolute;
  676. bottom: 0;
  677. left: 0;
  678. right: 0;
  679. padding: 20px;
  680. background: linear-gradient(to top, rgba(0, 0, 0, 0.8), transparent);
  681. color: #fff;
  682. }
  683. .overlayText {
  684. font-size: 16px;
  685. font-weight: 600;
  686. line-height: 1.4;
  687. text-shadow: 0 2px 4px rgba(0, 0, 0, 0.5);
  688. }
  689. .image-carousel-slide-enter-active,
  690. .image-carousel-slide-leave-active {
  691. transition: all 0.6s ease;
  692. }
  693. .image-carousel-slide-enter-from {
  694. opacity: 0;
  695. transform: translateY(50px);
  696. }
  697. .image-carousel-slide-leave-to {
  698. opacity: 0;
  699. transform: translateY(-50px);
  700. }
  701. .carouselBtn {
  702. width: 44px;
  703. height: 44px;
  704. border-radius: 50%;
  705. border: 2px solid rgba(7, 102, 205, 0.3);
  706. background: rgba(255, 255, 255, 0.95);
  707. color: #0766cd;
  708. cursor: pointer;
  709. display: flex;
  710. align-items: center;
  711. justify-content: center;
  712. transition: all 0.2s ease;
  713. flex-shrink: 0;
  714. z-index: 2;
  715. }
  716. .carouselBtn:hover {
  717. background: #0766cd;
  718. color: #fff;
  719. border-color: #0766cd;
  720. transform: scale(1.08);
  721. }
  722. .carouselBtn:focus-visible {
  723. outline: none;
  724. box-shadow: 0 0 0 3px rgba(7, 102, 205, 0.24);
  725. }
  726. .carouselIndicators {
  727. display: flex;
  728. flex-direction: column;
  729. justify-content: center;
  730. align-items: end;
  731. gap: 10px;
  732. /* margin-top: 16px; */
  733. padding-right: 10px;
  734. position: relative;
  735. top: -300px;
  736. }
  737. .indicator {
  738. width: 8px;
  739. height: 8px;
  740. /* border-radius: 50%; */
  741. border: none;
  742. background: rgba(7, 102, 205, 0.25);
  743. cursor: pointer;
  744. padding: 0;
  745. transition: all 0.3s ease;
  746. }
  747. .indicator:hover {
  748. background: rgba(7, 102, 205, 0.6);
  749. transform: scale(1.3);
  750. }
  751. .indicator.is-active {
  752. background: #0766cd;
  753. }
  754. .indicator:focus-visible {
  755. outline: none;
  756. box-shadow: 0 0 0 3px rgba(7, 102, 205, 0.24);
  757. }
  758. /* .tabFeatureRow:hover {
  759. transform: translateY(-2px);
  760. box-shadow: 0 12px 32px rgba(2, 6, 23, 0.12);
  761. } */
  762. /* .tabFeatureRow.is-highlighted {
  763. border-left: 3px solid #0766cd;
  764. box-shadow: 0 8px 24px rgba(7, 102, 205, 0.15);
  765. } */
  766. .tabFeatureRow {
  767. padding: 28px;
  768. background: #fff;
  769. /* border-radius: 16px; */
  770. box-shadow: 0 8px 24px rgba(2, 6, 23, 0.08);
  771. min-height: 140px;
  772. transition:
  773. transform 180ms ease,
  774. box-shadow 180ms ease,
  775. border-left 0.3s ease;
  776. border-left: 3px solid transparent;
  777. position: relative;
  778. }
  779. .tabFeatureRow.is-highlighted {
  780. /* border-left: none; */
  781. /* box-shadow:
  782. 0 8px 24px rgba(7, 102, 205, 0.15),
  783. inset 0 0 0 1px rgba(7, 102, 205, 0.1); */
  784. }
  785. .tabFeatureRow.is-highlighted::before {
  786. content: "";
  787. position: absolute;
  788. top: 0;
  789. left: 20px;
  790. width: calc(100% - 40px);
  791. height: 4px;
  792. background: linear-gradient(90deg, #0766cd, #0d4a9e);
  793. border-radius: 2px 2px 0 0;
  794. }
  795. .tabFeatureRow.is-highlighted::after {
  796. content: "";
  797. position: absolute;
  798. bottom: -4px;
  799. left: 50%;
  800. transform: translateX(-50%);
  801. width: 60px;
  802. height: 4px;
  803. background: linear-gradient(90deg, transparent, #0766cd, transparent);
  804. border-radius: 2px;
  805. opacity: 0.6;
  806. }
  807. .tabContentImage {
  808. /* border-radius: 16px; */
  809. --carousel-h: 500px;
  810. overflow: hidden;
  811. box-shadow: 0 12px 40px rgba(2, 6, 23, 0.12);
  812. position: sticky;
  813. top: calc(var(--header-h) + 100px);
  814. }
  815. .imageCarouselWrapper {
  816. position: relative;
  817. display: flex;
  818. align-items: center;
  819. gap: 12px;
  820. }
  821. .imageCarouselViewport {
  822. flex: 1;
  823. overflow: hidden;
  824. position: relative;
  825. }
  826. .imageCarouselSlide {
  827. position: absolute;
  828. top: 0;
  829. left: 0;
  830. width: 100%;
  831. height: 100%;
  832. }
  833. .imageCarouselSlide img {
  834. width: 100%;
  835. height: 500px;
  836. object-fit: cover;
  837. display: block;
  838. }
  839. .carouselIndicators {
  840. position: absolute;
  841. right: 20px;
  842. top: 50%;
  843. transform: translateY(-50%);
  844. display: flex;
  845. flex-direction: column;
  846. justify-content: center;
  847. align-items: center;
  848. gap: 10px;
  849. z-index: 10;
  850. }
  851. .indicator {
  852. width: 8px;
  853. height: 8px;
  854. border-radius: 50%;
  855. border: none;
  856. background: rgba(255, 255, 255, 0.5);
  857. cursor: pointer;
  858. padding: 0;
  859. transition: all 0.3s ease;
  860. }
  861. .indicator:hover {
  862. background: rgba(255, 255, 255, 0.9);
  863. transform: scale(1.3);
  864. }
  865. .indicator.is-active {
  866. background: #014198;
  867. width: 6px;
  868. height: 20px;
  869. border-radius: 3px;
  870. }
  871. .indicator:focus-visible {
  872. outline: none;
  873. box-shadow: 0 0 0 3px rgba(255, 255, 255, 0.5);
  874. }
  875. </style>