Skip to content

Commit

Permalink
Initial commit.
Browse files Browse the repository at this point in the history
  • Loading branch information
LambdAurora committed Jul 2, 2020
0 parents commit 06010fd
Show file tree
Hide file tree
Showing 4 changed files with 191 additions and 0 deletions.
70 changes: 70 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
#
# LambdAurora's ignore file
#
# v0.13

# JetBrains
.idea/
*.iml
*.ipr
*.iws
## Intellij IDEA
out/
## CLion
cmake-build-debug*/
cmake-build-release*/
## Eclipse
eclipse
*.launch
.settings
.metadata
.classpath
.project
## Visual Studio
.vs/
CMakeSettings.json

# Build system
## Cargo
Cargo.lock
## CMake
CMakeCache.txt
CMakeFiles/
## Gradle
.gradle/
## Node.JS
node_modules/

# Editors
## VSCode
.vscode/

# Logging
logs/

# Languages
## Java
classes/
## Python
__pycache__/
## Rust
**/*.rs.bk

# OS
## Windows
desktop.ini

# File types
*.dll
*.so
*.dylib
*.lib
lib*.a
*.png~

# Common
bin/
build/
dist/
run/
target/
76 changes: 76 additions & 0 deletions HOW_DOES_IT_WORK.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
# How does it work?

This is a little file explaining how the mod works and hopefully be as detailed as it can
to allow people to maintain the code over newer versions and remove the monopole OptiFine has.

## [UpcraftLP method](https://gist.github.com/UpcraftLP/93db478535cb3954cc9f3dc98bac91cf) & [AtomicStryker method](https://github.com/AtomicStryker/atomicstrykers-minecraft-mods/tree/1.14.3/DynamicLights)

Not used by this mod.

UpcraftLP method can be found here: https://gist.github.com/UpcraftLP/93db478535cb3954cc9f3dc98bac91cf

AtomicStryker method can be found here: https://github.com/AtomicStryker/atomicstrykers-minecraft-mods/tree/1.14.3/DynamicLights

Both methods suffer from a "laggy" dynamic lighting, the way the light moves is too much tied to
the block positions as it only injects at luminance getters in `WorldChunk` or a lighting provider.
It will only provide the dynamic light value of the light source if the block position is the position
of the light source.

It's better in a way as the F3 menu will actually report a light value instead of the lightmap coordinates
method, but will be significantly be less smooth in its transitions.

## Lightmap coordinates method

This method is used by this mod.

### Dynamic luminance injection

As [@Martmists](https://github.com/martmists) suggested, `WorldRenderer#getLightmapCoordinates` is a great entrypoint.

This method gives lightmap coordinates in the format `(skyLevel << 20 | blockLevel << 4)`
and to get the blockLevel from the lightmap use `LightmapTextureManager#getBlockLightCoordinates`.

The goal is to inject at TAIL into `getLightmapCoordinates` and get the vanilla value, then
get the dynamic value at the specified block position (which is calculated from all the dynamic
light sources, their distances and the luminance), if the dynamic value is higher than the block value
then we replace it.

This also means that we can't just do like the previous method and only give the source luminance but
we also have to calculate the surrounding luminance created by the dynamic light source in a specified
range (which is 7.75 to limit chunk rebuilding to 8 chunks (which is still a lot)).

When getting the dynamic light value at the specified position, it has to be a `double` and not
an integer as the light value calculated with the range has to be precise.
To modify the lightmap with the dynamic light value, it has to be multiplied by 16.0 instead of
using a bitshift to preserve as much as possible the precision.

Dynamic light value at a specified position is calculated in a for-loop with all the dynamic light
sources, only the highest light value is kept. The light value is calculated as follow:

```java
// dist being the distance between the light source origin and the position where the
// lightmap coordinates are requested.
double multiplier = 1.0 - dist / 7.75; // 7.75 because of the range
// luminace being the luminance of the dynamic light source.
double lightLevel = multiplier * (double) luminance;
```

### Chunk rebuilding

To apply the dynamic lighting we have to request the affected chunks a rebuild to rebuild the lightmap,
which is the most critic part performance-wise. Rebuilding a chunk cost performances,
that's why there's multiple mode: FASTEST, FAST and FANCY. The fast modes will limit the update.

To manage chunk rebuilding, we inject at `WorldRenderer#setupTerrain` (`SodiumWorldRender#setupTerrain` if Sodium is present),
and if fast mode then we check when the last update was done.
If we can update, then we get the current dynamic light source position and get its chunk.
After that we get the surrounding affect chunks and build a new Set of chunk coordinates which
need to be updated.

## Shader based method

This would be more ideal performance-wise as it would give the smoothness of the second method
and avoid too many chunk rebuilding.

This has still to be researched, if a method is found using shaders this mod is likely to switch
to this method.
21 changes: 21 additions & 0 deletions LICENSE
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
MIT License

Copyright © 2020 LambdAurora <[email protected]>

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
24 changes: 24 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
# LambdaDynamicLights

![Java 8](https://img.shields.io/badge/language-Java%208-9B599A.svg?style=flat-square)
[![GitHub license](https://img.shields.io/github/license/LambdAurora/LambdaDynamicLights?style=flat-square)](https://raw.githubusercontent.com/LambdAurora/LambdaDynamicLights/master/LICENSE)
![Environment: Client](https://img.shields.io/badge/environment-client-1976d2?style=flat-square)
![Mod loader: Fabric](https://img.shields.io/badge/modloader-Fabric-1976d2?style=flat-square&logo=)
![Version](https://img.shields.io/github/v/tag/LambdAurora/LambdaDynamicLights?label=version&style=flat-square)

A dynamic lights mod for Fabric.

## What's this mod?

This mod adds dynamic lights to Minecraft. Dynamic lights are lights created by an entity holding an
item which makes light as a block, or create by an entity on fire, etc.

Dynamic lights are present as an option in OptiFine or as a Forge mod (Dynamic Lights mod).

## Build

Just do `./gradlew build` and everything should build just fine!

## How does it work internally?

Check [this documentation](https://github.com/LambdAurora/LambdaDynamicLights/blob/mc1.16/HOW_DOES_IT_WORK.md).

0 comments on commit 06010fd

Please sign in to comment.