如何实现一个充满科技感的官网(二)
bigegpt 2025-01-09 10:42 6 浏览
背景
在上一篇文章 《如何实现一个充满科技感的官网(一)》 中,我们初步了解了该官网的整体设计,并与大家探讨了它的视觉呈现和用户体验。
我们前期的内部设计偏向简洁,所以开始思考如何提升网站的整体设计感。这些尝试便由此展开。
网站地址:https://infinilabs.com/
如果你对动态背景的实现感兴趣,这篇文章将带你深入探索,揭秘如何从零打造一个兼具美感与功能性的企业官网!
技术选型
- 前端框架:Next.js
- UI 框架:基于 Tailwind CSS
- CSS 样式:Tailwind CSS(快速开发、内置响应式、丰富工具类)
为什么选择 Next.js?
- 兼容团队技术栈:基于 React,便于团队协作。
- SEO 和性能优化:支持服务端渲染(SSR)和静态站点生成(SSG)。
- 路由强大:支持动态路由和文件路由,灵活易用。
- 内置优化:图片优化、国际化、多种性能提升。
- 动态内容支持:博客、新闻等动态场景轻松应对。
- 加载体验佳:用户体验和页面加载速度表现优秀。
动态的背景方案
动态背景可以显著提升视觉吸引力,以下是常用实现方案:
- CSS 动画背景:使用纯 CSS 实现动态背景,通过 @keyframes 配合渐变色、位置移动等属性。
- 动态 Canvas 背景:使用 <canvas> 元素,结合 JavaScript 绘制动态效果,比如粒子系统、波浪效果等。
- 动态视频背景:使用 <video> 元素播放循环视频作为背景。
- WebGL 动态背景:使用 WebGL 库(如 Three.js)渲染 3D 动态背景。
- 动态粒子背景:使用现有的粒子背景库快速实现动态粒子效果。(particles.js 或 tsparticles)
- ......
如何选择?
- 简单需求: 纯 CSS 动画、动态视频背景。
- 复杂交互:Canvas 动画、WebGL 动画(Three.js)。
- 快速实现:使用粒子背景库(particles.js / tsparticles)。
动态背景代码实现
以下示例通过 WebGL 创建了一个动态背景组件,支持 React 和 Tailwind CSS。
- 创建 GlobalBackground.tsx 文件:
"use client";
import dynamic from "next/dynamic";
import { Suspense, useEffect, useState } from "react";
import { Layout } from "./Layout";
const ShaderGradient = dynamic(
() => import("shadergradient").then((mod) => mod.ShaderGradient),
{ ssr: false }
);
const View = dynamic(() => import("./View").then((mod) => mod.View), {
ssr: false,
loading: () => (
<div
className="w-full h-full bg-cover bg-center"
style={{ backgroundImage: "url(/images/loading-bg.png)" }}
></div>
),
});
export default function GlobalBackground() {
const defaultProps: any = {
control: "props",
animate: "on",
brightness: 1.2,
cDistance: 3.6,
cameraZoom: 1,
color1: "#0600B8",
color2: "#9000E3",
color3: "#0B004F",
// embedMode: "off",
envPreset: "city",
// gizmoHelper: "hide",
grain: "off",
lightType: "3d",
reflection: 0.1,
shader: "defaults",
type: "waterPlane",
uSpeed: 0.2,
uTime: 0,
wireframe: false,
zoomOut: false,
toggleAxis: false,
};
const [suspenseWebgl, setSuspenseWebgl] = useState(false);
useEffect(() => {
const canvas = document.createElement("canvas");
const gl =
canvas.getContext("webgl") || canvas.getContext("experimental-webgl");
if (gl) {
// 浏览器支持 WebGL
console.log("The browser does support WebGL");
setSuspenseWebgl(true);
} else {
console.log("The browser does not support WebGL");
// 浏览器不支持 WebGL
}
}, []);
return (
<>
{suspenseWebgl ? (
<Layout>
<View className="w-full h-full">
<Suspense fallback={null}>
<ShaderGradient {...defaultProps} />
</Suspense>
</View>
</Layout>
) : null}
</>
);
}
- 创建 Layout.tsx 文件:
"use client";
import { useRef } from "react";
import dynamic from "next/dynamic";
const Scene = dynamic(() => import("./Scene"), { ssr: false });
const Layout = ({ children }: any) => {
const ref = useRef<any>();
return (
<div
ref={ref}
className="fade-in"
style={{
position: "fixed",
top: 0,
left: 0,
width: "100%",
height: "100%",
zIndex: -1,
overflow: "auto",
touchAction: "auto",
}}
>
{children}
<Scene
style={{
position: "fixed",
top: 0,
left: 0,
width: "100%",
height: "100%",
pointerEvents: "none",
}}
eventSource={ref}
eventPrefix="client"
pixelDensity={1}
pointerEvents="none"
/>
</div>
);
};
export { Layout };
- 创建 Scene.tsx 文件:
"use client";
import { ShaderGradientCanvas } from "shadergradient";
import { Canvas } from "@react-three/fiber";
import { Preload } from "@react-three/drei";
import tunnel from "tunnel-rat";
const r3f = tunnel();
export default function Scene({ ...props }) {
// Everything defined in here will persist between route changes, only children are swapped
return (
<ShaderGradientCanvas {...props}>
{/* @ts-ignore */}
<r3f.Out />
<Preload all />
</ShaderGradientCanvas>
);
}
- 创建 View.tsx 文件:
"use client";
import { forwardRef, Suspense, useImperativeHandle, useRef } from "react";
import {
OrbitControls,
PerspectiveCamera,
View as ViewImpl,
} from "@react-three/drei";
import tunnel from "tunnel-rat";
const r3f = tunnel();
const Three = ({ children }: any) => {
return <r3f.In>{children}</r3f.In>;
};
export const Common = ({ color }: any) => (
<Suspense fallback={null}>
{color && <color attach="background" args={[color]} />}
<ambientLight intensity={0.5} />
<pointLight position={[20, 30, 10]} intensity={1} />
<pointLight position={[-10, -10, -10]} color="blue" />
<PerspectiveCamera makeDefault fov={40} position={[0, 0, 6]} />
</Suspense>
);
const View = forwardRef(({ children, orbit, ...props }: any, ref) => {
const localRef = useRef<any>(null);
useImperativeHandle(ref, () => localRef.current);
return (
<>
<div ref={localRef} {...props} />
<Three>
<ViewImpl track={localRef}>
{children}
{orbit && <OrbitControls />}
</ViewImpl>
</Three>
</>
);
});
View.displayName = "View";
export { View };
- 直接在 app/page.tsx 使用背景组件:
import GlobalBackground from "@/components/GlobalBackground";
export default function Home() {
return (
<>
<GlobalBackground></GlobalBackground>
<div
className="min-h-screen bg-cover bg-center"
style={{ backgroundImage: "url(/svg/bg_n.svg)" }}
>
....
</div>
</>
);
}
- 当然,代码弄好了,要想让代码运行起来,还需要安装一下依赖:
pnpm add @react-three/drei @react-three/fiber shadergradient tunnel-rat
通过这些步骤,你将能够为网站实现高性能、响应式的动态背景效果。根据具体需求调整背景类型和交互复杂度,让你的官网更具吸引力!
效果
具体效果,可以直接在网站上浏览,效果更真实。网站地址:https://infinilabs.com/
分享
如果你也想配置自己的动态效果图,可以前往 shadergradient.co 网站进行自定义设置。完成后,将生成的配置参数复制到 GlobalBackground.tsx 文件的 defaultProps 中,即可实现属于你自己的动态背景效果。
参考
- https://github.com/ruucm/shadergradient
- https://www.shadergradient.co/
- https://infinilabs.com/
福利
INFINI Labs 一直致力于为开发者和企业提供优质的开源工具,提升整个技术生态的活力。除了维护国内最流行的分词器 analysis-ik 和 analysis-pinyin,也在不断推动更多高质量开源产品的诞生。
最近新开源的产品和工具:
- INFINI Framework https://github.com/infinilabs/framework
- INFINI Gateway https://github.com/infinilabs/gateway
- INFINI Console https://github.com/infinilabs/console
- INFINI Agent https://github.com/infinilabs/agent
- INFINI Loadgen https://github.com/infinilabs/loadgen
- INFINI Coco AI https://github.com/infinilabs/coco-app
以上开源软件都可以在 Github 上面找到: https://github.com/infinilabs
希望大家都能给个免费的 Star 支持一下!!!
作者:Rain9,极限科技(INFINI Labs) 高级前端开发工程师。
相关推荐
- 5分钟搭建公网https网页文件服务器,免费权威TLS证书
-
请关注本头条号,每天坚持更新原创干货技术文章。如需学习视频,请在微信搜索公众号“智传网优”直接开始自助视频学习前言本文主要讲解如何快速搭建一个https网页文件服务器,并免费申请权威机构颁发的tls证...
- nginx负载均衡配置(nginx负载均衡配置两个程序副本)
-
Nginx是什么没有听过Nginx?那么一定听过它的“同行”Apache吧!Nginx同Apache一样都是一种WEB服务器。基于REST架构风格,以统一资源描述符(UniformResources...
- 19《Nginx 入门教程》Nginx综合实践
-
今天我们将基于Nginx完成两个比较有用的场景,但是用到的Nginx的配置非常简单。内部Yum源搭建内部Pip源搭建1.实验环境ceph1centos7.6内网ip:172.16....
- Nginx性能调优与优化指南(nginx优化配置大全)
-
Nginx性能调优需要结合服务器硬件资源、业务场景和负载特征进行针对性优化。以下是一些关键优化方向和具体配置示例:一、Nginx配置优化1.进程与连接数优化nginxworker_process...
- C++后端开发必须彻底搞懂Nginx,从原理到实战(高级篇)
-
本文为Nginx实操高级篇。通过配置Nginx配置文件,实现正向代理、反向代理、负载均衡、Nginx缓存、动静分离和高可用Nginx6种功能,并对Nginx的原理作进一步的解析。当需...
- 【Nginx】史上最全的Nginx配置详解
-
Nginx服务器配置中最频繁的部分,代理、缓存和日志定义等绝大多数功能和第三方模块的配置都在这里,http块又包括http全局块和server块。Nginx是非常重要的负载均衡中间件,被广泛应用于大型...
- 【Nginx】Nginx 4种常见配置实例(nginx基本配置与参数说明)
-
本文主要介绍nginx4种常见的配置实例。Nginx实现反向代理;Nginx实现负载均衡;Nginx实现动静分离;Nginx实现高可用集群;Nginx4种常见配置实例如下:一、Nginx反向代理配...
- 使用nginx+allure管理自动化测试报告
-
allure在自动化测试中经常用来生成漂亮的报告,但是网上及官网上给出的例子都仅仅是针对单个测试用例文件的形式介绍的,实际使用中,自动化测试往往需要包含不止一个产品或项目,本文介绍如何使用nginx+...
- nginx配置文件详解(nginx配置文件详解高清版)
-
Nginx是一个强大的免费开源的HTTP服务器和反向代理服务器。在Web开发项目中,nginx常用作为静态文件服务器处理静态文件,并负责将动态请求转发至应用服务器(如Django,Flask,et...
- SpringCloud Eureka-服务注册与发现
-
1.Eureka介绍1.1学习Eureka前的说明目前主流的服务注册&发现的组件是Nacos,但是Eureka作为老牌经典的服务注册&发现技术还是有必要学习一下,原因:(1)一些早期的分布式微服...
- 微服务 Spring Cloud 实战 Eureka+Gateway+Feign+Hystrix
-
前言我所在项目组刚接到一个微服务改造需求,技术选型为SpringCloud,具体需求是把部分项目使用SpringCloud技术进行重构。本篇文章中介绍了Eureka、Gateway、Fe...
- 深度剖析 Spring Cloud Eureka 底层实现原理
-
你作为一名互联网大厂后端技术开发人员,在构建分布式系统时,是不是常常为服务的注册与发现而头疼?你是否好奇,像SpringCloudEureka这样被广泛使用的组件,它的底层实现原理到底是怎样的...
- 热爱生活,喜欢折腾。(很热爱生活)
-
原文是stackoverflow的一则高票回答,原文链接可能之前也有人翻译过,但是刚好自己也有疑惑,所以搬运一下,个人水平有限所以可能翻译存在误差,欢迎指正(如侵删)。尽管classmethod和st...
- GDB调试的高级技巧(详细描述gdb调试程序的全过程)
-
GDB是我们平时调试c/c++程序的利器,查起复杂的bug问题,比打印大法要好得多,但是也不得不说,gdb在默认情况下用起来并不是很好用,最近学习到几个高级点的技巧,分享下:一美化打印先上个例子...
- Arduino 实例(二十三)Arduino 给Python 编译器发送信息
-
1首先Python需要安装Pyserial库,在命令提示符中输入pipintallpyserial若是遇到提示‘pip‘不是内部或外部命令,也不是可运行的程序或批处理文件,则需要设置环境变...
- 一周热门
- 最近发表
- 标签列表
-
- mybatiscollection (79)
- mqtt服务器 (88)
- keyerror (78)
- c#map (65)
- resize函数 (64)
- xftp6 (83)
- bt搜索 (75)
- c#var (76)
- mybatis大于等于 (64)
- xcode-select (66)
- mysql授权 (74)
- 下载测试 (70)
- skip-name-resolve (63)
- linuxlink (65)
- pythonwget (67)
- logstashinput (65)
- hadoop端口 (65)
- vue阻止冒泡 (67)
- oracle时间戳转换日期 (64)
- jquery跨域 (68)
- php写入文件 (73)
- kafkatools (66)
- mysql导出数据库 (66)
- jquery鼠标移入移出 (71)
- 取小数点后两位的函数 (73)