
ループがあるのでCustomノードに書きます。発散の判定は絶対値が2を超えた時点とし、発散と判定されるまでの計算回数が多い程輝度が上がるようにしました。

探検できるように300倍に拡大した'Plane'にマテリアルを貼り付けます。
Spectator Pawnで探検します。









離れてみると分かりにくいですが、近くから輪郭部を見ると非常に輝度が高いのがわかります。集合の左側は特に輝度が高く、更に拡大しても複雑なパターンが現れてきそうです。
プロジェクトをgithubにアップしました。
struct Wave | |
{ | |
Vec2 dir; | |
float amplitude; | |
float waveLength; | |
}; | |
struct ImmutableCB | |
{ | |
float waveSteepness; | |
float waveSpeed; | |
Wave waves[20]; | |
}; | |
static Vec3 CalcGerstnerWaveOffset(const ImmutableCB& cb, Vec3 location, float time) | |
{ | |
Vec3 sum = Vec3(0, 0, 0); | |
for (int i = 0; i < dimof(cb.waves); i++) | |
{ | |
const Wave& wave = cb.waves[i]; | |
float wi = 2 / wave.waveLength; | |
float rad = (dot(wave.dir, Vec2(location.x, location.z)) + time * cb.waveSpeed) * wi; | |
float sine = std::sin(rad); | |
float cosine = std::cos(rad); | |
sum.y += sine * wave.amplitude; | |
sum.x += wave.dir.x * cosine * cb.waveSteepness / (wi * dimof(cb.waves)); | |
sum.z += wave.dir.y * cosine * cb.waveSteepness / (wi * dimof(cb.waves)); | |
} | |
return sum; | |
} |
Vec3 center = pos + CalcGerstnerWaveOffset(immutableCb, pos, terrainFrameCB.g_time); | |
Vec3 offsetXZ = pos + Vec3(1, 0, 1) + CalcGerstnerWaveOffset(immutableCb, pos + Vec3(1, 0, 1), terrainFrameCB.g_time); | |
Vec3 offsetNegX = pos + Vec3(-1, 0, 0) + CalcGerstnerWaveOffset(immutableCb, pos + Vec3(-1, 0, 0), terrainFrameCB.g_time); | |
Vec3 offsetNegZ = pos + Vec3(0, 0, -1) + CalcGerstnerWaveOffset(immutableCb, pos + Vec3(0, 0, -1), terrainFrameCB.g_time); | |
Vec3 normal = normalize(cross(offsetNegZ - offsetXZ, offsetNegX - offsetXZ)); | |
Vec3 rotAxis = normalize(cross(normal, Vec3(0, 1, 0))); | |
float rotRad = acos(dot(normal, Vec3(0, 1, 0))); | |
cube.Draw(MeshXAnimResult(), scale(5, 5, 5) * q2m(Quat(rotAxis, -rotRad)) * translate(center.x, center.y, center.z)); |
float3 ofsByGerstnerWave = CalcGerstnerWaveOffset(vertexPosition); | |
float3 dx = float3(0.01, 0, 0) + CalcGerstnerWaveOffset(vertexPosition + float3(0.01, 0, 0)); | |
float3 dz = float3(0, 0, 0.01) + CalcGerstnerWaveOffset(vertexPosition + float3(0, 0, 0.01)); | |
float3 N = normalize(cross(dz - ofsByGerstnerWave, dx - ofsByGerstnerWave)); |
float3 CalcGerstnerWaveNormal(float3 P) | |
{ | |
float3 normal = float3(0, 1, 0); | |
[unroll] | |
for (int i = 0; i < numWaves; i++) | |
{ | |
Wave wave = waves[i]; | |
float wi = 2 / wave.waveLength; | |
float WA = wi * wave.amplitude; | |
float phi = speed * wi; | |
float rad = wi * dot(wave.dir, P.xz) + phi * g_time; | |
float Qi = steepness / (wave.amplitude * wi * numWaves); | |
normal.xz -= wave.dir * WA * cos(rad); | |
normal.y -= Qi * WA * sin(rad); | |
} | |
return normalize(normal); | |
} | |
float3 CalcGerstnerWaveBinormal(float3 P) | |
{ | |
float3 binormal = float3(1, 0, 0); | |
[unroll] | |
for (int i = 0; i < numWaves; i++) | |
{ | |
Wave wave = waves[i]; | |
float wi = 2 / wave.waveLength; | |
float WA = wi * wave.amplitude; | |
float phi = speed * wi; | |
float rad = wi * dot(wave.dir, P.xz) + phi * g_time; | |
float Qi = steepness / (wave.amplitude * wi * numWaves); | |
binormal.x -= Qi * wave.dir.x * wave.dir.x * WA * sin(rad); | |
binormal.z -= Qi * wave.dir.x * wave.dir.y * WA * sin(rad); | |
binormal.y += wave.dir.x * WA * cos(rad); | |
} | |
return normalize(binormal); | |
} | |
float3 CalcGerstnerWaveTangent(float3 P) | |
{ | |
float3 tangent = float3(0, 0, 1); | |
[unroll] | |
for (int i = 0; i < numWaves; i++) | |
{ | |
Wave wave = waves[i]; | |
float wi = 2 / wave.waveLength; | |
float WA = wi * wave.amplitude; | |
float phi = speed * wi; | |
float rad = wi * dot(wave.dir, P.xz) + phi * g_time; | |
float Qi = steepness / (wave.amplitude * wi * numWaves); | |
tangent.x -= Qi * wave.dir.x * wave.dir.y * WA * sin(rad); | |
tangent.z -= Qi * wave.dir.y * wave.dir.y * WA * sin(rad); | |
tangent.y += wave.dir.y * WA * cos(rad); | |
} | |
return normalize(tangent); | |
} |
float3 gerstnerWaveN = CalcGerstnerWaveNormal(vertexPosition + ofsByGerstnerWave); | |
float3 gerstnerWaveB = CalcGerstnerWaveBinormal(vertexPosition + ofsByGerstnerWave); | |
float3 gerstnerWaveT = CalcGerstnerWaveTangent(vertexPosition + ofsByGerstnerWave); | |
float3x3 rotator; | |
rotator[0] = gerstnerWaveB; | |
rotator[1] = gerstnerWaveN; | |
rotator[2] = gerstnerWaveT; | |
output.normal = mul(water_normal.xyz, rotator); |
struct Wave | |
{ | |
float2 dir; | |
float amplitude; | |
float waveLength; | |
}; | |
cbuffer cb2 : register(b2) | |
{ | |
Wave waves[100]; | |
}; | |
static int numWaves = 20; | |
static float steepness = 0.8; | |
static float speed = 15; | |
float3 CalcGerstnerWaveOffset(float3 v) | |
{ | |
float3 sum = float3(0, 0, 0); | |
[unroll] | |
for (int i = 0; i < numWaves; i++) | |
{ | |
Wave wave = waves[i]; | |
float wi = 2 / wave.waveLength; | |
float Qi = steepness / (wave.amplitude * wi * numWaves); | |
float phi = speed * wi; | |
float rad = dot(wave.dir, v.xz) * wi + g_time * phi; | |
sum.y += sin(rad) * wave.amplitude; | |
sum.xz += cos(rad) * wave.amplitude * Qi * wave.dir; | |
} | |
return sum; | |
} |
struct Wave | |
{ | |
Vec2 dir; | |
float amplitude; | |
float waveLength; | |
}; | |
struct ImmutableCB | |
{ | |
Wave waves[100]; | |
} cb; | |
for (int i = 0; i < dimof(cb.waves); i++) | |
{ | |
Wave& w = cb.waves[i]; | |
float randomRad = (float)(Random() * M_PI * 2 * 0.3f); | |
w.dir.x = sinf(randomRad); | |
w.dir.y = cosf(randomRad); | |
w.amplitude = 0.03f + powf(2.0f, (float)Random() * 2.0f) * 0.05f; | |
w.waveLength = 1.0f + powf(2.f, 1.f + (float)Random()) * 10.f; | |
} |
gdal_translate.exe -of PNG -ot UInt16 -scale -32768 32767 0 65535 N042E140_AVE_DSM.tif
float3 lightPosition = { 100, 0, 1000 }; // This makes an implicit constant buffer, but the light position never assigned! | |
float3 CalcLightDir(float3 myPosition) | |
{ | |
return normalize(lightPosition - myPosition); | |
} |
static float3 lightPosition = { 100, 0, 1000 }; // 'static' needed. not 'const'! | |
float3 CalcLightDir(float3 myPosition) | |
{ | |
return normalize(lightPosition - myPosition); | |
} |
D3D11 ERROR: ID3D11Device::CreateTexture2D: The format (0x28, D32_FLOAT) cannot be bound as a ShaderResource or cast to a format that could be bound as a ShaderResource. Therefore this format cannot support D3D11_BIND_SHADER_RESOURCE. Specifiying the format R32_TYPELESS instead would solve this issue. [ STATE_CREATION ERROR #92: CREATETEXTURE2D_UNSUPPORTEDFORMAT]