@promptbook/cli 0.112.0-103 → 0.112.0-104
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/apps/agents-server/src/app/AddAgentButton.tsx +0 -5
- package/apps/agents-server/src/app/actions.ts +50 -0
- package/apps/agents-server/src/app/agents/[agentName]/AgentProfileChat.tsx +3 -4
- package/apps/agents-server/src/app/api/health/route.ts +18 -0
- package/apps/agents-server/src/components/AgentProfile/AgentProfile.tsx +1 -4
- package/apps/agents-server/src/components/Header/Header.tsx +0 -11
- package/apps/agents-server/src/components/Header/useHeaderAgentMenus.tsx +0 -5
- package/apps/agents-server/src/components/NewAgentDialog/useNewAgentDialog.tsx +39 -16
- package/apps/agents-server/src/constants/defaultAgentAvatarVisual.ts +1 -1
- package/apps/agents-server/src/database/migrations/2026-06-0200-default-agent-avatar-visual-octopus3d3.sql +16 -0
- package/apps/agents-server/src/middleware.ts +2 -1
- package/apps/agents-server/src/tools/$provideCdnForServer.ts +43 -2
- package/apps/agents-server/src/utils/agentRouting/resolveAgentRouteTarget.ts +27 -4
- package/apps/agents-server/src/utils/defaultAgents/defaultAgents.ts +168 -0
- package/apps/agents-server/src/utils/defaultAgents/installDefaultAgents.ts +139 -0
- package/esm/index.es.js +518 -7
- package/esm/index.es.js.map +1 -1
- package/esm/src/avatars/types/AvatarVisualDefinition.d.ts +1 -1
- package/esm/src/avatars/visuals/octopus3d3AvatarVisual.d.ts +7 -0
- package/package.json +1 -1
- package/src/avatars/types/AvatarVisualDefinition.ts +1 -0
- package/src/avatars/visuals/avatarVisualRegistry.ts +2 -0
- package/src/avatars/visuals/octopus3d3AvatarVisual.ts +903 -0
- package/src/other/templates/getTemplatesPipelineCollection.ts +784 -716
- package/src/utils/agents/resolveAgentAvatarImageUrl.ts +1 -1
- package/src/version.ts +1 -1
- package/src/versions.txt +1 -1
- package/umd/index.umd.js +518 -7
- package/umd/index.umd.js.map +1 -1
- package/umd/src/avatars/types/AvatarVisualDefinition.d.ts +1 -1
- package/umd/src/avatars/visuals/octopus3d3AvatarVisual.d.ts +7 -0
|
@@ -51,7 +51,7 @@ export type ResolveAgentAvatarImageUrlOptions = ResolveAgentAvatarOptions;
|
|
|
51
51
|
*
|
|
52
52
|
* @private shared avatar contract
|
|
53
53
|
*/
|
|
54
|
-
export const DEFAULT_AGENT_AVATAR_VISUAL_ID: AvatarVisualId = '
|
|
54
|
+
export const DEFAULT_AGENT_AVATAR_VISUAL_ID: AvatarVisualId = 'octopus3d3';
|
|
55
55
|
|
|
56
56
|
/**
|
|
57
57
|
* Resolved avatar descriptor used by interactive UIs to pick either an image or a live canvas visual.
|
package/src/version.ts
CHANGED
|
@@ -16,7 +16,7 @@ export const BOOK_LANGUAGE_VERSION: string_semantic_version = '2.0.0';
|
|
|
16
16
|
* @generated
|
|
17
17
|
* @see https://github.com/webgptorg/promptbook
|
|
18
18
|
*/
|
|
19
|
-
export const PROMPTBOOK_ENGINE_VERSION: string_promptbook_version = '0.112.0-
|
|
19
|
+
export const PROMPTBOOK_ENGINE_VERSION: string_promptbook_version = '0.112.0-104';
|
|
20
20
|
|
|
21
21
|
/**
|
|
22
22
|
* Represents the version string of the Promptbook engine.
|
package/src/versions.txt
CHANGED
package/umd/index.umd.js
CHANGED
|
@@ -60,7 +60,7 @@
|
|
|
60
60
|
* @generated
|
|
61
61
|
* @see https://github.com/webgptorg/promptbook
|
|
62
62
|
*/
|
|
63
|
-
const PROMPTBOOK_ENGINE_VERSION = '0.112.0-
|
|
63
|
+
const PROMPTBOOK_ENGINE_VERSION = '0.112.0-104';
|
|
64
64
|
/**
|
|
65
65
|
* TODO: string_promptbook_version should be constrained to the all versions of Promptbook engine
|
|
66
66
|
* Note: [💞] Ignore a discrepancy between file name and entity name
|
|
@@ -11043,7 +11043,7 @@
|
|
|
11043
11043
|
*
|
|
11044
11044
|
* @private helper of `minecraft2AvatarVisual`
|
|
11045
11045
|
*/
|
|
11046
|
-
const LIGHT_DIRECTION$
|
|
11046
|
+
const LIGHT_DIRECTION$3 = normalizeVector3({
|
|
11047
11047
|
x: 0.4,
|
|
11048
11048
|
y: -0.65,
|
|
11049
11049
|
z: 0.92,
|
|
@@ -11255,7 +11255,7 @@
|
|
|
11255
11255
|
corners: projectedCorners,
|
|
11256
11256
|
texture: faceDefinition.texture,
|
|
11257
11257
|
averageDepth: transformedCorners.reduce((depthSum, corner) => depthSum + corner.z, 0) / transformedCorners.length,
|
|
11258
|
-
lightIntensity: clampNumber$1(dotProduct3D(faceNormal, LIGHT_DIRECTION$
|
|
11258
|
+
lightIntensity: clampNumber$1(dotProduct3D(faceNormal, LIGHT_DIRECTION$3), -1, 1),
|
|
11259
11259
|
outlineColor: cuboid.outlineColor,
|
|
11260
11260
|
};
|
|
11261
11261
|
});
|
|
@@ -12446,7 +12446,7 @@
|
|
|
12446
12446
|
*
|
|
12447
12447
|
* @private helper of `octopus3dAvatarVisual`
|
|
12448
12448
|
*/
|
|
12449
|
-
const LIGHT_DIRECTION$
|
|
12449
|
+
const LIGHT_DIRECTION$2 = normalizeVector3({
|
|
12450
12450
|
x: 0.48,
|
|
12451
12451
|
y: -0.62,
|
|
12452
12452
|
z: 0.94,
|
|
@@ -12656,7 +12656,7 @@
|
|
|
12656
12656
|
corners: projectedCorners,
|
|
12657
12657
|
averageDepth: transformedCorners.reduce((depthSum, transformedCorner) => depthSum + transformedCorner.z, 0) /
|
|
12658
12658
|
transformedCorners.length,
|
|
12659
|
-
lightIntensity: clampNumber$1(dotProduct3D(surfaceNormal, LIGHT_DIRECTION$
|
|
12659
|
+
lightIntensity: clampNumber$1(dotProduct3D(surfaceNormal, LIGHT_DIRECTION$2), -1, 1),
|
|
12660
12660
|
fillStyle: resolveSurfacePatchFillStyle(palette, verticalProgress + verticalColorBias),
|
|
12661
12661
|
outlineColor,
|
|
12662
12662
|
});
|
|
@@ -12845,7 +12845,7 @@
|
|
|
12845
12845
|
*
|
|
12846
12846
|
* @private helper of `octopus3d2AvatarVisual`
|
|
12847
12847
|
*/
|
|
12848
|
-
const LIGHT_DIRECTION = normalizeVector3({
|
|
12848
|
+
const LIGHT_DIRECTION$1 = normalizeVector3({
|
|
12849
12849
|
x: 0.38,
|
|
12850
12850
|
y: -0.6,
|
|
12851
12851
|
z: 0.98,
|
|
@@ -12994,7 +12994,7 @@
|
|
|
12994
12994
|
corners: projectedCorners,
|
|
12995
12995
|
averageDepth: transformedCorners.reduce((depthSum, transformedCorner) => depthSum + transformedCorner.z, 0) /
|
|
12996
12996
|
transformedCorners.length,
|
|
12997
|
-
lightIntensity: clampNumber$1(dotProduct3D(surfaceNormal, LIGHT_DIRECTION), -1, 1),
|
|
12997
|
+
lightIntensity: clampNumber$1(dotProduct3D(surfaceNormal, LIGHT_DIRECTION$1), -1, 1),
|
|
12998
12998
|
fillStyle: resolveBlobbySurfacePatchFillStyle(palette, verticalProgress, Math.max(0, Math.cos(centerLongitude)), resolveLowerLobeWave(centerLongitude, morphologyProfile, animationPhase, timeMs)),
|
|
12999
12999
|
outlineColor: verticalProgress < 0.58 ? `${palette.highlight}73` : `${palette.shadow}8a`,
|
|
13000
13000
|
});
|
|
@@ -13094,6 +13094,516 @@
|
|
|
13094
13094
|
context.restore();
|
|
13095
13095
|
}
|
|
13096
13096
|
|
|
13097
|
+
/* eslint-disable no-magic-numbers */
|
|
13098
|
+
/**
|
|
13099
|
+
* Light direction used by the continuous octopus mesh shading.
|
|
13100
|
+
*
|
|
13101
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13102
|
+
*/
|
|
13103
|
+
const LIGHT_DIRECTION = normalizeVector3({
|
|
13104
|
+
x: 0.34,
|
|
13105
|
+
y: -0.62,
|
|
13106
|
+
z: 1,
|
|
13107
|
+
});
|
|
13108
|
+
/**
|
|
13109
|
+
* Real-octopus tentacle count used by the continuous lower mesh.
|
|
13110
|
+
*
|
|
13111
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13112
|
+
*/
|
|
13113
|
+
const OCTOPUS_TENTACLE_COUNT = 8;
|
|
13114
|
+
/**
|
|
13115
|
+
* Octopus 3D 3 avatar visual.
|
|
13116
|
+
*
|
|
13117
|
+
* @private built-in avatar visual
|
|
13118
|
+
*/
|
|
13119
|
+
const octopus3d3AvatarVisual = {
|
|
13120
|
+
id: 'octopus3d3',
|
|
13121
|
+
title: 'Octopus 3D 3',
|
|
13122
|
+
description: 'Cute continuous 3D octopus with a blobby single mesh, waving tentacle lobes, rich shading, and cursor-aware eyes.',
|
|
13123
|
+
isAnimated: true,
|
|
13124
|
+
supportsPointerTracking: true,
|
|
13125
|
+
render({ context, size, palette, createRandom, timeMs, interaction }) {
|
|
13126
|
+
const morphologyProfile = createOctopus3MorphologyProfile(createRandom);
|
|
13127
|
+
const animationRandom = createRandom('octopus3d3-animation-profile');
|
|
13128
|
+
const eyeRandom = createRandom('octopus3d3-eye-profile');
|
|
13129
|
+
const animationPhase = animationRandom() * Math.PI * 2;
|
|
13130
|
+
const tentacleProfiles = createContinuousTentacleProfiles(createRandom, morphologyProfile);
|
|
13131
|
+
const sceneCenterX = size * 0.5;
|
|
13132
|
+
const sceneCenterY = size * 0.535;
|
|
13133
|
+
const bob = Math.sin(timeMs / 960 + animationPhase) * size * 0.012;
|
|
13134
|
+
const meshCenter = {
|
|
13135
|
+
x: interaction.bodyOffsetX * size * 0.048 + size * morphologyProfile.body.centerXJitterRatio * 0.44,
|
|
13136
|
+
y: -size * 0.07 + interaction.bodyOffsetY * size * 0.026 + bob,
|
|
13137
|
+
z: interaction.intensity * size * 0.018,
|
|
13138
|
+
};
|
|
13139
|
+
const rotationY = -0.1 +
|
|
13140
|
+
Math.sin(timeMs / 2700 + animationPhase) * 0.035 +
|
|
13141
|
+
interaction.bodyOffsetX * 0.22 +
|
|
13142
|
+
interaction.gazeX * 0.88;
|
|
13143
|
+
const rotationX = -0.07 +
|
|
13144
|
+
Math.cos(timeMs / 3100 + animationPhase * 0.7) * 0.018 -
|
|
13145
|
+
interaction.bodyOffsetY * 0.08 -
|
|
13146
|
+
interaction.gazeY * 0.38;
|
|
13147
|
+
const surfaceOptions = {
|
|
13148
|
+
radiusX: size * morphologyProfile.body.bodyRadiusRatio * morphologyProfile.body.horizontalStretch * 1.1,
|
|
13149
|
+
radiusY: size * morphologyProfile.body.bodyRadiusRatio * morphologyProfile.body.verticalStretch * 1.08,
|
|
13150
|
+
radiusZ: size *
|
|
13151
|
+
morphologyProfile.body.bodyRadiusRatio *
|
|
13152
|
+
(1.02 + (morphologyProfile.body.horizontalStretch - 1) * 0.18),
|
|
13153
|
+
morphologyProfile,
|
|
13154
|
+
timeMs,
|
|
13155
|
+
animationPhase,
|
|
13156
|
+
tentacleProfiles,
|
|
13157
|
+
};
|
|
13158
|
+
const surfacePatches = resolveVisibleContinuousOctopusPatches({
|
|
13159
|
+
...surfaceOptions,
|
|
13160
|
+
center: meshCenter,
|
|
13161
|
+
rotationX,
|
|
13162
|
+
rotationY,
|
|
13163
|
+
sceneCenterX,
|
|
13164
|
+
sceneCenterY,
|
|
13165
|
+
size,
|
|
13166
|
+
palette,
|
|
13167
|
+
});
|
|
13168
|
+
const eyeLatitude = clampNumber$1(morphologyProfile.face.eyeCenterYOffsetRatio * 4.2 - 0.03, -0.22, 0.08);
|
|
13169
|
+
const eyeLongitude = clampNumber$1(morphologyProfile.face.eyeSpacingRatio * 3.1, 0.18, 0.32);
|
|
13170
|
+
const mouthLatitude = clampNumber$1(eyeLatitude + 0.2 + morphologyProfile.face.mouthYOffsetRatio, 0.08, 0.34);
|
|
13171
|
+
const mouthCenterLongitude = clampNumber$1(morphologyProfile.face.mouthCenterOffsetRatio * 5.6, -0.08, 0.08);
|
|
13172
|
+
const mouthHalfLongitude = clampNumber$1(eyeLongitude * 0.78, 0.15, 0.28);
|
|
13173
|
+
const mouthCurveLatitude = clampNumber$1(mouthLatitude + morphologyProfile.face.mouthCurveDepthRatio * 0.78, mouthLatitude + 0.03, 0.42);
|
|
13174
|
+
const eyeRadiusX = size * morphologyProfile.face.eyeRadiusXRatio * 0.76;
|
|
13175
|
+
const eyeRadiusY = eyeRadiusX * morphologyProfile.face.eyeHeightRatio * 0.9;
|
|
13176
|
+
drawAvatarFrame(context, size, palette);
|
|
13177
|
+
drawContinuousOctopusAtmosphere(context, size, palette, sceneCenterX, sceneCenterY, interaction, timeMs);
|
|
13178
|
+
drawContinuousOctopusShadow(context, size, palette, interaction, timeMs, morphologyProfile);
|
|
13179
|
+
for (const surfacePatch of surfacePatches.sort((firstSurfacePatch, secondSurfacePatch) => firstSurfacePatch.averageDepth - secondSurfacePatch.averageDepth)) {
|
|
13180
|
+
drawContinuousSurfacePatch(context, surfacePatch);
|
|
13181
|
+
}
|
|
13182
|
+
drawProjectedSurfaceCurrents({
|
|
13183
|
+
context,
|
|
13184
|
+
surfaceOptions,
|
|
13185
|
+
center: meshCenter,
|
|
13186
|
+
rotationX,
|
|
13187
|
+
rotationY,
|
|
13188
|
+
sceneCenterX,
|
|
13189
|
+
sceneCenterY,
|
|
13190
|
+
size,
|
|
13191
|
+
palette,
|
|
13192
|
+
morphologyProfile,
|
|
13193
|
+
timeMs,
|
|
13194
|
+
animationPhase,
|
|
13195
|
+
});
|
|
13196
|
+
drawProjectedTentacleSuckers({
|
|
13197
|
+
context,
|
|
13198
|
+
surfaceOptions,
|
|
13199
|
+
center: meshCenter,
|
|
13200
|
+
rotationX,
|
|
13201
|
+
rotationY,
|
|
13202
|
+
sceneCenterX,
|
|
13203
|
+
sceneCenterY,
|
|
13204
|
+
size,
|
|
13205
|
+
palette,
|
|
13206
|
+
});
|
|
13207
|
+
drawProjectedOrganicEye(context, sampleContinuousOctopusSurfacePoint(surfaceOptions, eyeLatitude, -eyeLongitude), eyeRadiusX, eyeRadiusY, meshCenter, rotationX, rotationY, sceneCenterX, sceneCenterY, size, palette, timeMs, animationPhase + eyeRandom() * 0.7, interaction, morphologyProfile.face.eyeStyle);
|
|
13208
|
+
drawProjectedOrganicEye(context, sampleContinuousOctopusSurfacePoint(surfaceOptions, eyeLatitude, eyeLongitude), eyeRadiusX, eyeRadiusY, meshCenter, rotationX, rotationY, sceneCenterX, sceneCenterY, size, palette, timeMs, animationPhase + 0.85 + eyeRandom() * 0.7, interaction, morphologyProfile.face.eyeStyle);
|
|
13209
|
+
drawProjectedOrganicMouth(context, [
|
|
13210
|
+
sampleContinuousOctopusSurfacePoint(surfaceOptions, mouthLatitude, mouthCenterLongitude - mouthHalfLongitude),
|
|
13211
|
+
sampleContinuousOctopusSurfacePoint(surfaceOptions, mouthCurveLatitude, mouthCenterLongitude),
|
|
13212
|
+
sampleContinuousOctopusSurfacePoint(surfaceOptions, mouthLatitude, mouthCenterLongitude + mouthHalfLongitude),
|
|
13213
|
+
], meshCenter, rotationX, rotationY, sceneCenterX, sceneCenterY, palette, size);
|
|
13214
|
+
},
|
|
13215
|
+
};
|
|
13216
|
+
/**
|
|
13217
|
+
* Creates seeded tentacle-lobe profiles around the visible lower octopus body.
|
|
13218
|
+
*
|
|
13219
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13220
|
+
*/
|
|
13221
|
+
function createContinuousTentacleProfiles(createRandom, morphologyProfile) {
|
|
13222
|
+
return Array.from({ length: OCTOPUS_TENTACLE_COUNT }, (_, tentacleIndex) => {
|
|
13223
|
+
const tentacleRandom = createRandom(`octopus3d3-tentacle-${tentacleIndex}`);
|
|
13224
|
+
const progress = tentacleIndex / (OCTOPUS_TENTACLE_COUNT - 1);
|
|
13225
|
+
return {
|
|
13226
|
+
centerLongitude: -Math.PI * 0.86 +
|
|
13227
|
+
progress * Math.PI * 1.72 +
|
|
13228
|
+
(tentacleRandom() - 0.5) * (0.08 + morphologyProfile.tentacles.rootSpreadScale * 0.03),
|
|
13229
|
+
widthScale: 0.86 + tentacleRandom() * 0.34 + (morphologyProfile.tentacles.baseWidthScale - 1) * 0.16,
|
|
13230
|
+
lengthScale: 0.86 + tentacleRandom() * 0.36 + (morphologyProfile.tentacles.flowLengthScale - 1) * 0.22,
|
|
13231
|
+
swayScale: 0.82 + tentacleRandom() * 0.38 + (morphologyProfile.tentacles.swayScale - 1) * 0.2,
|
|
13232
|
+
depthScale: 0.86 + tentacleRandom() * 0.32 + (morphologyProfile.tentacles.tipReachScale - 1) * 0.2,
|
|
13233
|
+
phase: tentacleRandom() * Math.PI * 2,
|
|
13234
|
+
suckerSide: tentacleRandom() > 0.5 ? 1 : -1,
|
|
13235
|
+
};
|
|
13236
|
+
});
|
|
13237
|
+
}
|
|
13238
|
+
/**
|
|
13239
|
+
* Draws the soft underwater atmosphere behind the continuous octopus mesh.
|
|
13240
|
+
*
|
|
13241
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13242
|
+
*/
|
|
13243
|
+
function drawContinuousOctopusAtmosphere(context, size, palette, sceneCenterX, sceneCenterY, interaction, timeMs) {
|
|
13244
|
+
const glowGradient = context.createRadialGradient(sceneCenterX + interaction.gazeX * size * 0.11, sceneCenterY - size * 0.17 + interaction.gazeY * size * 0.05, size * 0.04, sceneCenterX, sceneCenterY, size * (0.66 + interaction.intensity * 0.02));
|
|
13245
|
+
glowGradient.addColorStop(0, `${palette.highlight}66`);
|
|
13246
|
+
glowGradient.addColorStop(0.34, `${palette.accent}2e`);
|
|
13247
|
+
glowGradient.addColorStop(1, `${palette.highlight}00`);
|
|
13248
|
+
context.fillStyle = glowGradient;
|
|
13249
|
+
context.fillRect(0, 0, size, size);
|
|
13250
|
+
const lowerGradient = context.createRadialGradient(sceneCenterX + Math.sin(timeMs / 1550) * size * 0.05, sceneCenterY + size * 0.29, size * 0.06, sceneCenterX, sceneCenterY + size * 0.3, size * 0.54);
|
|
13251
|
+
lowerGradient.addColorStop(0, `${palette.secondary}25`);
|
|
13252
|
+
lowerGradient.addColorStop(1, `${palette.secondary}00`);
|
|
13253
|
+
context.fillStyle = lowerGradient;
|
|
13254
|
+
context.fillRect(0, 0, size, size);
|
|
13255
|
+
}
|
|
13256
|
+
/**
|
|
13257
|
+
* Draws the soft lower shadow that anchors the octopus in the avatar frame.
|
|
13258
|
+
*
|
|
13259
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13260
|
+
*/
|
|
13261
|
+
function drawContinuousOctopusShadow(context, size, palette, interaction, timeMs, morphologyProfile) {
|
|
13262
|
+
context.save();
|
|
13263
|
+
context.fillStyle = `${palette.shadow}66`;
|
|
13264
|
+
context.filter = `blur(${size * 0.025}px)`;
|
|
13265
|
+
context.beginPath();
|
|
13266
|
+
context.ellipse(size * 0.5 + interaction.gazeX * size * 0.045, size * 0.9 + Math.sin(timeMs / 980) * size * 0.007, size * (0.19 + morphologyProfile.tentacles.rootSpreadScale * 0.022 + interaction.intensity * 0.02), size * 0.06, 0, 0, Math.PI * 2);
|
|
13267
|
+
context.fill();
|
|
13268
|
+
context.restore();
|
|
13269
|
+
}
|
|
13270
|
+
/**
|
|
13271
|
+
* Resolves visible projected patches for the continuous octopus mesh.
|
|
13272
|
+
*
|
|
13273
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13274
|
+
*/
|
|
13275
|
+
function resolveVisibleContinuousOctopusPatches(options) {
|
|
13276
|
+
const { center, rotationX, rotationY, sceneCenterX, sceneCenterY, size, palette } = options;
|
|
13277
|
+
const latitudePatchCount = 16;
|
|
13278
|
+
const longitudePatchCount = 40;
|
|
13279
|
+
const surfacePatches = [];
|
|
13280
|
+
for (let latitudeIndex = 0; latitudeIndex < latitudePatchCount; latitudeIndex++) {
|
|
13281
|
+
const startLatitude = -Math.PI / 2 + (latitudeIndex / latitudePatchCount) * Math.PI;
|
|
13282
|
+
const endLatitude = -Math.PI / 2 + ((latitudeIndex + 1) / latitudePatchCount) * Math.PI;
|
|
13283
|
+
const centerLatitude = (startLatitude + endLatitude) / 2;
|
|
13284
|
+
const verticalProgress = (Math.sin(centerLatitude) + 1) / 2;
|
|
13285
|
+
for (let longitudeIndex = 0; longitudeIndex < longitudePatchCount; longitudeIndex++) {
|
|
13286
|
+
const startLongitude = -Math.PI + (longitudeIndex / longitudePatchCount) * Math.PI * 2;
|
|
13287
|
+
const endLongitude = -Math.PI + ((longitudeIndex + 1) / longitudePatchCount) * Math.PI * 2;
|
|
13288
|
+
const centerLongitude = (startLongitude + endLongitude) / 2;
|
|
13289
|
+
const localCorners = [
|
|
13290
|
+
sampleContinuousOctopusSurfacePoint(options, startLatitude, startLongitude),
|
|
13291
|
+
sampleContinuousOctopusSurfacePoint(options, startLatitude, endLongitude),
|
|
13292
|
+
sampleContinuousOctopusSurfacePoint(options, endLatitude, endLongitude),
|
|
13293
|
+
sampleContinuousOctopusSurfacePoint(options, endLatitude, startLongitude),
|
|
13294
|
+
];
|
|
13295
|
+
const transformedCorners = localCorners.map((localCorner) => transformScenePoint(localCorner, center, rotationX, rotationY));
|
|
13296
|
+
const surfaceNormal = normalizeVector3(crossProduct3D(subtractPoint3D(transformedCorners[1], transformedCorners[0]), subtractPoint3D(transformedCorners[2], transformedCorners[0])));
|
|
13297
|
+
if (surfaceNormal.z <= 0.008) {
|
|
13298
|
+
continue;
|
|
13299
|
+
}
|
|
13300
|
+
const projectedCorners = transformedCorners.map((transformedCorner) => projectScenePoint(transformedCorner, size, sceneCenterX, sceneCenterY));
|
|
13301
|
+
const tentacleInfluence = resolveContinuousTentacleInfluence(options, centerLongitude);
|
|
13302
|
+
const lowerLobeWave = resolveContinuousLobeWave(options, centerLongitude);
|
|
13303
|
+
surfacePatches.push({
|
|
13304
|
+
corners: projectedCorners,
|
|
13305
|
+
averageDepth: transformedCorners.reduce((depthSum, transformedCorner) => depthSum + transformedCorner.z, 0) /
|
|
13306
|
+
transformedCorners.length,
|
|
13307
|
+
lightIntensity: clampNumber$1(dotProduct3D(surfaceNormal, LIGHT_DIRECTION), -1, 1),
|
|
13308
|
+
fillStyle: resolveContinuousSurfacePatchFillStyle(palette, verticalProgress, Math.max(0, Math.cos(centerLongitude)), tentacleInfluence.core, lowerLobeWave),
|
|
13309
|
+
outlineColor: verticalProgress < 0.54 ? `${palette.highlight}69` : `${palette.shadow}78`,
|
|
13310
|
+
});
|
|
13311
|
+
}
|
|
13312
|
+
}
|
|
13313
|
+
return surfacePatches;
|
|
13314
|
+
}
|
|
13315
|
+
/**
|
|
13316
|
+
* Samples one point on the continuous Octopus 3D 3 surface.
|
|
13317
|
+
*
|
|
13318
|
+
* The lower hemisphere is pulled into eight seeded waving lobes, so the portrait reads as
|
|
13319
|
+
* tentacled while still being rendered as one connected blobby mesh.
|
|
13320
|
+
*
|
|
13321
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13322
|
+
*/
|
|
13323
|
+
function sampleContinuousOctopusSurfacePoint(options, latitude, longitude) {
|
|
13324
|
+
const { radiusX, radiusY, radiusZ, morphologyProfile, timeMs, animationPhase } = options;
|
|
13325
|
+
const cosineLatitude = Math.max(0, Math.cos(latitude));
|
|
13326
|
+
const verticalProgress = (Math.sin(latitude) + 1) / 2;
|
|
13327
|
+
const upperBlend = Math.pow(1 - verticalProgress, 1.28);
|
|
13328
|
+
const lowerBlend = smoothStep(0.38, 1, verticalProgress);
|
|
13329
|
+
const tipBlend = smoothStep(0.68, 1, verticalProgress);
|
|
13330
|
+
const tentacleInfluence = resolveContinuousTentacleInfluence(options, longitude);
|
|
13331
|
+
const centerPull = resolveSignedAngularDistance(longitude, tentacleInfluence.centerLongitude);
|
|
13332
|
+
const effectiveLongitude = longitude + centerPull * lowerBlend * tentacleInfluence.core * (0.24 + tipBlend * 0.2);
|
|
13333
|
+
const lowerLobeWave = resolveContinuousLobeWave(options, longitude);
|
|
13334
|
+
const mantleRipple = Math.sin(longitude * morphologyProfile.body.lobeCount +
|
|
13335
|
+
animationPhase * 0.6 +
|
|
13336
|
+
timeMs / (1750 + morphologyProfile.body.lobeCount * 30)) *
|
|
13337
|
+
(0.018 + morphologyProfile.body.wobbleAmplitudeRatio * 0.8) *
|
|
13338
|
+
(0.3 + lowerBlend * 0.7);
|
|
13339
|
+
const tentacleWave = Math.sin(timeMs / 760 + tentacleInfluence.phase + verticalProgress * 2.4) *
|
|
13340
|
+
lowerBlend *
|
|
13341
|
+
tentacleInfluence.core *
|
|
13342
|
+
tentacleInfluence.swayScale;
|
|
13343
|
+
const horizontalScale = 1.04 +
|
|
13344
|
+
mantleRipple +
|
|
13345
|
+
lowerBlend * (0.16 + (morphologyProfile.tentacles.rootSpreadScale - 1) * 0.1) +
|
|
13346
|
+
lowerBlend * tentacleInfluence.core * (0.2 + lowerLobeWave * 0.12) -
|
|
13347
|
+
upperBlend * 0.08;
|
|
13348
|
+
const depthScale = 1.06 +
|
|
13349
|
+
upperBlend * 0.16 +
|
|
13350
|
+
Math.max(0, Math.cos(effectiveLongitude)) * 0.1 +
|
|
13351
|
+
lowerBlend * tentacleInfluence.core * (0.1 + tentacleInfluence.depthScale * 0.06) -
|
|
13352
|
+
Math.max(0, -Math.cos(effectiveLongitude)) * 0.05;
|
|
13353
|
+
const tentacleTubeRadius = lowerBlend *
|
|
13354
|
+
tentacleInfluence.core *
|
|
13355
|
+
(0.11 + tipBlend * 0.06 + tentacleInfluence.widthScale * 0.025) *
|
|
13356
|
+
radiusX;
|
|
13357
|
+
const planarRadiusX = cosineLatitude * radiusX * horizontalScale + tentacleTubeRadius;
|
|
13358
|
+
const planarRadiusZ = cosineLatitude * radiusZ * depthScale + tentacleTubeRadius * 0.72;
|
|
13359
|
+
const lowerDrop = lowerBlend *
|
|
13360
|
+
radiusY *
|
|
13361
|
+
(0.18 +
|
|
13362
|
+
tentacleInfluence.core *
|
|
13363
|
+
(0.38 +
|
|
13364
|
+
tentacleInfluence.lengthScale * 0.22 +
|
|
13365
|
+
(morphologyProfile.tentacles.flowLengthScale - 1) * 0.08));
|
|
13366
|
+
return {
|
|
13367
|
+
x: Math.sin(effectiveLongitude) * planarRadiusX +
|
|
13368
|
+
tentacleWave * radiusX * (0.052 + tipBlend * 0.05),
|
|
13369
|
+
y: Math.sin(latitude) * radiusY * (1 + upperBlend * 0.12) -
|
|
13370
|
+
upperBlend * radiusY * 0.1 +
|
|
13371
|
+
lowerDrop +
|
|
13372
|
+
Math.sin(timeMs / 1420 + animationPhase + latitude * 1.6) * lowerBlend * radiusY * 0.018 +
|
|
13373
|
+
Math.cos(timeMs / 880 + tentacleInfluence.phase) *
|
|
13374
|
+
lowerBlend *
|
|
13375
|
+
tipBlend *
|
|
13376
|
+
tentacleInfluence.core *
|
|
13377
|
+
radiusY *
|
|
13378
|
+
0.034,
|
|
13379
|
+
z: Math.cos(effectiveLongitude) * planarRadiusZ +
|
|
13380
|
+
Math.cos(timeMs / 980 + tentacleInfluence.phase + verticalProgress) *
|
|
13381
|
+
lowerBlend *
|
|
13382
|
+
tentacleInfluence.core *
|
|
13383
|
+
radiusZ *
|
|
13384
|
+
0.04,
|
|
13385
|
+
};
|
|
13386
|
+
}
|
|
13387
|
+
/**
|
|
13388
|
+
* Blends nearby seeded tentacle profiles at one mesh longitude.
|
|
13389
|
+
*
|
|
13390
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13391
|
+
*/
|
|
13392
|
+
function resolveContinuousTentacleInfluence(options, longitude) {
|
|
13393
|
+
let totalWeight = 0;
|
|
13394
|
+
let weightedSin = 0;
|
|
13395
|
+
let weightedCos = 0;
|
|
13396
|
+
let weightedWidthScale = 0;
|
|
13397
|
+
let weightedLengthScale = 0;
|
|
13398
|
+
let weightedSwayScale = 0;
|
|
13399
|
+
let weightedDepthScale = 0;
|
|
13400
|
+
let weightedPhase = 0;
|
|
13401
|
+
for (const tentacleProfile of options.tentacleProfiles) {
|
|
13402
|
+
const distance = Math.abs(resolveSignedAngularDistance(longitude, tentacleProfile.centerLongitude));
|
|
13403
|
+
const width = 0.2 * tentacleProfile.widthScale;
|
|
13404
|
+
const weight = Math.exp(-(distance * distance) / (width * width));
|
|
13405
|
+
totalWeight += weight;
|
|
13406
|
+
weightedSin += Math.sin(tentacleProfile.centerLongitude) * weight;
|
|
13407
|
+
weightedCos += Math.cos(tentacleProfile.centerLongitude) * weight;
|
|
13408
|
+
weightedWidthScale += tentacleProfile.widthScale * weight;
|
|
13409
|
+
weightedLengthScale += tentacleProfile.lengthScale * weight;
|
|
13410
|
+
weightedSwayScale += tentacleProfile.swayScale * weight;
|
|
13411
|
+
weightedDepthScale += tentacleProfile.depthScale * weight;
|
|
13412
|
+
weightedPhase += tentacleProfile.phase * weight;
|
|
13413
|
+
}
|
|
13414
|
+
if (totalWeight < 0.0001) {
|
|
13415
|
+
return {
|
|
13416
|
+
core: 0,
|
|
13417
|
+
centerLongitude: longitude,
|
|
13418
|
+
widthScale: 1,
|
|
13419
|
+
lengthScale: 1,
|
|
13420
|
+
swayScale: 1,
|
|
13421
|
+
depthScale: 1,
|
|
13422
|
+
phase: 0,
|
|
13423
|
+
};
|
|
13424
|
+
}
|
|
13425
|
+
return {
|
|
13426
|
+
core: clampNumber$1(totalWeight, 0, 1),
|
|
13427
|
+
centerLongitude: Math.atan2(weightedSin / totalWeight, weightedCos / totalWeight),
|
|
13428
|
+
widthScale: weightedWidthScale / totalWeight,
|
|
13429
|
+
lengthScale: weightedLengthScale / totalWeight,
|
|
13430
|
+
swayScale: weightedSwayScale / totalWeight,
|
|
13431
|
+
depthScale: weightedDepthScale / totalWeight,
|
|
13432
|
+
phase: weightedPhase / totalWeight,
|
|
13433
|
+
};
|
|
13434
|
+
}
|
|
13435
|
+
/**
|
|
13436
|
+
* Resolves the soft lower wave that makes the continuous mesh read as a set of tentacles.
|
|
13437
|
+
*
|
|
13438
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13439
|
+
*/
|
|
13440
|
+
function resolveContinuousLobeWave(options, longitude) {
|
|
13441
|
+
const { morphologyProfile, animationPhase, timeMs } = options;
|
|
13442
|
+
return (Math.cos(longitude * OCTOPUS_TENTACLE_COUNT + animationPhase + timeMs / (980 + morphologyProfile.body.lobeCount * 18)) +
|
|
13443
|
+
1) / 2;
|
|
13444
|
+
}
|
|
13445
|
+
/**
|
|
13446
|
+
* Resolves one base fill tone for a patch on the continuous octopus mesh.
|
|
13447
|
+
*
|
|
13448
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13449
|
+
*/
|
|
13450
|
+
function resolveContinuousSurfacePatchFillStyle(palette, verticalProgress, forwardness, tentacleCore, lowerLobeWave) {
|
|
13451
|
+
const tonalProgress = clampNumber$1(verticalProgress + lowerLobeWave * 0.1 + tentacleCore * 0.08 - forwardness * 0.08, 0, 1);
|
|
13452
|
+
if (tonalProgress < 0.14) {
|
|
13453
|
+
return palette.highlight;
|
|
13454
|
+
}
|
|
13455
|
+
if (tonalProgress < 0.32) {
|
|
13456
|
+
return palette.secondary;
|
|
13457
|
+
}
|
|
13458
|
+
if (tonalProgress < 0.72) {
|
|
13459
|
+
return forwardness > 0.55 ? palette.secondary : palette.primary;
|
|
13460
|
+
}
|
|
13461
|
+
return tentacleCore > 0.44 ? `${palette.primary}f4` : `${palette.shadow}ee`;
|
|
13462
|
+
}
|
|
13463
|
+
/**
|
|
13464
|
+
* Draws one projected mesh patch with soft shading and a subtle edge.
|
|
13465
|
+
*
|
|
13466
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13467
|
+
*/
|
|
13468
|
+
function drawContinuousSurfacePatch(context, surfacePatch) {
|
|
13469
|
+
drawProjectedQuad(context, surfacePatch.corners, surfacePatch.fillStyle);
|
|
13470
|
+
if (surfacePatch.lightIntensity > 0) {
|
|
13471
|
+
drawProjectedQuad(context, surfacePatch.corners, `rgba(255, 255, 255, ${0.18 * surfacePatch.lightIntensity})`);
|
|
13472
|
+
}
|
|
13473
|
+
else if (surfacePatch.lightIntensity < 0) {
|
|
13474
|
+
drawProjectedQuad(context, surfacePatch.corners, `rgba(0, 0, 0, ${0.25 * Math.abs(surfacePatch.lightIntensity)})`);
|
|
13475
|
+
}
|
|
13476
|
+
context.save();
|
|
13477
|
+
context.beginPath();
|
|
13478
|
+
context.moveTo(surfacePatch.corners[0].x, surfacePatch.corners[0].y);
|
|
13479
|
+
for (let cornerIndex = 1; cornerIndex < surfacePatch.corners.length; cornerIndex++) {
|
|
13480
|
+
context.lineTo(surfacePatch.corners[cornerIndex].x, surfacePatch.corners[cornerIndex].y);
|
|
13481
|
+
}
|
|
13482
|
+
context.closePath();
|
|
13483
|
+
context.strokeStyle = surfacePatch.outlineColor;
|
|
13484
|
+
context.lineWidth = Math.max(0.7, getProjectedQuadPerimeter(surfacePatch.corners) * 0.0032);
|
|
13485
|
+
context.lineJoin = 'round';
|
|
13486
|
+
context.stroke();
|
|
13487
|
+
context.restore();
|
|
13488
|
+
}
|
|
13489
|
+
/**
|
|
13490
|
+
* Draws projected mantle-current lines on the front of the mesh.
|
|
13491
|
+
*
|
|
13492
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13493
|
+
*/
|
|
13494
|
+
function drawProjectedSurfaceCurrents(options) {
|
|
13495
|
+
const { context, surfaceOptions, center, rotationX, rotationY, sceneCenterX, sceneCenterY, size, palette, morphologyProfile, timeMs, animationPhase, } = options;
|
|
13496
|
+
const currentCount = Math.min(6, morphologyProfile.details.mantleCurrentCount);
|
|
13497
|
+
const centerIndex = (currentCount - 1) / 2;
|
|
13498
|
+
context.save();
|
|
13499
|
+
context.lineCap = 'round';
|
|
13500
|
+
context.lineJoin = 'round';
|
|
13501
|
+
for (let currentIndex = 0; currentIndex < currentCount; currentIndex++) {
|
|
13502
|
+
const baseLongitude = (currentIndex - centerIndex) * 0.15;
|
|
13503
|
+
const projectedPoints = [];
|
|
13504
|
+
for (let sampleIndex = 0; sampleIndex < 8; sampleIndex++) {
|
|
13505
|
+
const progress = sampleIndex / 7;
|
|
13506
|
+
const latitude = -0.46 + progress * 0.74;
|
|
13507
|
+
const longitude = baseLongitude +
|
|
13508
|
+
Math.sin(timeMs / 1160 + animationPhase + currentIndex * 0.7 + progress * 2) * 0.035;
|
|
13509
|
+
const scenePoint = transformScenePoint(sampleContinuousOctopusSurfacePoint(surfaceOptions, latitude, longitude), center, rotationX, rotationY);
|
|
13510
|
+
if (scenePoint.z > center.z - size * 0.016) {
|
|
13511
|
+
projectedPoints.push(projectScenePoint(scenePoint, size, sceneCenterX, sceneCenterY));
|
|
13512
|
+
}
|
|
13513
|
+
}
|
|
13514
|
+
if (projectedPoints.length < 3) {
|
|
13515
|
+
continue;
|
|
13516
|
+
}
|
|
13517
|
+
context.beginPath();
|
|
13518
|
+
context.moveTo(projectedPoints[0].x, projectedPoints[0].y);
|
|
13519
|
+
for (const projectedPoint of projectedPoints.slice(1)) {
|
|
13520
|
+
context.lineTo(projectedPoint.x, projectedPoint.y);
|
|
13521
|
+
}
|
|
13522
|
+
context.strokeStyle = currentIndex % 2 === 0 ? `${palette.highlight}3d` : `${palette.accent}33`;
|
|
13523
|
+
context.lineWidth = size * (0.0055 + currentIndex * 0.00045);
|
|
13524
|
+
context.stroke();
|
|
13525
|
+
}
|
|
13526
|
+
context.restore();
|
|
13527
|
+
}
|
|
13528
|
+
/**
|
|
13529
|
+
* Draws small projected sucker highlights on the waving lower mesh lobes.
|
|
13530
|
+
*
|
|
13531
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13532
|
+
*/
|
|
13533
|
+
function drawProjectedTentacleSuckers(options) {
|
|
13534
|
+
const { surfaceOptions, size } = options;
|
|
13535
|
+
const { timeMs } = surfaceOptions;
|
|
13536
|
+
for (const tentacleProfile of surfaceOptions.tentacleProfiles) {
|
|
13537
|
+
if (Math.cos(tentacleProfile.centerLongitude) < -0.12) {
|
|
13538
|
+
continue;
|
|
13539
|
+
}
|
|
13540
|
+
for (let suckerIndex = 0; suckerIndex < 3; suckerIndex++) {
|
|
13541
|
+
const latitude = 0.52 + suckerIndex * 0.14;
|
|
13542
|
+
const sideOffset = tentacleProfile.suckerSide * (0.035 + suckerIndex * 0.012) * tentacleProfile.widthScale;
|
|
13543
|
+
const waveOffset = Math.sin(timeMs / 900 + tentacleProfile.phase + suckerIndex * 0.8) * 0.018;
|
|
13544
|
+
drawProjectedSurfaceSpot({
|
|
13545
|
+
...options,
|
|
13546
|
+
latitude,
|
|
13547
|
+
longitude: tentacleProfile.centerLongitude + sideOffset + waveOffset,
|
|
13548
|
+
radiusScale: size * (0.0065 - suckerIndex * 0.0007),
|
|
13549
|
+
});
|
|
13550
|
+
}
|
|
13551
|
+
}
|
|
13552
|
+
}
|
|
13553
|
+
/**
|
|
13554
|
+
* Draws one tiny projected surface spot by sampling local mesh tangents.
|
|
13555
|
+
*
|
|
13556
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13557
|
+
*/
|
|
13558
|
+
function drawProjectedSurfaceSpot(options) {
|
|
13559
|
+
const { context, surfaceOptions, center, rotationX, rotationY, sceneCenterX, sceneCenterY, size, palette, latitude, longitude, radiusScale, } = options;
|
|
13560
|
+
const localCenter = sampleContinuousOctopusSurfacePoint(surfaceOptions, latitude, longitude);
|
|
13561
|
+
const localHorizontal = sampleContinuousOctopusSurfacePoint(surfaceOptions, latitude, longitude + 0.018);
|
|
13562
|
+
const localVertical = sampleContinuousOctopusSurfacePoint(surfaceOptions, latitude + 0.018, longitude);
|
|
13563
|
+
const sceneCenterPoint = transformScenePoint(localCenter, center, rotationX, rotationY);
|
|
13564
|
+
if (sceneCenterPoint.z <= center.z - size * 0.012) {
|
|
13565
|
+
return;
|
|
13566
|
+
}
|
|
13567
|
+
const projectedCenterPoint = projectScenePoint(sceneCenterPoint, size, sceneCenterX, sceneCenterY);
|
|
13568
|
+
const projectedHorizontalPoint = projectScenePoint(transformScenePoint(localHorizontal, center, rotationX, rotationY), size, sceneCenterX, sceneCenterY);
|
|
13569
|
+
const projectedVerticalPoint = projectScenePoint(transformScenePoint(localVertical, center, rotationX, rotationY), size, sceneCenterX, sceneCenterY);
|
|
13570
|
+
const horizontalRadius = clampNumber$1(Math.hypot(projectedHorizontalPoint.x - projectedCenterPoint.x, projectedHorizontalPoint.y - projectedCenterPoint.y) *
|
|
13571
|
+
radiusScale *
|
|
13572
|
+
0.74, size * 0.003, size * 0.018);
|
|
13573
|
+
const verticalRadius = clampNumber$1(Math.hypot(projectedVerticalPoint.x - projectedCenterPoint.x, projectedVerticalPoint.y - projectedCenterPoint.y) *
|
|
13574
|
+
radiusScale *
|
|
13575
|
+
0.52, size * 0.0024, size * 0.014);
|
|
13576
|
+
const rotation = Math.atan2(projectedHorizontalPoint.y - projectedCenterPoint.y, projectedHorizontalPoint.x - projectedCenterPoint.x);
|
|
13577
|
+
context.save();
|
|
13578
|
+
context.translate(projectedCenterPoint.x, projectedCenterPoint.y);
|
|
13579
|
+
context.rotate(rotation);
|
|
13580
|
+
context.beginPath();
|
|
13581
|
+
context.ellipse(0, 0, horizontalRadius, verticalRadius, 0, 0, Math.PI * 2);
|
|
13582
|
+
context.fillStyle = `${palette.highlight}73`;
|
|
13583
|
+
context.fill();
|
|
13584
|
+
context.strokeStyle = `${palette.highlight}99`;
|
|
13585
|
+
context.lineWidth = Math.max(0.7, size * 0.0028);
|
|
13586
|
+
context.stroke();
|
|
13587
|
+
context.restore();
|
|
13588
|
+
}
|
|
13589
|
+
/**
|
|
13590
|
+
* Resolves a signed angular distance from the source longitude to the target longitude.
|
|
13591
|
+
*
|
|
13592
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13593
|
+
*/
|
|
13594
|
+
function resolveSignedAngularDistance(sourceLongitude, targetLongitude) {
|
|
13595
|
+
return Math.atan2(Math.sin(targetLongitude - sourceLongitude), Math.cos(targetLongitude - sourceLongitude));
|
|
13596
|
+
}
|
|
13597
|
+
/**
|
|
13598
|
+
* Smoothly maps a value between two bounds into `[0, 1]`.
|
|
13599
|
+
*
|
|
13600
|
+
* @private helper of `octopus3d3AvatarVisual`
|
|
13601
|
+
*/
|
|
13602
|
+
function smoothStep(edgeStart, edgeEnd, value) {
|
|
13603
|
+
const progress = clampNumber$1((value - edgeStart) / (edgeEnd - edgeStart), 0, 1);
|
|
13604
|
+
return progress * progress * (3 - 2 * progress);
|
|
13605
|
+
}
|
|
13606
|
+
|
|
13097
13607
|
/* eslint-disable no-magic-numbers */
|
|
13098
13608
|
/**
|
|
13099
13609
|
* Octopus avatar visual.
|
|
@@ -13864,6 +14374,7 @@
|
|
|
13864
14374
|
octopus3AvatarVisual,
|
|
13865
14375
|
octopus3dAvatarVisual,
|
|
13866
14376
|
octopus3d2AvatarVisual,
|
|
14377
|
+
octopus3d3AvatarVisual,
|
|
13867
14378
|
asciiOctopusAvatarVisual,
|
|
13868
14379
|
minecraftAvatarVisual,
|
|
13869
14380
|
minecraft2AvatarVisual,
|