新闻中心
在本地运行CodePen项目:解决外部依赖与模块导入问题

本文旨在提供一个将codepen项目成功部署到本地环境的专业教程,重点解决直接复制粘贴代码时遇到的外部依赖、css预处理器和j*ascript模块导入问题。我们将以一个mediapipe facelandmarker项目为例,详细讲解如何正确引入cdn资源、处理es模块导入,并提供完整的本地运行代码示例,确保项目功能在本地浏览器中正常运行。
引言:CodePen项目本地化挑战
CodePen等在线代码编辑平台为前端开发提供了极大的便利,用户可以快速编写、测试和分享HTML、CSS和J*aScript代码。然而,当尝试将这些项目直接复制到本地环境运行时,开发者常会遇到功能不正常的问题。这通常是由于以下几个原因:
- 外部依赖未正确引入:CodePen项目可能通过CDN链接引入了外部库(如Material Components Web、MediaPipe等),这些链接在本地环境中需要被精确复制。
- CSS预处理器:CodePen支持Sass、Less等CSS预处理器。项目中的CSS代码可能使用了@use、@import等预处理器特性,这些在本地环境中未经编译时将无法识别。
- J*aScript模块导入:现代J*aScript项目通常采用ES模块(ESM)规范进行模块化开发。CodePen可能隐式处理了模块导入路径,但在本地,尤其是通过file://协议直接打开HTML文件时,需要确保import语句指向正确的、可访问的模块URL。
- 资源路径问题:图片、字体或其他静态资源的相对路径在本地文件系统中可能与CodePen服务器环境不同。
本教程将以一个使用MediaPipe FaceLandmarker进行人脸关键点检测的CodePen项目为例,详细演示如何解决这些问题,使其在本地环境成功运行。
准备工作:创建本地项目文件
首先,在你的本地计算机上创建一个新的文件夹,例如my-codepen-project。在这个文件夹中,创建一个名为index.html的文件。所有代码都将整合到这个文件中,以便于演示和理解。
核心步骤:代码整合与依赖处理
要使CodePen项目在本地正常运行,我们需要将HTML、CSS和J*aScript代码整合到index.html中,并特别注意处理所有外部依赖和模块导入。
1. HTML结构与元数据
项目的基本HTML结构应包含必要的head和body标签,以及元数据设置,如字符集、视口配置等。这些是现代网页开发的标准实践。
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta http-equiv="Cache-control" content="no-cache, no-store, must-revalidate"> <meta http-equiv="Pragma" content="no-cache"> <meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no"> <title>Face Landmarker</title> <!-- 其他CSS和JS链接将在此处添加 --> </head> <body> <!-- 页面内容将在此处添加 --> </body> </html>
2. CSS样式处理
CodePen中的CSS通常会直接应用,但如果使用了预处理器(如Sass的@use),在本地需要进行调整。对于本例,CSS可以直接嵌入到
Visla
AI视频生成器,快速轻松地将您的想法转化为视觉上令人惊叹的视频。
100
查看详情
<head>
<!-- ...其他head内容... -->
<style>
/* CodePen中的CSS内容,去除预处理器指令(如@use "@material";)
并确保所有样式都直接可解析。
本例中的CSS是纯CSS,可以直接复制。
*/
body {
font-family: helvetica, arial, sans-serif;
margin: 2em;
color: #3d3d3d;
--mdc-theme-primary: #007f8b;
--mdc-theme-on-primary: #f1f3f4;
}
h1 {
font-style: italic;
color: #ff6f00;
color: #007f8b;
}
h2 {
clear: both;
}
em {
font-weight: bold;
}
video {
clear: both;
display: block;
transform: rotateY(180deg);
-webkit-transform: rotateY(180deg);
-moz-transform: rotateY(180deg);
}
section {
opacity: 1;
transition: opacity 500ms ease-in-out;
}
header,
footer {
clear: both;
}
.removed {
display: none;
}
.invisible {
opacity: 0.2;
}
.note {
font-style: italic;
font-size: 130%;
}
.videoView,
.detectOnClick,
.blend-shapes {
position: relative;
float: left;
width: 48%;
margin: 2% 1%;
cursor: pointer;
}
.videoView p,
.detectOnClick p {
position: absolute;
padding: 5px;
background-color: #007f8b;
color: #fff;
border: 1px dashed rgba(255, 255, 255, 0.7);
z-index: 2;
font-size: 12px;
margin: 0;
}
.highlighter {
background: rgba(0, 255, 0, 0.25);
border: 1px dashed #fff;
z-index: 1;
position: absolute;
}
.canvas {
z-index: 1;
position: absolute;
pointer-events: none;
}
.output_canvas {
transform: rotateY(180deg);
-webkit-transform: rotateY(180deg);
-moz-transform: rotateY(180deg);
}
.detectOnClick {
z-index: 0;
}
.detectOnClick img {
width: 100%;
}
.blend-shapes-item {
display: flex;
align-items: center;
height: 20px;
}
.blend-shapes-label {
display: flex;
width: 120px;
justify-content: flex-end;
align-items: center;
margin-right: 4px;
}
.blend-shapes-value {
display: flex;
height: 16px;
align-items: center;
background-color: #007f8b;
}
</style>
<!-- 引入Material Components Web CSS -->
<link href="https://unpkg.com/material-components-web@latest/dist/material-components-web.min.css" rel="stylesheet">
<!-- 引入Material Components Web JS (在head中加载以确保样式正确应用) -->
<script src="https://unpkg.com/material-components-web@latest/dist/material-components-web.min.js"></script>
</head>3. J*aScript模块导入与CDN资源
这是最关键的部分。CodePen中的J*aScript代码可能使用了ES模块(import语句),并且依赖于通过CDN加载的MediaPipe库。我们需要确保:
- <script>标签使用type="module"属性。</script>
- 所有import语句指向正确的、可访问的CDN URL。
- MediaPipe模型的资产路径也是正确的CDN URL。
特别注意,原始问答中提供的[email protected]链接是Cloudflare的邮件保护机制,在本地直接复制会导致错误。我们需要将其替换为实际的CDN路径。根据CodePen项目,MediaPipe tasks-vision的正确CDN路径应为 https://cdn.jsdelivr.net/npm/@mediapipe/tasks-vision@0.10.0。
<body>
<h1>Face landmark detection using the MediaPipe FaceLandmarker task</h1>
<section id="demos" class="invisible">
<h2>Demo: Detecting Images</h2>
<p><b>Click on an image below</b> to see the key landmarks of the face.</p>
<div class="detectOnClick">
@@##@@
</div>
<div class="blend-shapes">
<ul class="blend-shapes-list" id="image-blend-shapes"></ul>
</div>
<h2>Demo: Webcam continuous face landmarks detection</h2>
<p>Hold your face in front of your webcam to get real-time face landmarker detection.</br>Click <b>enable webcam</b> below and grant access to the webcam if prompted.</p>
<div id="liveView" class="videoView">
<button id="webcamButton" class="mdc-button mdc-button--raised">
<span class="mdc-button__ripple"></span>
<span class="mdc-button__label">ENABLE WEBCAM</span>
</button>
<div style="position: relative;">
<video id="webcam" style="position: abso" autoplay playsinline></video>
<canvas class="output_canvas" id="output_canvas" style="position: absolute; left: 0px; top: 0px;"></canvas>
</div>
</div>
<div class="blend-shapes">
<ul class="blend-shapes-list" id="video-blend-shapes"></ul>
</div>
</section>
<script type="module">
// 修正后的MediaPipe tasks-vision CDN导入路径
import vision from "https://cdn.jsdelivr.net/npm/@mediapipe/tasks-vision@0.10.0";
const { FaceLandmarker, FilesetResolver, DrawingUtils } = vision;
const demosSection = document.getElementById("demos");
const imageBlendShapes = document.getElementById("image-blend-shapes");
const videoBlendShapes = document.getElementById("video-blend-shapes");
let faceLandmarker;
let runningMode = "IMAGE"; // 初始运行模式
let enableWebcamButton; // HTMLButtonElement;
let webcamRunning = false; // Boolean = false;
const videoWidth = 480;
// Before we can use HandLandmarker class we must wait for it to finish
// loading. Machine Learning models can be large and take a moment to
// get everything needed to run.
async function runDemo() {
// 修正FilesetResolver的WASM路径
const filesetResolver = await FilesetResolver.forVisionTasks(
"https://cdn.jsdelivr.net/npm/@mediapipe/tasks-vision@0.10.0/wasm"
);
faceLandmarker = await FaceLandmarker.createFromOptions(filesetResolver, {
baseOptions: {
modelAssetPath: `https://storage.googleapis.com/mediapipe-models/face_landmarker/face_landmarker/float16/1/face_landmarker.task`,
delegate: "GPU"
},
outputFaceBlendshapes: true,
runningMode,
numFaces: 1
});
demosSection.classList.remove("invisible");
}
runDemo();
/********************************************************************
// Demo 1: Grab a bunch of images from the page and detection them
// upon click.
********************************************************************/
// In this demo, we h*e put all our clickable images in divs with the
// CSS class 'detectionOnClick'. Lets get all the elements that h*e
// this class.
const imageContainers = document.getElementsByClassName("detectOnClick");
// Now let's go through all of these and add a click event listener.
for (let i = 0; i < imageContainers.length; i++) {
// Add event listener to the child element whichis the img element.
imageContainers[i].children[0].addEventListener("click", handleClick);
}
// When an image is clicked, let's detect it and display results!
async function handleClick(event) {
if (!faceLandmarker) {
console.log("Wait for faceLandmarker to load before clicking!");
return;
}
if (runningMode === "VIDEO") {
runningMode = "IMAGE";
await faceLandmarker.setOptions({ runningMode });
}
// Remove all landmarks drawed before
const allCanvas = event.target.parentNode.getElementsByClassName("canvas");
for (var i = allCanvas.length - 1; i >= 0; i--) {
const n = allCanvas[i];
n.parentNode.removeChild(n);
}
// We can call faceLandmarker.detect as many times as we like with
// different image data each time. This returns a promise
// which we wait to complete and then call a function to
// print out the results of the prediction.
const faceLandmarkerResult = faceLandmarker.detect(event.target);
const canvas = document.createElement("canvas");
canvas.setAttribute("class", "canvas");
canvas.setAttribute("width", event.target.naturalWidth + "px");
canvas.setAttribute("height", event.target.naturalHeight + "px");
canvas.style.left = "0px";
canvas.style.top = "0px";
canvas.style.width = `${event.target.width}px`;
canvas.style.height = `${event.target.height}px`;
event.target.parentNode.appendChild(canvas);
const ctx = canvas.getContext("2d");
const drawingUtils = new DrawingUtils(ctx);
for (const landmarks of faceLandmarkerResult.faceLandmarks) {
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_TESSELATION,
{ color: "#C0C0C070", lineWidth: 1 }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_RIGHT_EYE,
{ color: "#FF3030" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_RIGHT_EYEBROW,
{ color: "#FF3030" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_LEFT_EYE,
{ color: "#30FF30" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_LEFT_EYEBROW,
{ color: "#30FF30" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_FACE_OVAL,
{ color: "#E0E0E0" }
);
drawingUtils.drawConnectors(landmarks, FaceLandmarker.FACE_LANDMARKS_LIPS, {
color: "#E0E0E0"
});
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_RIGHT_IRIS,
{ color: "#FF3030" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_LEFT_IRIS,
{ color: "#30FF30" }
);
}
drawBlendShapes(imageBlendShapes, faceLandmarkerResult.faceBlendshapes);
}
/********************************************************************
// Demo 2: Continuously grab image from webcam stream and detect it.
********************************************************************/
const video = document.getElementById("webcam");
const canvasElement = document
以上就是在本地运行CodePen项目:解决外部依赖与模块导入问题的详细内容,更多请关注其它相关文章!
# javascript
# 潍城区网站推广服务部门
# 秦淮区公司网站优化方案
# 一月 营销推广
# 舒兰网站建设
# 网站架设 seo
# 复选框
# 如何实现
# 正常运行
# 创建一个
# 使用了
# 将以
# 为例
# 可以直接
# 弹出
# a
# css
# java
# html
# js
# 前端
# node
# go
# 计算机
# 处理器
# npm
# seo
# 浏览器
# 将在
# 网站优化推广话术
# seo优化是什么 discuz
# 托里关键词排名
# 抖音综合营销怎么做推广
# 肇庆网站建设批发商城
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
msn官网入口地址手机版 msn官方网站手机最新链接
css卡片内容溢出如何处理_使用overflow隐藏或scroll显示内容
QQ网页版官方账号入口 QQ网页版网页版登录指南
LINUX的perf命令入门_LINUX官方性能分析工具的使用与解读
AO3最新可访问网址 Archive of Our Own官方在线入口
《刺客信条4:黑旗》重制版新细节曝光:无缝加载 地图更细致!
在J*a里如何理解依赖关系的方向_依赖方向在模块结构中的作用
C++的std::mdspan是什么_C++23中用于操作多维数组的非拥有视图
Composer的 archive 命令怎么用_快速打包你的PHP项目及其Composer依赖
J*aScript DOM操作:高效清空列表元素的策略与实践
谷歌浏览器浏览体验优化_谷歌浏览器新版直连永久可用提示
C++如何实现异步操作_C++11使用std::future和std::async进行异步编程
ACG动漫手机版官网入口 手机ACG动漫APP在线观看正版
C++如何进行游戏物理模拟_使用Box2D库为C++游戏添加2D物理效果
微信语音通话掉线如何解决 微信语音通话稳定优化方法
Python Socket多播通信中指定源IP地址的实践指南
夸克AO3官网入口_AO3镜像网站2025推荐
文心一言怎样用插件调度API数据_文心一言用插件调度API数据【API调用】
Basecamp怎样用留言钉固定重点_Basecamp用留言钉固定重点【重点标记】
mysql通配符支持数字匹配吗_mysql通配符能否用于数字匹配的解析
解决 MongoDB 聚合查询中对象数组 _id 匹配问题
为什么我的微信朋友圈看不到别人的更新_微信朋友圈更新显示异常解决方法
微博网页版怎么开启两步验证_微博网页版账号安全两步验证设置方法
腾讯视频怎么使用多账号家庭管理_腾讯视频家庭多账号统一管理与权限分配教程
Golang并发任务中错误如何聚合_Golang goroutine error收集方式
AO3最新镜像入口 Archive of Our Own官方平台访问
UC浏览器如何安装插件 UC浏览器添加扩展程序详细教程【进阶】
在Go开发中优雅管理ListenAndServe进程:GoSublime集成方案
在Socket.IO连接中实现Access Token自动更新与动态重连
Yandex搜索引擎一键访问入口_俄罗斯Yandex官网免登录
漫蛙网页登录入口 漫蛙漫画官方授权网址
知乎APP怎么管理已购盐选内容_知乎APP盐选内容购买记录与查看方法
J*aScript类型检查_j*ascript代码规范
天猫2025双十一0点秒杀攻略 天猫爆款抢购时间
微信群消息显示延迟如何解决 微信群消息刷新优化方法
NVIDIA股价11月重挫12%:下月有望好转 但难回5万亿美元巅峰
Composer如何处理Git子模块(submodule)依赖_Composer与Git Submodule的对比与选择
神庙逃亡小游戏在线玩 神庙逃亡小游戏入口
解决深度学习模型训练初期异常高损失与完美验证准确率问题
J*aScript Promise链中如何正确终止后续.then执行并处理错误
Golang如何使用net/url解析URL_Golang URL解析与处理方法
css绝对定位元素脱离父容器怎么办_确保父元素position非static
TikTok评论显示延迟如何处理 TikTok评论刷新优化方法
谷歌google账号注册详细步骤 谷歌账号注册官方教程
ACG动漫视频网入口 ACG动漫*免费正版观看地址
我的世界官方游戏入口 我的世界官网平台直达链接
如何在更新Composer依赖后自动运行测试_使用post-update-cmd钩子触发PHPUnit
如何在复杂的电商平台中优雅地管理共享资源并确保正确重定向,使用spryker-shop/resource-share-page模块助你一臂之力
Win11怎么开启卓越性能模式 Win11电源选项启用高性能释放硬件潜力【方法】
Angular中父组件异步更新子组件复选框状态的实践指南


2025-11-10
浏览次数:次
返回列表
);
faceLandmarker = await FaceLandmarker.createFromOptions(filesetResolver, {
baseOptions: {
modelAssetPath: `https://storage.googleapis.com/mediapipe-models/face_landmarker/face_landmarker/float16/1/face_landmarker.task`,
delegate: "GPU"
},
outputFaceBlendshapes: true,
runningMode,
numFaces: 1
});
demosSection.classList.remove("invisible");
}
runDemo();
/********************************************************************
// Demo 1: Grab a bunch of images from the page and detection them
// upon click.
********************************************************************/
// In this demo, we h*e put all our clickable images in divs with the
// CSS class 'detectionOnClick'. Lets get all the elements that h*e
// this class.
const imageContainers = document.getElementsByClassName("detectOnClick");
// Now let's go through all of these and add a click event listener.
for (let i = 0; i < imageContainers.length; i++) {
// Add event listener to the child element whichis the img element.
imageContainers[i].children[0].addEventListener("click", handleClick);
}
// When an image is clicked, let's detect it and display results!
async function handleClick(event) {
if (!faceLandmarker) {
console.log("Wait for faceLandmarker to load before clicking!");
return;
}
if (runningMode === "VIDEO") {
runningMode = "IMAGE";
await faceLandmarker.setOptions({ runningMode });
}
// Remove all landmarks drawed before
const allCanvas = event.target.parentNode.getElementsByClassName("canvas");
for (var i = allCanvas.length - 1; i >= 0; i--) {
const n = allCanvas[i];
n.parentNode.removeChild(n);
}
// We can call faceLandmarker.detect as many times as we like with
// different image data each time. This returns a promise
// which we wait to complete and then call a function to
// print out the results of the prediction.
const faceLandmarkerResult = faceLandmarker.detect(event.target);
const canvas = document.createElement("canvas");
canvas.setAttribute("class", "canvas");
canvas.setAttribute("width", event.target.naturalWidth + "px");
canvas.setAttribute("height", event.target.naturalHeight + "px");
canvas.style.left = "0px";
canvas.style.top = "0px";
canvas.style.width = `${event.target.width}px`;
canvas.style.height = `${event.target.height}px`;
event.target.parentNode.appendChild(canvas);
const ctx = canvas.getContext("2d");
const drawingUtils = new DrawingUtils(ctx);
for (const landmarks of faceLandmarkerResult.faceLandmarks) {
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_TESSELATION,
{ color: "#C0C0C070", lineWidth: 1 }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_RIGHT_EYE,
{ color: "#FF3030" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_RIGHT_EYEBROW,
{ color: "#FF3030" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_LEFT_EYE,
{ color: "#30FF30" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_LEFT_EYEBROW,
{ color: "#30FF30" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_FACE_OVAL,
{ color: "#E0E0E0" }
);
drawingUtils.drawConnectors(landmarks, FaceLandmarker.FACE_LANDMARKS_LIPS, {
color: "#E0E0E0"
});
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_RIGHT_IRIS,
{ color: "#FF3030" }
);
drawingUtils.drawConnectors(
landmarks,
FaceLandmarker.FACE_LANDMARKS_LEFT_IRIS,
{ color: "#30FF30" }
);
}
drawBlendShapes(imageBlendShapes, faceLandmarkerResult.faceBlendshapes);
}
/********************************************************************
// Demo 2: Continuously grab image from webcam stream and detect it.
********************************************************************/
const video = document.getElementById("webcam");
const canvasElement = document