1
0

Compare commits

...

6 Commits

Author SHA1 Message Date
5234d77434 fix 2025-03-18 06:33:11 +09:00
e79dc42a23 fix 2024-10-05 14:17:04 +09:00
72456a96f3 fix 2024-10-04 13:37:23 +09:00
7bc0ba9e93 fix 2024-10-02 14:58:36 +09:00
70366cc5ae add holo 2024-10-02 13:25:44 +09:00
b4c67796b7 fix 2024-10-02 13:18:13 +09:00
28 changed files with 556 additions and 127 deletions

1
.gitignore vendored
View File

@@ -7,3 +7,4 @@
**_site** **_site**
**.jekyll** **.jekyll**
galaxy-react/public/models galaxy-react/public/models
holo/public/models

View File

@@ -5,6 +5,7 @@
"dependencies": { "dependencies": {
"@pixiv/three-vrm": "^3.0.0", "@pixiv/three-vrm": "^3.0.0",
"@pixiv/three-vrm-animation": "^3.0.0", "@pixiv/three-vrm-animation": "^3.0.0",
"@pixiv/three-vrm-materials-mtoon": "^3.0.0",
"@pixiv/three-vrm-springbone": "^3.0.0", "@pixiv/three-vrm-springbone": "^3.0.0",
"@react-three/drei": "^9.109.2", "@react-three/drei": "^9.109.2",
"@react-three/fiber": "^8.16.8", "@react-three/fiber": "^8.16.8",

View File

@@ -12,7 +12,13 @@ $ npm run start
# npm install three three-stdlib @types/three @react-three/fiber @react-three/fiber @react-three/drei @react-three/postprocessing # npm install three three-stdlib @types/three @react-three/fiber @react-three/fiber @react-three/drei @react-three/postprocessing
``` ```
## tips
- `?ms=0`
- `?g=moon`
## ref ## ref
- galaxy.glb : https://sketchfab.com/3d-models/need-some-space-d6521362b37b48e3a82bce4911409303 - galaxy.glb : https://sketchfab.com/3d-models/need-some-space-d6521362b37b48e3a82bce4911409303
- tsx : https://gist.github.com/artokun/fb7f0c68a01ba5d9813abb3ccce254c4 - tsx : https://gist.github.com/artokun/fb7f0c68a01ba5d9813abb3ccce254c4

View File

@@ -3,7 +3,7 @@ html {
} }
body { body {
background-color: #343434; background-color: #fff;
background-color: #000; background-color: #000;
margin: 0; margin: 0;
height: 100%; height: 100%;

View File

@@ -1,5 +1,5 @@
import * as THREE from 'three' import * as THREE from 'three'
import { Points, useGLTF, OrbitControls, ScrollControls, Scroll } from '@react-three/drei' import { Points, useGLTF, OrbitControls } from '@react-three/drei'
import { GLTF } from 'three-stdlib' import { GLTF } from 'three-stdlib'
import { useFrame, Canvas, useThree } from '@react-three/fiber' import { useFrame, Canvas, useThree } from '@react-three/fiber'
import { useMemo, useRef, Suspense, useState, useEffect } from 'react' import { useMemo, useRef, Suspense, useState, useEffect } from 'react'
@@ -7,50 +7,17 @@ import { EffectComposer, SelectiveBloom } from '@react-three/postprocessing'
import { useLoader } from '@react-three/fiber' import { useLoader } from '@react-three/fiber'
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader' import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader'
import { CubeTextureLoader } from 'three' import { CubeTextureLoader } from 'three'
import { useLocation } from 'react-router-dom'; import { SphereGeometry } from 'three';
import { VRMLoaderPlugin, VRM, VRMUtils, VRMSpringBoneLoaderPlugin } from "@pixiv/three-vrm";
// vrm
import { VRMLoaderPlugin, VRM, VRMUtils,VRMSpringBoneLoaderPlugin } from "@pixiv/three-vrm";
import { VRMAnimationLoaderPlugin, VRMAnimation, createVRMAnimationClip } from "@pixiv/three-vrm-animation";
import { VRMSpringBoneManager, VRMSpringBoneJoint, VRMSpringBoneJointHelper } from '@pixiv/three-vrm-springbone';
// circle(sphere)
import { SphereGeometry, BoxGeometry } from 'three';
import { AnimationMixer, Clock } from 'three';
interface ModelProps {
url: string
position?: [number, number, number]
scale?: number | [number, number, number]
}
interface ModelPropsVrm { interface ModelPropsVrm {
url: string url: string
position?: [number, number, number] position?: [number, number, number]
scale?: number scale?: number
} }
interface ModelPropsVrmModel {
url: string
vrmaurl: string
}
function QueryParamExample() {
const location = useLocation();
const searchParams = new URLSearchParams(location.search);
const query = searchParams.get('q');
return <div>Search query: {query}</div>;
}
const Model: React.FC<ModelProps> = ({ url, position = [0, 0, 0], scale = 1 }) => {
const gltf = useLoader(GLTFLoader, url)
return (
<group>
<primitive object={gltf.scene} position={position} scale={scale} />
</group>
)
}
const Vrm: React.FC<ModelPropsVrm> = ({ url, position = [0, 0, 0], scale = 1}) => { const Vrm: React.FC<ModelPropsVrm> = ({ url, position = [0, 0, 0], scale = 1}) => {
const gltf = useLoader(GLTFLoader, url, (loader) => { const gltf = useLoader(GLTFLoader, url, (loader) => {
loader.register((parser) => new VRMLoaderPlugin(parser)); loader.register((parser) => new VRMLoaderPlugin(parser));
loader.register((parser) => new VRMSpringBoneLoaderPlugin(parser)); loader.register((parser) => new VRMSpringBoneLoaderPlugin(parser));
@@ -69,10 +36,7 @@ const Vrm: React.FC<ModelPropsVrm> = ({ url, position = [0, 0, 0], scale = 1})
vrmModel.scene.position.set(...position); vrmModel.scene.position.set(...position);
vrmModel.scene.scale.setScalar(scale); vrmModel.scene.scale.setScalar(scale);
} }
// Spring Boneの設定を調整
if (vrmModel.springBoneManager) {
adjustSpringBones(vrmModel.springBoneManager);
}
} }
}, [gltf, position, scale]); }, [gltf, position, scale]);
useFrame((state, delta) => { useFrame((state, delta) => {
@@ -84,82 +48,8 @@ const Vrm: React.FC<ModelPropsVrm> = ({ url, position = [0, 0, 0], scale = 1})
return gltf ? <primitive object={gltf.scene} /> : null; return gltf ? <primitive object={gltf.scene} /> : null;
}; };
function adjustSpringBones(springBoneManager: any) {
console.log('Spring Bone Manager:', springBoneManager);
if (Array.isArray(springBoneManager.springBoneGroupList)) {
springBoneManager.springBoneGroupList.forEach((group: any) => {
if (Array.isArray(group.joints)) {
group.joints.forEach(adjustJoint);
} else if (Array.isArray(group.bones)) {
group.bones.forEach(adjustJoint);
}
});
}
else if (Array.isArray(springBoneManager.springs)) {
springBoneManager.springs.forEach(adjustJoint);
}
}
function adjustJoint(joint: any) {
if (joint) {
joint.stiffnessForce = joint.stiffness = 0.8;
joint.gravityPower = 0.1;
joint.dragForce = 0.5;
joint.hitRadius = 0.02;
// joint.colliders = [...];
console.log('Adjusted joint:', joint);
}
}
export default Vrm; export default Vrm;
const VRMModel: React.FC<ModelPropsVrmModel> = ({ url, vrmaurl }) => {
const gltf = useLoader(GLTFLoader, url, (loader) => {
loader.register((parser) => new VRMLoaderPlugin(parser));
});
const vrma = useLoader(GLTFLoader, vrmaurl, (loader) => {
loader.register((parser) => new VRMAnimationLoaderPlugin(parser));
});
const [vrm, setVrm] = useState<VRM | null>(null);
const [mixer, setMixer] = useState<AnimationMixer | null>(null);
const clock = useRef(new Clock());
useEffect(() => {
if (gltf && vrma) {
const vrmModel = gltf.userData.vrm as VRM;
VRMUtils.removeUnnecessaryJoints(vrmModel.scene);
setVrm(vrmModel);
const vrmAnimation = vrma.userData.vrmAnimation as VRMAnimation;
const mixer = new AnimationMixer(vrmModel.scene);
//const mixer = new THREE.AnimationMixer(vrmModel.scene);
const clip = createVRMAnimationClip(vrmAnimation, vrmModel);
const action = mixer.clipAction(clip);
action.play();
setMixer(mixer);
}
}, [gltf, vrma]);
useFrame((state, delta) => {
if (mixer) {
mixer.update(delta);
}
if (vrm) {
vrm.update(delta);
}
});
return gltf ? <primitive object={gltf.scene} /> : null;
};
export function Skybox() {
const { scene } = useThree()
const loader = new CubeTextureLoader()
const texture = loader.load([
])
scene.background = texture
return null
}
type GLTFResult = GLTF & { type GLTFResult = GLTF & {
nodes: { nodes: {
Object_2: THREE.Mesh Object_2: THREE.Mesh
@@ -251,7 +141,6 @@ export function Galaxy(props: JSX.IntrinsicElements['group']) {
return ( return (
<group {...props} dispose={null} ref={ref}> <group {...props} dispose={null} ref={ref}>
<Vrm url={model_path} position={[2, -4, 8]} scale={10} /> <Vrm url={model_path} position={[2, -4, 8]} scale={10} />
<pointLight <pointLight
position={[0, 0, 0]} position={[0, 0, 0]}
@@ -284,7 +173,28 @@ export function Galaxy(props: JSX.IntrinsicElements['group']) {
export const ThreeFiberGalaxy = () => { export const ThreeFiberGalaxy = () => {
return ( return (
<Canvas camera={{ position: [0, 12, 0], fov: 75 }}> <Canvas camera={{ position: [0, 12, 0], fov: 75 }}
shadows
gl={{
//toneMapping: THREE.ACESFilmicToneMapping,
//toneMapping: THREE.ReinhardToneMapping,
toneMapping: THREE.NeutralToneMapping,
toneMappingExposure: 1.5,
alpha: true,
powerPreference: "high-performance",
antialias: true,
//stencil: false,
//depth: false
}}
>
<ambientLight intensity={1} />
<pointLight position={[10, 10, 10]} />
<directionalLight
color="white"
castShadow
position={[0, 10, 0]}
intensity={1.5}
shadow-mapSize={[1024, 1024]}/>
<OrbitControls <OrbitControls
//minAzimuthAngle={-Math.PI / 4} //minAzimuthAngle={-Math.PI / 4}
//maxAzimuthAngle={Math.PI / 4} //maxAzimuthAngle={Math.PI / 4}
@@ -299,17 +209,10 @@ export const ThreeFiberGalaxy = () => {
<Suspense fallback={null}> <Suspense fallback={null}>
<Galaxy position={[0, 9, 0]} /> <Galaxy position={[0, 9, 0]} />
</Suspense> </Suspense>
<Skybox />
</Canvas> </Canvas>
) )
} }
useGLTF.preload('./models/galaxy.glb') useGLTF.preload('./models/galaxy.glb')
//useGLTF.preload('./models/ai.glb')
//<color attach="background" args={['#fff']} />
//<Model url="./models/ai.vrm" position={[0, 0, 0]} scale={1} />
//<Vrm url="./models/ai.vrm" position={[0, 0, 0]} scale={1} />
//<ScrollControls pages={2}>
//<Scroll>
//</Scroll>
//</ScrollControls>

1
holo/.env Normal file
View File

@@ -0,0 +1 @@
GENERATE_SOURCEMAP=false

24
holo/.gitignore vendored Normal file
View File

@@ -0,0 +1,24 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
# dependencies
/node_modules
/.pnp
.pnp.js
# testing
/coverage
# production
/build
# misc
.DS_Store
.env.local
.env.development.local
.env.test.local
.env.production.local
npm-debug.log*
yarn-debug.log*
yarn-error.log*
*-lock.json

52
holo/package.json Normal file
View File

@@ -0,0 +1,52 @@
{
"name": "holo",
"version": "0.1.0",
"private": true,
"dependencies": {
"@pixiv/three-vrm": "^3.1.1",
"@pixiv/three-vrm-animation": "^3.1.1",
"@react-three/drei": "^9.114.0",
"@react-three/fiber": "^8.17.9",
"@react-three/postprocessing": "^2.16.3",
"@testing-library/jest-dom": "^5.17.0",
"@testing-library/react": "^13.4.0",
"@testing-library/user-event": "^13.5.0",
"@types/jest": "^27.5.2",
"@types/node": "^16.18.112",
"@types/react": "^18.3.10",
"@types/react-dom": "^18.3.0",
"@types/three": "^0.167.2",
"axios": "^1.7.7",
"react": "^18.3.1",
"react-dom": "^18.3.1",
"react-scripts": "5.0.1",
"three": "^0.167.1",
"three-stdlib": "^2.30.5",
"typescript": "^4.9.5",
"web-vitals": "^2.1.4"
},
"scripts": {
"start": "react-scripts start",
"build": "react-scripts build",
"test": "react-scripts test",
"eject": "react-scripts eject"
},
"eslintConfig": {
"extends": [
"react-app",
"react-app/jest"
]
},
"browserslist": {
"production": [
">0.2%",
"not dead",
"not op_mini all"
],
"development": [
"last 1 chrome version",
"last 1 firefox version",
"last 1 safari version"
]
}
}

BIN
holo/public/favicon.ico Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

43
holo/public/index.html Normal file
View File

@@ -0,0 +1,43 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" href=".%PUBLIC_URL%/favicon.ico" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="theme-color" content="#000000" />
<meta
name="description"
content="Web site created using create-react-app"
/>
<link rel="apple-touch-icon" href=".%PUBLIC_URL%/logo192.png" />
<!--
manifest.json provides metadata used when your web app is installed on a
user's mobile device or desktop. See https://developers.google.com/web/fundamentals/web-app-manifest/
-->
<link rel="manifest" href=".%PUBLIC_URL%/manifest.json" />
<!--
Notice the use of %PUBLIC_URL% in the tags above.
It will be replaced with the URL of the `public` folder during the build.
Only files inside the `public` folder can be referenced from the HTML.
Unlike "/favicon.ico" or "favicon.ico", "%PUBLIC_URL%/favicon.ico" will
work correctly both with client-side routing and a non-root public URL.
Learn how to configure a non-root public URL by running `npm run build`.
-->
<title>React App</title>
</head>
<body>
<noscript>You need to enable JavaScript to run this app.</noscript>
<div id="root"></div>
<!--
This HTML file is a template.
If you open it directly in the browser, you will see an empty page.
You can add webfonts, meta tags, or analytics to this file.
The build step will place the bundled scripts into the <body> tag.
To begin the development, run `npm start` or `yarn start`.
To create a production bundle, use `npm run build` or `yarn build`.
-->
</body>
</html>

BIN
holo/public/logo192.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.2 KiB

BIN
holo/public/logo512.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.4 KiB

25
holo/public/manifest.json Normal file
View File

@@ -0,0 +1,25 @@
{
"short_name": "React App",
"name": "Create React App Sample",
"icons": [
{
"src": "favicon.ico",
"sizes": "64x64 32x32 24x24 16x16",
"type": "image/x-icon"
},
{
"src": "logo192.png",
"type": "image/png",
"sizes": "192x192"
},
{
"src": "logo512.png",
"type": "image/png",
"sizes": "512x512"
}
],
"start_url": ".",
"display": "standalone",
"theme_color": "#000000",
"background_color": "#ffffff"
}

3
holo/public/robots.txt Normal file
View File

@@ -0,0 +1,3 @@
# https://www.robotstxt.org/robotstxt.html
User-agent: *
Disallow:

20
holo/readme.md Normal file
View File

@@ -0,0 +1,20 @@
react-three-fiber + three-vrm
```sh
$ npm i
$ npm run start
```
## install
```sh
# npx create-react-app vrm1 --template typescript
# npm install three three-stdlib @types/three @react-three/fiber @react-three/fiber @react-three/drei @react-three/postprocessing
```
## three-vrm
`vrm 1.0`
unity export

39
holo/src/App.css Normal file
View File

@@ -0,0 +1,39 @@
.App {
text-align: center;
}
.App-logo {
height: 40vmin;
pointer-events: none;
}
@media (prefers-reduced-motion: no-preference) {
.App-logo {
animation: App-logo-spin infinite 20s linear;
}
}
.App-header {
background-color: #282c34;
min-height: 100vh;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
font-size: calc(10px + 2vmin);
color: white;
}
.App-link {
color: #61dafb;
}
@keyframes App-logo-spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}

9
holo/src/App.test.tsx Normal file
View File

@@ -0,0 +1,9 @@
import React from 'react';
import { render, screen } from '@testing-library/react';
import App from './App';
test('renders learn react link', () => {
render(<App />);
const linkElement = screen.getByText(/learn react/i);
expect(linkElement).toBeInTheDocument();
});

16
holo/src/App.tsx Normal file
View File

@@ -0,0 +1,16 @@
import React from 'react'
import VRMModelCanvas from './pages/vrm'
import ScreenTimeCanvas from './pages/time'
import ScreenApiCanvas from './pages/api'
const App = () => {
return (
<>
<VRMModelCanvas/>
<ScreenTimeCanvas/>
<ScreenApiCanvas/>
</>
)
}
export default App;

39
holo/src/index.css Normal file
View File

@@ -0,0 +1,39 @@
body {
height: 100%;
margin: 0;
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
monospace;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
code {
font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',
monospace;
}
html {
height: 100%;
}
.time {
transform:scale(-1,1);
position: absolute;
top: 10px;
padding: 10px;
z-index: 100;
color: #e9ff00;
font-size: 30px;
text-align: center;
}
.api {
transform:scale(-1,1);
position: absolute;
top: 50px;
padding: 10px;
z-index: 100;
color: #e9ff00;
font-size: 25px;
text-align: center;
}

19
holo/src/index.tsx Normal file
View File

@@ -0,0 +1,19 @@
import React from 'react';
import ReactDOM from 'react-dom/client';
import './index.css';
import App from './App';
import reportWebVitals from './reportWebVitals';
const root = ReactDOM.createRoot(
document.getElementById('root') as HTMLElement
);
root.render(
<React.StrictMode>
<App />
</React.StrictMode>
);
// If you want to start measuring performance in your app, pass a function
// to log results (for example: reportWebVitals(console.log))
// or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals
reportWebVitals();

1
holo/src/logo.svg Normal file
View File

@@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 841.9 595.3"><g fill="#61DAFB"><path d="M666.3 296.5c0-32.5-40.7-63.3-103.1-82.4 14.4-63.6 8-114.2-20.2-130.4-6.5-3.8-14.1-5.6-22.4-5.6v22.3c4.6 0 8.3.9 11.4 2.6 13.6 7.8 19.5 37.5 14.9 75.7-1.1 9.4-2.9 19.3-5.1 29.4-19.6-4.8-41-8.5-63.5-10.9-13.5-18.5-27.5-35.3-41.6-50 32.6-30.3 63.2-46.9 84-46.9V78c-27.5 0-63.5 19.6-99.9 53.6-36.4-33.8-72.4-53.2-99.9-53.2v22.3c20.7 0 51.4 16.5 84 46.6-14 14.7-28 31.4-41.3 49.9-22.6 2.4-44 6.1-63.6 11-2.3-10-4-19.7-5.2-29-4.7-38.2 1.1-67.9 14.6-75.8 3-1.8 6.9-2.6 11.5-2.6V78.5c-8.4 0-16 1.8-22.6 5.6-28.1 16.2-34.4 66.7-19.9 130.1-62.2 19.2-102.7 49.9-102.7 82.3 0 32.5 40.7 63.3 103.1 82.4-14.4 63.6-8 114.2 20.2 130.4 6.5 3.8 14.1 5.6 22.5 5.6 27.5 0 63.5-19.6 99.9-53.6 36.4 33.8 72.4 53.2 99.9 53.2 8.4 0 16-1.8 22.6-5.6 28.1-16.2 34.4-66.7 19.9-130.1 62-19.1 102.5-49.9 102.5-82.3zm-130.2-66.7c-3.7 12.9-8.3 26.2-13.5 39.5-4.1-8-8.4-16-13.1-24-4.6-8-9.5-15.8-14.4-23.4 14.2 2.1 27.9 4.7 41 7.9zm-45.8 106.5c-7.8 13.5-15.8 26.3-24.1 38.2-14.9 1.3-30 2-45.2 2-15.1 0-30.2-.7-45-1.9-8.3-11.9-16.4-24.6-24.2-38-7.6-13.1-14.5-26.4-20.8-39.8 6.2-13.4 13.2-26.8 20.7-39.9 7.8-13.5 15.8-26.3 24.1-38.2 14.9-1.3 30-2 45.2-2 15.1 0 30.2.7 45 1.9 8.3 11.9 16.4 24.6 24.2 38 7.6 13.1 14.5 26.4 20.8 39.8-6.3 13.4-13.2 26.8-20.7 39.9zm32.3-13c5.4 13.4 10 26.8 13.8 39.8-13.1 3.2-26.9 5.9-41.2 8 4.9-7.7 9.8-15.6 14.4-23.7 4.6-8 8.9-16.1 13-24.1zM421.2 430c-9.3-9.6-18.6-20.3-27.8-32 9 .4 18.2.7 27.5.7 9.4 0 18.7-.2 27.8-.7-9 11.7-18.3 22.4-27.5 32zm-74.4-58.9c-14.2-2.1-27.9-4.7-41-7.9 3.7-12.9 8.3-26.2 13.5-39.5 4.1 8 8.4 16 13.1 24 4.7 8 9.5 15.8 14.4 23.4zM420.7 163c9.3 9.6 18.6 20.3 27.8 32-9-.4-18.2-.7-27.5-.7-9.4 0-18.7.2-27.8.7 9-11.7 18.3-22.4 27.5-32zm-74 58.9c-4.9 7.7-9.8 15.6-14.4 23.7-4.6 8-8.9 16-13 24-5.4-13.4-10-26.8-13.8-39.8 13.1-3.1 26.9-5.8 41.2-7.9zm-90.5 125.2c-35.4-15.1-58.3-34.9-58.3-50.6 0-15.7 22.9-35.6 58.3-50.6 8.6-3.7 18-7 27.7-10.1 5.7 19.6 13.2 40 22.5 60.9-9.2 20.8-16.6 41.1-22.2 60.6-9.9-3.1-19.3-6.5-28-10.2zM310 490c-13.6-7.8-19.5-37.5-14.9-75.7 1.1-9.4 2.9-19.3 5.1-29.4 19.6 4.8 41 8.5 63.5 10.9 13.5 18.5 27.5 35.3 41.6 50-32.6 30.3-63.2 46.9-84 46.9-4.5-.1-8.3-1-11.3-2.7zm237.2-76.2c4.7 38.2-1.1 67.9-14.6 75.8-3 1.8-6.9 2.6-11.5 2.6-20.7 0-51.4-16.5-84-46.6 14-14.7 28-31.4 41.3-49.9 22.6-2.4 44-6.1 63.6-11 2.3 10.1 4.1 19.8 5.2 29.1zm38.5-66.7c-8.6 3.7-18 7-27.7 10.1-5.7-19.6-13.2-40-22.5-60.9 9.2-20.8 16.6-41.1 22.2-60.6 9.9 3.1 19.3 6.5 28.1 10.2 35.4 15.1 58.3 34.9 58.3 50.6-.1 15.7-23 35.6-58.4 50.6zM320.8 78.4z"/><circle cx="420.9" cy="296.5" r="45.7"/><path d="M520.5 78.1z"/></g></svg>

After

Width:  |  Height:  |  Size: 2.6 KiB

58
holo/src/pages/api.tsx Normal file
View File

@@ -0,0 +1,58 @@
//import React, { useState, useEffect } from 'react';
//import axios from 'axios';
//interface Post {
// id: number;
// username: string;
// planet: number;
//}
//const ScreenApiCanvas: React.FC = () => {
// const [posts, setPosts] = useState<Post[]>([]);
// const [loading, setLoading] = useState(true);
// const [error, setError] = useState<string | null>(null);
// const searchParams = new URLSearchParams(window.location.search);
// var id = searchParams.get('id') ?? '2';
// var url = 'https://api.syui.ai/users/' + id;
// useEffect(() => {
// const fetchPosts = async () => {
// try {
// const response = await axios.get<Post[]>(url);
// setPosts(response.data);
// setLoading(false);
// } catch (err) {
// setError('error network api');
// setLoading(false);
// }
// };
//
// fetchPosts();
// }, [url]);
//
// if (loading) return <div>読み込み中...</div>;
// if (error) return <div>{error}</div>;
//
// return (
// <div className="api">
// <ul>
// {posts.map(post => (
// <li key={post.id}>{post.username} / M {post.planet}</li>
// ))}
// </ul>
// </div>
// );
//};
import React from 'react';
const ScreenApiCanvas: React.FC = () => {
const searchParams = new URLSearchParams(window.location.search);
var m = searchParams.get('m') ?? '0';
var u = searchParams.get('u') ?? 'ai';
return (
<div className="api">
<p>{u} / M{m}</p>
</div>
);
};
export default ScreenApiCanvas;

37
holo/src/pages/time.tsx Normal file
View File

@@ -0,0 +1,37 @@
import React, { useState, useEffect } from 'react';
//function reverseString(str: string): string {
// return str.split('').reverse().join('');
//}
const ScreenTimeCanvas: React.FC = () => {
const [currentDateTime, setCurrentDateTime] = useState<string>('');
//const [reversedDateTime, setReversedDateTime] = useState<string>('');
useEffect(() => {
const updateDateTime = () => {
const now = new Date();
const formatted = now.toLocaleString("ja-JP", {
year: 'numeric',
month: '2-digit',
day: '2-digit',
hour: '2-digit',
minute: '2-digit',
second: '2-digit'
});
setCurrentDateTime(formatted);
//setReversedDateTime(reverseString(formatted));
};
updateDateTime();
const timer = setInterval(updateDateTime, 1000);
return () => clearInterval(timer);
}, []);
return (
<div className="time">
<p>{currentDateTime}</p>
</div>
);
};
export default ScreenTimeCanvas;

85
holo/src/pages/vrm.tsx Normal file
View File

@@ -0,0 +1,85 @@
import * as THREE from 'three'
import React, { useState, useEffect, useRef } from 'react';
import { OrbitControls } from '@react-three/drei'
import { useFrame, Canvas } from '@react-three/fiber';
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader';
import { VRM, VRMUtils, VRMLoaderPlugin } from '@pixiv/three-vrm';
import { VRMAnimationLoaderPlugin, VRMAnimation, createVRMAnimationClip } from "@pixiv/three-vrm-animation";
import { VRMSpringBoneManager } from '@pixiv/three-vrm-springbone';
interface ModelProps {
url: string
url_anim: string
scale: number
position: [number, number, number]
rotation: [number, number, number]
}
const VRMModel: React.FC<ModelProps> = ({ url, url_anim, position, rotation, scale }) => {
const [vrm, setVrm] = useState<VRM | null>(null);
const mixerRef = useRef<THREE.AnimationMixer | null>(null);
const springBoneManagerRef = useRef<VRMSpringBoneManager | null>(null);
useEffect(() => {
const loader = new GLTFLoader();
loader.register((parser) => new VRMLoaderPlugin(parser));
loader.register((parser) => new VRMAnimationLoaderPlugin(parser));
loader.load(url, (gltf) => {
const vrmModel = gltf.userData.vrm as VRM;
VRMUtils.removeUnnecessaryJoints(vrmModel.scene);
springBoneManagerRef.current = vrmModel.springBoneManager as VRMSpringBoneManager;
springBoneManagerRef.current?.reset();
setVrm(vrmModel);
if (vrmModel) {
vrmModel.scene.rotation.set(...rotation);
vrmModel.scene.position.set(...position);
vrmModel.scene.scale.setScalar(scale);
}
const mixer = new THREE.AnimationMixer(vrmModel.scene);
mixerRef.current = mixer;
loader.load(url_anim, (animGltf) => {
const vrmAnimations = animGltf.userData.vrmAnimations as VRMAnimation[];
if (vrmAnimations && vrmAnimations.length > 0) {
const clip = createVRMAnimationClip(vrmAnimations[0], vrmModel);
mixer.clipAction(clip).play();
}
});
});
}, [url, url_anim, position, scale, rotation]);
useFrame((state, delta) => {
if (mixerRef.current) mixerRef.current.update(delta);
if (springBoneManagerRef.current) springBoneManagerRef.current.update(delta);
if (vrm) vrm.update(delta);
});
return vrm ? <primitive object={vrm.scene} /> : null;
};
export const VRMModelCanvas = () => {
return (
<div style={{ height: '100vh', width: '100vw' }}>
<Canvas
shadows
gl={{
//toneMapping: THREE.ACESFilmicToneMapping,
//toneMapping: THREE.ReinhardToneMapping,
toneMapping: THREE.NeutralToneMapping,
toneMappingExposure: 1,
alpha: true,
powerPreference: "high-performance",
antialias: true,
//stencil: false,
//depth: false
}}
camera={{ position: [1.2, 0, 0] }}>
<color attach="background" args={["#000"]} /> {/* Light gray background */}
<OrbitControls dampingFactor={0.05} rotateSpeed={0.1} zoomSpeed={0.5}/>
<ambientLight intensity={10} />
<pointLight position={[10, 10, 10]} />
<VRMModel url="./models/t.vrm" url_anim="./models/default.vrma" position={[0, -0.6, 0]} rotation={[0, 89.5, 0]} scale={1} />
</Canvas>
</div>
)
}
export default VRMModelCanvas;

1
holo/src/react-app-env.d.ts vendored Normal file
View File

@@ -0,0 +1 @@
/// <reference types="react-scripts" />

View File

@@ -0,0 +1,15 @@
import { ReportHandler } from 'web-vitals';
const reportWebVitals = (onPerfEntry?: ReportHandler) => {
if (onPerfEntry && onPerfEntry instanceof Function) {
import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => {
getCLS(onPerfEntry);
getFID(onPerfEntry);
getFCP(onPerfEntry);
getLCP(onPerfEntry);
getTTFB(onPerfEntry);
});
}
};
export default reportWebVitals;

5
holo/src/setupTests.ts Normal file
View File

@@ -0,0 +1,5 @@
// jest-dom adds custom jest matchers for asserting on DOM nodes.
// allows you to do things like:
// expect(element).toHaveTextContent(/react/i)
// learn more: https://github.com/testing-library/jest-dom
import '@testing-library/jest-dom';

26
holo/tsconfig.json Normal file
View File

@@ -0,0 +1,26 @@
{
"compilerOptions": {
"target": "es5",
"lib": [
"dom",
"dom.iterable",
"esnext"
],
"allowJs": true,
"skipLibCheck": true,
"esModuleInterop": true,
"allowSyntheticDefaultImports": true,
"strict": true,
"forceConsistentCasingInFileNames": true,
"noFallthroughCasesInSwitch": true,
"module": "esnext",
"moduleResolution": "node",
"resolveJsonModule": true,
"isolatedModules": true,
"noEmit": true,
"jsx": "react-jsx"
},
"include": [
"src"
]
}