-- For more info on the day/night cycle and examples of cycles see: https://github.com/Refactorio/RedMew/wiki/Day-Night-cycle -- Dependencies local Event = require 'utils.event' local math = require 'utils.math' local RS = require 'map_gen.shared.redmew_surface' -- Localized global table local config = storage.config.day_night -- Localized functions local round = math.round local format = string.format -- Constants local day_night_cycle_keys = { 'ticks_per_day', 'dusk', 'evening', 'morning', 'dawn' } -- Local vars local Public = {} --- Checks that a table is a valid day night cycle. local function check_cycle_validity(day_night_cycle) for _, required_key in pairs(day_night_cycle_keys) do if not day_night_cycle[required_key] then return false end end if (day_night_cycle['dusk'] > day_night_cycle['evening']) or (day_night_cycle['evening'] > day_night_cycle['morning']) or (day_night_cycle['morning'] > day_night_cycle['dawn']) then return false else return true end end --- On init, check the config settings local function init() if config.use_day_night_cycle and config.use_fixed_brightness then error('Cannot use both a day/night cycle and a fixed brightness') return elseif config.use_day_night_cycle then Public.set_cycle(config.day_night_cycle, RS.get_surface()) elseif config.use_fixed_brightness then Public.set_fixed_brightness(config.fixed_brightness, RS.get_surface()) end end Event.on_init(init) -- Public functions --- Sets the day/night cycle according to the table it is given. -- Can only be called during or after init. -- @param day_night_cycle containing specific, required keys: ticks_per_day, dusk, evening, morning, dawn -- @param surface to set the day/night cycle of -- @returns true if set properly, nil if not -- @see Venus::world_settings function Public.set_cycle(day_night_cycle, surface) if not check_cycle_validity(day_night_cycle) then error('Provided day/night cycle is invalid') return end if not surface or not surface.valid then error('Provided surface is invalid') return end if not Public.unfreeze_daytime then error('Time is stuck in a frozen state') return end surface.ticks_per_day = day_night_cycle.ticks_per_day surface.dusk = 0 surface.evening = 0.001 surface.morning = 0.002 surface.dawn = day_night_cycle.dawn surface.morning = day_night_cycle.morning surface.evening = day_night_cycle.evening surface.dusk = day_night_cycle.dusk return true end --- Sets the brightness to a fixed level -- Can only be called during or after init. -- @param daylight between 0.15 and 1 representing the percentage of daylight (0.15 brightness is the darkest available) -- @param surface to set the day/night cycle of -- @return true if time is set properly, nil if not function Public.set_fixed_brightness(daylight, surface) if not surface or not surface.valid then error('Provided surface is invalid') return end if daylight < 0.15 then error('Daylight set too low. 0.15 is the darkest available.') return elseif daylight > 1 then error('Daylight set too high. 1.00 is the lightest available.') return end -- Do some algebra to find the correct point in the daytime to freeze local brightness_slope = math.calculate_slope(surface.morning, 0.0, surface.dawn, 1) local y_intercept = math.calculate_y_intercept(surface.morning, 0.0, brightness_slope) local breakpoint = (daylight - y_intercept) / brightness_slope -- Set the daytime to the point we just calculated surface.daytime = breakpoint -- Freeze the day/night cycle surface.freeze_daytime = true Debug.print(format('breakpoint/surface_daytime: %s/%s', breakpoint, surface.daytime)) Debug.print(format('brightness/surface_brightness: %s/%s', round(daylight, 2), round((1 - surface.darkness), 2))) if round(daylight, 2) == round((1 - surface.darkness), 2) then return true end end --- Unfreezes daytime (usually frozen by set_fixed_brightness) -- Can only be called during or after init. -- @param surface to unfreeze the day/night cycle of -- @return true if daytime unfrozen, nil if not function Public.unfreeze_daytime(surface) if not surface or not surface.valid then log('Provided surface is invalid') return end surface.freeze_daytime = false return true end return Public