extern crate chrono; extern crate clap; extern crate spa; use chrono::prelude::*; use clap::{value_t_or_exit, App, Arg}; use spa::{calc_sunrise_and_set, SunriseAndSet}; use std::os::raw::{c_ushort, c_void}; use std::ptr; use std::thread; use std::time::Duration; use x11::xlib::{XDefaultScreen, XFree, XOpenDisplay, XRootWindow}; use x11::xrandr::{ XRRAllocGamma, XRRCrtcGamma, XRRGetCrtcGammaSize, XRRGetScreenResourcesCurrent, XRRSetCrtcGamma, }; fn set_temp(temp: u32) { let ratio: f64 = (temp % 500) as f64 / 500f64; unsafe { let display = XOpenDisplay(ptr::null_mut()); let screen = XDefaultScreen(display); let root = XRootWindow(display, screen); let resource = XRRGetScreenResourcesCurrent(display, root); for x in 0..(*resource).ncrtc { let crtcxid = (*resource).crtcs.offset(x as isize); let size = XRRGetCrtcGammaSize(display, *crtcxid); let crtc_gamma: *mut XRRCrtcGamma = XRRAllocGamma(size); let gamma = sctd::avg(temp, ratio); for i in 0..size { let g = (65535f64 * i as f64) / size as f64; *((*crtc_gamma).red as *mut c_ushort).offset(i as isize) = (g * gamma.red) as u16; *((*crtc_gamma).green as *mut c_ushort).offset(i as isize) = (g * gamma.green) as u16; *((*crtc_gamma).blue as *mut c_ushort).offset(i as isize) = (g * gamma.blue) as u16; } XRRSetCrtcGamma(display, *crtcxid, crtc_gamma); XFree(crtc_gamma as *mut c_void); } } } fn get_temp(utc: DateTime, ss: &SunriseAndSet) -> u32 { let low_temp = 2500; let high_temp = 5500; match *ss { SunriseAndSet::Daylight(sunrise, sunset) => { let since_sunrise = utc.signed_duration_since(sunrise); let since_sunset = utc.signed_duration_since(sunset); // this is where gradual increase/decrease should happen if since_sunrise.num_seconds() < 0 { return low_temp; } else if since_sunset.num_seconds() < 0 { return high_temp; } else { return low_temp; } } SunriseAndSet::PolarDay => return high_temp, SunriseAndSet::PolarNight => return low_temp, } } fn main() { let matches = App::new("sctd") .about("set color temperature daemon") .arg( Arg::with_name("latitude") .long("latitude") .takes_value(true), ) .arg( Arg::with_name("longitude") .long("longitude") .takes_value(true), ) .arg(Arg::with_name("reset").long("reset")) .get_matches(); if matches.is_present("reset") { set_temp(5500); } else { let latitude = value_t_or_exit!(matches, "latitude", f64); let longitude = value_t_or_exit!(matches, "longitude", f64); loop { let utc: DateTime = Utc::now(); match calc_sunrise_and_set(utc, latitude, longitude) { Ok(ss) => set_temp(get_temp(utc, &ss)), Err(e) => println!("Error calculating sunrise and sunset: {:?}", e), } thread::sleep(Duration::from_secs(300)); } } }