H5 洋火人科目三和GitHub获取仓库点星星的用户列表发生了艺术的碰撞 ...

打印 上一主题 下一主题

主题 988|帖子 988|积分 2964

先看结果,代码写的比较乱,有待优化
  结果

https://linyisonger.github.io/H5.Examples/?name=./089.%E7%9C%8B%E6%98%9F%E6%98%9F%E7%9A%84%E8%88%9E%E8%80%85.html

思绪

看起来很简朴,实则也不是很难,就是必要思绪要打开。
一开始的流程思绪是
思绪一


  • 通过视频获取骨骼节点动画 ✔
  • 利用AI文生图+骨骼节点生成人物信息 ❌ 结果不达预期
  • 确定人物头部位置+序列帧动画
思绪二


  • 通过视频获取骨骼节点动画 ✔
  • 通过骨骼动画进行canvas渲染,节点毗连从而打到洋火人的结果。✔
  • 确定人物头部位置+序列帧动画 ❌ 画布太大无法渲染一张图
  • 确定人物头部位置+序列帧动画 + JSON存储 ✔
实现


  • 通过视频播放+requestAnimationFrame获取每帧图片
  • 通过@tensorflow/tfjs+@tensorflow-models/posenet来获取图片骨骼节点
  • 通过canvas进行骨骼毗连
这又是一篇新的内容,AI方面不是很了解,只是看着教程做的。
https://linyisonger.github.io/H5.Examples/?name=./090.%E7%81%AB%E6%9F%B4%E4%BA%BA%E7%94%9F%E6%88%90%E5%99%A8.html

上传视频后输出的JSON文件是这个示例所必要的。
内里包含每一帧的洋火人Base64图片,头像应该放置的位置。
代码

获取GitHub仓库点星星的用户列表
   ⚠ 当然这不是很好的写法,一旦出现报错就是死循环
  1. /**
  2. * 获取star的用户 默认30一页
  3. * @author          linyisonger
  4. * @date          2025-02-18
  5. */
  6. async function getStargazers(page = 1) {
  7.     const result = await fetch(`https://api.github.com/repos/linyisonger/H5.Examples/stargazers?page=${page}`)
  8.     return await result.json()
  9. }
  10. /**
  11. * 获取所有star的用户
  12. * @author          linyisonger
  13. * @date          2025-02-18
  14. */
  15. async function getAllStargazers(page = 1, users = []) {
  16.     let stargazers = await getStargazers(page)
  17.     users = users.concat(stargazers)
  18.     if (stargazers.length < 30) return users
  19.     return await getAllStargazers(page + 1, users)
  20. }
复制代码
其他的感觉没什么重点
完备代码

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4.     <meta charset="UTF-8">
  5.     <meta http-equiv="X-UA-Compatible" content="IE=edge">
  6.     <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7.     <link rel="stylesheet" href="./assets/global.css">
  8.     <style>
  9.         #container {
  10.             display: flex;
  11.             justify-content: center;
  12.             align-items: center;
  13.             flex-wrap: wrap;
  14.             height: 100vh;
  15.             align-content: center;
  16.             position: relative;
  17.         }
  18.         #container canvas {
  19.             margin-top: -140px
  20.         }
  21.         .welcome-statement {
  22.             position: absolute;
  23.             top: 100px;
  24.             font-size: 40px;
  25.             color: #999;
  26.         }
  27.         .join-us {
  28.             position: absolute;
  29.             bottom: 200px;
  30.             z-index: 100;
  31.             display: inline-flex;
  32.             padding: 0 20px 3px;
  33.             line-height: 40px;
  34.             background: linear-gradient(to bottom, rgb(87, 196, 245), rgb(26, 147, 206));
  35.             color: rgb(254, 252, 255);
  36.             cursor: pointer;
  37.             border-radius: 4px;
  38.             font-weight: bold;
  39.             box-shadow: inset 0px -3px 0 rgb(19, 98, 139);
  40.         }
  41.         .join-us:active {
  42.             opacity: .7;
  43.             box-shadow: inset 0px 0px 0 transparent;
  44.         }
  45.         .bgm-controller {
  46.             position: absolute;
  47.             right: 20px;
  48.             top: 20px;
  49.             width: 40px;
  50.         }
  51.         .bgm-controller:active {
  52.             opacity: .7;
  53.         }
  54.     </style>
  55. </head>
  56. <body>
  57.     <div id="container">
  58.         <!-- <audio class="bgm" muted="true">
  59.             <source src="./assets/dance/swing-dance.mp3" />
  60.         </audio> -->
  61.         <video class="bgm" muted style="display: none;">
  62.             <source src="./assets/dance/kemusan.mp4" />
  63.         </video>
  64.         <div class="welcome-statement">感谢各位给 H5.Examples 点⭐⭐~</div>
  65.         <a class="join-us" href="https://github.com/linyisonger/H5.Examples">
  66.             加入我们
  67.         </a>
  68.         <img class="bgm-controller" src="./assets/dance/bgm-c.png">
  69.     </div>
  70.     <script type="module">
  71.         /**
  72.          * 加载图
  73.          * @param {string} src
  74.          * @returns {Promise<HTMLImageElement>}
  75.          */
  76.         function loadImage(src) {
  77.             return new Promise((resolve) => {
  78.                 let image = new Image()
  79.                 image.src = src;
  80.                 image.onload = (ev) => {
  81.                     resolve(image)
  82.                 }
  83.             })
  84.         }
  85.         /**
  86.          * 加载音乐
  87.          * @param {string} src
  88.          * @returns {Promise<HTMLImageElement>}
  89.          */
  90.         function loadAudio(src) {
  91.             return new Promise((resolve) => {
  92.                 let audio = new Audio(src)
  93.                 audio.addEventListener("loadeddata", resolve)
  94.             })
  95.         }
  96.         /**
  97.          * 获取star的用户 默认30一页
  98.          * @author          linyisonger
  99.          * @date          2025-02-18
  100.          */
  101.         async function getStargazers(page = 1) {
  102.             const result = await fetch(`https://api.github.com/repos/linyisonger/H5.Examples/stargazers?page=${page}`)
  103.             return await result.json()
  104.         }
  105.         /**
  106.          * 获取所有star的用户
  107.          * @author          linyisonger
  108.          * @date          2025-02-18
  109.          */
  110.         async function getAllStargazers(page = 1, users = []) {
  111.             let stargazers = await getStargazers(page)
  112.             users = users.concat(stargazers)
  113.             if (stargazers.length < 30) return users
  114.             return await getAllStargazers(page + 1, users)
  115.         }
  116.         // getAllStargazers().then((res) => {
  117.         //     console.log("获取star的用户", res);
  118.         // })
  119.         let dancers = [
  120.             {
  121.                 "login": "AnChangSu",
  122.                 "id": 5037050,
  123.                 "node_id": "MDQ6VXNlcjUwMzcwNTA=",
  124.                 "avatar_url": "https://avatars.githubusercontent.com/u/5037050?v=4",
  125.                 "gravatar_id": "",
  126.                 "url": "https://api.github.com/users/AnChangSu",
  127.                 "html_url": "https://github.com/AnChangSu",
  128.                 "followers_url": "https://api.github.com/users/AnChangSu/followers",
  129.                 "following_url": "https://api.github.com/users/AnChangSu/following{/other_user}",
  130.                 "gists_url": "https://api.github.com/users/AnChangSu/gists{/gist_id}",
  131.                 "starred_url": "https://api.github.com/users/AnChangSu/starred{/owner}{/repo}",
  132.                 "subscriptions_url": "https://api.github.com/users/AnChangSu/subscriptions",
  133.                 "organizations_url": "https://api.github.com/users/AnChangSu/orgs",
  134.                 "repos_url": "https://api.github.com/users/AnChangSu/repos",
  135.                 "events_url": "https://api.github.com/users/AnChangSu/events{/privacy}",
  136.                 "received_events_url": "https://api.github.com/users/AnChangSu/received_events",
  137.                 "type": "User",
  138.                 "user_view_type": "public",
  139.                 "site_admin": false
  140.             },
  141.             {
  142.                 "login": "HGinGitHub",
  143.                 "id": 103415496,
  144.                 "node_id": "U_kgDOBin-yA",
  145.                 "avatar_url": "https://avatars.githubusercontent.com/u/103415496?v=4",
  146.                 "gravatar_id": "",
  147.                 "url": "https://api.github.com/users/HGinGitHub",
  148.                 "html_url": "https://github.com/HGinGitHub",
  149.                 "followers_url": "https://api.github.com/users/HGinGitHub/followers",
  150.                 "following_url": "https://api.github.com/users/HGinGitHub/following{/other_user}",
  151.                 "gists_url": "https://api.github.com/users/HGinGitHub/gists{/gist_id}",
  152.                 "starred_url": "https://api.github.com/users/HGinGitHub/starred{/owner}{/repo}",
  153.                 "subscriptions_url": "https://api.github.com/users/HGinGitHub/subscriptions",
  154.                 "organizations_url": "https://api.github.com/users/HGinGitHub/orgs",
  155.                 "repos_url": "https://api.github.com/users/HGinGitHub/repos",
  156.                 "events_url": "https://api.github.com/users/HGinGitHub/events{/privacy}",
  157.                 "received_events_url": "https://api.github.com/users/HGinGitHub/received_events",
  158.                 "type": "User",
  159.                 "user_view_type": "public",
  160.                 "site_admin": false
  161.             },
  162.             {
  163.                 "login": "harris2012",
  164.                 "id": 12846977,
  165.                 "node_id": "MDQ6VXNlcjEyODQ2OTc3",
  166.                 "avatar_url": "https://avatars.githubusercontent.com/u/12846977?v=4",
  167.                 "gravatar_id": "",
  168.                 "url": "https://api.github.com/users/harris2012",
  169.                 "html_url": "https://github.com/harris2012",
  170.                 "followers_url": "https://api.github.com/users/harris2012/followers",
  171.                 "following_url": "https://api.github.com/users/harris2012/following{/other_user}",
  172.                 "gists_url": "https://api.github.com/users/harris2012/gists{/gist_id}",
  173.                 "starred_url": "https://api.github.com/users/harris2012/starred{/owner}{/repo}",
  174.                 "subscriptions_url": "https://api.github.com/users/harris2012/subscriptions",
  175.                 "organizations_url": "https://api.github.com/users/harris2012/orgs",
  176.                 "repos_url": "https://api.github.com/users/harris2012/repos",
  177.                 "events_url": "https://api.github.com/users/harris2012/events{/privacy}",
  178.                 "received_events_url": "https://api.github.com/users/harris2012/received_events",
  179.                 "type": "User",
  180.                 "user_view_type": "public",
  181.                 "site_admin": false
  182.             },
  183.             {
  184.                 "login": "Lavenir7",
  185.                 "id": 105573717,
  186.                 "node_id": "U_kgDOBkrtVQ",
  187.                 "avatar_url": "https://avatars.githubusercontent.com/u/105573717?v=4",
  188.                 "gravatar_id": "",
  189.                 "url": "https://api.github.com/users/Lavenir7",
  190.                 "html_url": "https://github.com/Lavenir7",
  191.                 "followers_url": "https://api.github.com/users/Lavenir7/followers",
  192.                 "following_url": "https://api.github.com/users/Lavenir7/following{/other_user}",
  193.                 "gists_url": "https://api.github.com/users/Lavenir7/gists{/gist_id}",
  194.                 "starred_url": "https://api.github.com/users/Lavenir7/starred{/owner}{/repo}",
  195.                 "subscriptions_url": "https://api.github.com/users/Lavenir7/subscriptions",
  196.                 "organizations_url": "https://api.github.com/users/Lavenir7/orgs",
  197.                 "repos_url": "https://api.github.com/users/Lavenir7/repos",
  198.                 "events_url": "https://api.github.com/users/Lavenir7/events{/privacy}",
  199.                 "received_events_url": "https://api.github.com/users/Lavenir7/received_events",
  200.                 "type": "User",
  201.                 "user_view_type": "public",
  202.                 "site_admin": false
  203.             },
  204.             {
  205.                 "login": "linyisonger",
  206.                 "id": 34770610,
  207.                 "node_id": "MDQ6VXNlcjM0NzcwNjEw",
  208.                 "avatar_url": "https://avatars.githubusercontent.com/u/34770610?v=4",
  209.                 "gravatar_id": "",
  210.                 "url": "https://api.github.com/users/linyisonger",
  211.                 "html_url": "https://github.com/linyisonger",
  212.                 "followers_url": "https://api.github.com/users/linyisonger/followers",
  213.                 "following_url": "https://api.github.com/users/linyisonger/following{/other_user}",
  214.                 "gists_url": "https://api.github.com/users/linyisonger/gists{/gist_id}",
  215.                 "starred_url": "https://api.github.com/users/linyisonger/starred{/owner}{/repo}",
  216.                 "subscriptions_url": "https://api.github.com/users/linyisonger/subscriptions",
  217.                 "organizations_url": "https://api.github.com/users/linyisonger/orgs",
  218.                 "repos_url": "https://api.github.com/users/linyisonger/repos",
  219.                 "events_url": "https://api.github.com/users/linyisonger/events{/privacy}",
  220.                 "received_events_url": "https://api.github.com/users/linyisonger/received_events",
  221.                 "type": "User",
  222.                 "user_view_type": "public",
  223.                 "site_admin": false
  224.             },
  225.             {
  226.                 "login": "lpleipeng",
  227.                 "id": 39250004,
  228.                 "node_id": "MDQ6VXNlcjM5MjUwMDA0",
  229.                 "avatar_url": "https://avatars.githubusercontent.com/u/39250004?v=4",
  230.                 "gravatar_id": "",
  231.                 "url": "https://api.github.com/users/lpleipeng",
  232.                 "html_url": "https://github.com/lpleipeng",
  233.                 "followers_url": "https://api.github.com/users/lpleipeng/followers",
  234.                 "following_url": "https://api.github.com/users/lpleipeng/following{/other_user}",
  235.                 "gists_url": "https://api.github.com/users/lpleipeng/gists{/gist_id}",
  236.                 "starred_url": "https://api.github.com/users/lpleipeng/starred{/owner}{/repo}",
  237.                 "subscriptions_url": "https://api.github.com/users/lpleipeng/subscriptions",
  238.                 "organizations_url": "https://api.github.com/users/lpleipeng/orgs",
  239.                 "repos_url": "https://api.github.com/users/lpleipeng/repos",
  240.                 "events_url": "https://api.github.com/users/lpleipeng/events{/privacy}",
  241.                 "received_events_url": "https://api.github.com/users/lpleipeng/received_events",
  242.                 "type": "User",
  243.                 "user_view_type": "public",
  244.                 "site_admin": false
  245.             },
  246.             {
  247.                 "login": "xxxggg-ctrl",
  248.                 "id": 63829555,
  249.                 "node_id": "MDQ6VXNlcjYzODI5NTU1",
  250.                 "avatar_url": "https://avatars.githubusercontent.com/u/63829555?v=4",
  251.                 "gravatar_id": "",
  252.                 "url": "https://api.github.com/users/xxxggg-ctrl",
  253.                 "html_url": "https://github.com/xxxggg-ctrl",
  254.                 "followers_url": "https://api.github.com/users/xxxggg-ctrl/followers",
  255.                 "following_url": "https://api.github.com/users/xxxggg-ctrl/following{/other_user}",
  256.                 "gists_url": "https://api.github.com/users/xxxggg-ctrl/gists{/gist_id}",
  257.                 "starred_url": "https://api.github.com/users/xxxggg-ctrl/starred{/owner}{/repo}",
  258.                 "subscriptions_url": "https://api.github.com/users/xxxggg-ctrl/subscriptions",
  259.                 "organizations_url": "https://api.github.com/users/xxxggg-ctrl/orgs",
  260.                 "repos_url": "https://api.github.com/users/xxxggg-ctrl/repos",
  261.                 "events_url": "https://api.github.com/users/xxxggg-ctrl/events{/privacy}",
  262.                 "received_events_url": "https://api.github.com/users/xxxggg-ctrl/received_events",
  263.                 "type": "User",
  264.                 "user_view_type": "public",
  265.                 "site_admin": false
  266.             }
  267.         ]
  268.         let bgmControllerDom = document.querySelector('.bgm-controller')
  269.         bgmControllerDom.addEventListener("click", () => {
  270.             const bgm = document.body.querySelector('.bgm')
  271.             bgm.muted = !bgm.muted;
  272.             bgmControllerDom.setAttribute('src', bgm.muted ? './assets/dance/bgm-c.png' : './assets/dance/bgm-o.png')
  273.         })
  274.         // 2D火柴人 贴图
  275.         function fetchLoad(url) {
  276.             return new Promise((resolve) => {
  277.                 fetch(url).then((response) => response.json()).then(resolve)
  278.             })
  279.         }
  280.         async function initGame() {
  281.             dancers = await getAllStargazers()
  282.             let dance = await fetchLoad("./assets/dance/kemusan.json")
  283.             const DROP_FRAME = 5 // 抽帧
  284.             const ZOOM_OUT = .5
  285.             // 检查动作信息
  286.             // for (let i = 0; i < dance.frames.length; i++) {
  287.             //     console.log(i);
  288.             //     const frame = dance.frames[i];
  289.             //     const img = document.createElement('img')
  290.             //     img.src = frame.url;
  291.             //     document.body.appendChild(img)
  292.             // }
  293.             let danceCvsList = []
  294.             for (let i = 0; i < dancers.length; i++) {
  295.                 const dancer = dancers[i];
  296.                 let danceCvs = await createCanvas(dancer)
  297.                 danceCvsList.push({
  298.                     dancer,
  299.                     cvs: danceCvs
  300.                 })
  301.             }
  302.             let i = 0
  303.             async function animationFrame() {
  304.                 if (i % DROP_FRAME == 0) {
  305.                     for (let j = 0; j < danceCvsList.length; j++) {
  306.                         const { cvs, dancer } = danceCvsList[j];
  307.                         await drawFrame(cvs, dance.frames[(i / DROP_FRAME) % dance.frames.length], dancer.avatar_url)
  308.                     }
  309.                 }
  310.                 requestAnimationFrame(animationFrame)
  311.                 i++;
  312.             }
  313.             await animationFrame()
  314.             document.body.querySelector('.bgm').play()
  315.             document.body.querySelector('.bgm').loop = true;
  316.             /**
  317.              * 创建一个用户
  318.              * @author          linyisonger
  319.              * @date          2025-02-23
  320.              */
  321.             function createCanvas(dancer) {
  322.                 let avatarUrl = dancer.avatar_url
  323.                 let cvs = document.createElement("canvas")
  324.                 cvs.setAttribute('width', dance.width)
  325.                 cvs.setAttribute('height', dance.height)
  326.                 cvs.width = dance.width * ZOOM_OUT;
  327.                 cvs.height = dance.height * ZOOM_OUT;
  328.                 document.body.querySelector("#container").appendChild(cvs)
  329.                 return cvs
  330.             }
  331.             /**
  332.              * 渲染一帧
  333.              * @author          linyisonger
  334.              * @date          2025-02-23
  335.              */
  336.             async function drawFrame(cvs, frame, avatar) {
  337.                 /** @type {CanvasRenderingContext2D } */
  338.                 let ctx = cvs.getContext('2d')
  339.                 let roleImg = await loadImage(frame.url)
  340.                 let avatarImg = await loadImage(avatar)
  341.                 ctx.clearRect(0, 0, cvs.width, cvs.height)
  342.                 ctx.drawImage(roleImg, 0, 0, cvs.width, cvs.height)
  343.                 let avatarWidth = 40 * ZOOM_OUT
  344.                 ctx.drawImage(avatarImg, (frame.avatar.x * ZOOM_OUT - avatarWidth / 2), (frame.avatar.y * ZOOM_OUT - avatarWidth / 2), avatarWidth, avatarWidth)
  345.             }
  346.         }
  347.         initGame()
  348.     </script>
  349. </body>
  350. </html>
复制代码
源码仓库

更新的话文章大概不肯定会更新,仓库会大概更新,有问题可以提issue~
   https://github.com/linyisonger/H5.Examples

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

吴旭华

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表