#region Usings using System; using System.Text; using System.IO; using System.Globalization; using System.Diagnostics; using System.Configuration; using System.Collections; using System.Collections.Specialized; #endregion // *** Raw Therapee sample Custom Profile builder (version 2013-08-12) *** // // // WARNING: The command line parameters has changed since this file has been created by Oduis. The new mechanism involves a // temporary communication file (.ini style) to provide system parameters and metadata read by RawTherapee. This script has // to be updated by some C# developer in order to work. // // // WARNING: PP3 format may change in the future versions! If this happens there will probably be no automatic migration path, // you'll have to adjust on your own. This is a sample, and therefore not supported by the RT team (just by oduis) // // // How to use: // 1. Modify the GetCorrectedSettings function below according to your needs. // 2. Download and install Microsoft .Net Runtime (latest version is 4.0 as of writing), if it's not already on your machine. // You can get it for free via Windows Update or from microsoft.com. No need for Visual Studio etc. // 3. Open a command line and compile this CS-File using the C# 32bit compiler. It is usually installed somewhere here: // C:\Windows\Microsoft.NET\Framework\v4.0.30319\csc.exe // Call csc.exe (C#-Compiler) with your .CS file as parameter like this (one big line): // // C:\Windows\Microsoft.NET\Framework\v4.0.30319\csc // /r:C:\Windows\Microsoft.NET\Framework\v4.0.30319\System.dll // /r:C:\Windows\Microsoft.NET\Framework\v4.0.30319\System.Configuration.dll // RTProfileBuilderSample.cs // // (On most machines it already works with "C:\Windows\Microsoft.NET\Framework\v4.0.30319\csc RTProfileBuilderSample.cs") // CSC will compile it and emit an EXE. // 4. Open your RT options files and find the entry [Profiles]/CustomProfileBuilder // 5. Enter the path to your newly built exe here. On Windows, don't forget double slashes (e.g. "C:\\MyDir\\Mybuilder.exe") // And you're done! The EXE is only called on opening the image editor and there is no PP3 already // // If you want to use EXIFTOOL to gather more details information to build queries: // 1. Download exiftool.exe from http://www.sno.phy.queensu.ca/~phil/exiftool/ // 2. Rename it to exiftool.exe (NOT exiftool(-k).. or something!) // 3. Copy the RTProfilerBuilder.exe.config next to your own EXE. If you renamed it, rename config to "(Yourname).exe.config" // 4. Open the config with notepad (it's an XML file). Set ExifToolPath to your downloaded and renamed exe // // If you want to know what parameters are available, call "exiftool.exe -tab -short" // // This description is for Windows. The C# code does not use anything fancy, will probably work with MONO on Linux/OSX, too namespace RTProfilerBuilder { /// Main class. Mostly change GetCorrectedSettings. class RTProfileBuilder { /// Adds the Nikkor zoom distortion correction profile. /// First array is list of focal lengths, second array is the RT setting that should correct the /// distortion for the corresponding focal length. Values between these values are automatically interpolated. /// The focal length values must already be ordered. The number of sample points is not limited. static DistortionCorrectProf distNikkor24120f4 = new DistortionCorrectProf( new double[] { 24, 28, 35, 50, 70, 85, 120 }, new double[] { -0.1, -0.063, -0.012, 0.018, 0.034, 0.04, 0.048 } ); /// This is your personalisation function /// Full EXIF from EXIFTOOL (if configured). /// Entry, like "Sharpening/Radius" /// Current value (from default file) /// FNumberExposure in seconds /// Focal length in MMISO value /// Lens from EXIFCamera from EXIF /// The value to be written. Simply take the current value if you have nothing to touch. static string GetCorrectedSetting(NameValueCollection exif, string sectionEntry, string value, double fNumber, double exposureSecs, double focalLength, long iso, string lens, string camera) { string s; // We don't do anything to the value if it's not our camera if (camera.EndsWith("NIKON D700", StringComparison.InvariantCultureIgnoreCase) && lens.Contains("24.0-120.0 mm f/4.0")) { switch (sectionEntry) { // Here is the place to adjust your settings // Pretty simple: "SectionName/EntryName" in options file case "Vignetting Correction/Amount": value = (fNumber < 8 && focalLength < 30) ? "30" : "0"; break; case "RAW/CA": value = ToBool(fNumber < 11); // Means "Enabled if fnumber<11, otherwise disabled" break; case "Impulse Denoising/Enabled": value = ToBool(iso >= 3200); break; case "HLRecovery/Enabled": value = ToBool(iso >= 1600); // Dynamic range decreases, so we'll probably need it break; case "Color Boost/Amount": if (iso >= 6400) value = "0"; // Colors will get poppy anyway... break; case "Distortion/Amount": // we already checked in the IF upstairs that this is "our" lens value = distNikkor24120f4.GetDistortionAmount(focalLength); break; // Add other parameters here. Mention this is case sensitive! default: break; // we don't touch values we don't care about } } // end if camera=xxx // This is for camera independent settings switch (sectionEntry) { // These are parsed from EXIFTOOL and XMP in DNG (see http://en.wikipedia.org/wiki/Extensible_Metadata_Platform) case "IPTC/City": s = exif.Get("City"); if (!String.IsNullOrEmpty(s)) value = s; break; case "IPTC/Country": s = exif.Get("Country"); if (!String.IsNullOrEmpty(s)) value = s; break; case "IPTC/Caption": case "IPTC/Title": s = exif.Get("Headline"); if (!String.IsNullOrEmpty(s)) value = s; break; // Add other parameters here. Mention this is case sensitive! default: break; // we don't touch values we don't care about } return value; } #region * Main and Helpers static string ToBool(bool condition) { return condition ? "true" : "false"; } static string ToFloat(float f) { return f.ToString(CultureInfo.InvariantCulture); } /// Reads default file and parses it. No need to touch it for your personal settings. /// Command line args /// 0 on all OK. static int Main(string[] args) { int exitCode = 0; try { #region Parse input parameters int argNo = 0; // Name of raw/JPG to process string sourceFile = args[argNo++]; // What the user selected as his base profile string defaultProfileFilePath = args[argNo++]; // Cache directory, for any logging file string cachePath = args[argNo++]; // True if the image is only being flagged as inTrash, rank or colorLabel but still need valid PP3 - actually not used by this script bool forFlaggingPurpose = bool.Parse(args[argNo++], CultureInfo.InvariantCulture); // Note that old C++ has no automatic number globalization double fNumber = double.Parse(args[argNo++], CultureInfo.InvariantCulture); double exposureSecs = double.Parse(args[argNo++], CultureInfo.InvariantCulture); double focalLength = double.Parse(args[argNo++], CultureInfo.InvariantCulture); long iso = long.Parse(args[argNo++], CultureInfo.InvariantCulture); string lens = args[argNo++]; string cameraMake = args[argNo++]; string cameraModel = args[argNo++]; string camera = cameraMake + " " + cameraModel; #endregion // Read default file as basis string[] lines = File.ReadAllLines(defaultProfileFilePath); NameValueCollection nvEXIF = ParseFullExifData(sourceFile); // File should be Windows ANSI using (TextWriter tw = new StreamWriter(sourceFile + ".pp3", false, new UTF8Encoding(false))) { string section = ""; foreach (string line in lines) { string l = line.Trim(); if (!String.IsNullOrEmpty(line)) { if (l.StartsWith("[")) section = l.Trim(new char[] { '[', ']' }); else if (char.IsLetterOrDigit(l[0]) && l.Contains("=")) { int valPos = l.IndexOf("=") + 1; string newValue = GetCorrectedSetting(nvEXIF, section + "/" + l.Substring(0, valPos - 1), l.Substring(valPos).Trim(), fNumber, exposureSecs, focalLength, iso, lens, camera); // Merge in new value l = l.Substring(0, valPos) + (newValue ?? ""); } } tw.WriteLine(l); } } } catch (Exception ex) { Console.WriteLine("Error: " + ex.ToString()); // can be seen in the RT console window exitCode = 1; } return exitCode; } static NameValueCollection ParseFullExifData(string filePath) { NameValueCollection nv = new NameValueCollection(); string exifToolPath = ConfigurationManager.AppSettings["ExifToolPath"]; if (!String.IsNullOrEmpty(exifToolPath)) { ProcessStartInfo psi = new ProcessStartInfo(exifToolPath, "\"" + filePath + "\" -tab -short"); psi.CreateNoWindow = false; psi.UseShellExecute = false; psi.StandardOutputEncoding = System.Text.Encoding.UTF8; psi.RedirectStandardOutput = true; Process p = Process.Start(psi); using (StreamReader sr = p.StandardOutput) { while (!sr.EndOfStream) { string line = sr.ReadLine(); if (line.Contains("\t")) { string[] split = line.Split('\t'); nv.Add(split[0], split[1]); } } } p.WaitForExit(); } return nv; } #endregion } #region DistortionCorrectProf /// Holds a distortion correction profile for one lens. Uses sample points (focal length vs. dist. correction) as input. class DistortionCorrectProf { double[] adFocLen, adCorrect; /// Parses array to internal structure /// Focal lengths /// Correction factors public DistortionCorrectProf(double[] focLen, double[] correct) { if (focLen == null || correct == null || focLen.Length != correct.Length || focLen.Length < 2) throw new Exception("DistortionCorrectProf inputs must be valid and of the same lengths, at least 2 points"); adFocLen = focLen; adCorrect = correct; for (int i = 0; i < adFocLen.Length - 1; i++) if (adFocLen[i] >= adFocLen[i + 1]) throw new Exception("The distortion correction focal length points must be ordered!"); } /// Calculates regression value of RT distortion amount for the given focal length. /// Input focal length. /// Distortion in RT format. public string GetDistortionAmount(double focalLength) { // if it's out of area (which should just happen with e.g. rounding errors), return flat defaults. if (focalLength <= adFocLen[0]) return adCorrect[0].ToString("G", CultureInfo.InvariantCulture); if (focalLength >= adFocLen[adFocLen.Length - 1]) return adCorrect[adFocLen.Length - 1].ToString("G", CultureInfo.InvariantCulture); for (int i = 0; i < adFocLen.Length - 1; i++) { if (focalLength >= adFocLen[i] && focalLength < adFocLen[i + 1]) { // from the sample curves taken so far, it it safe to take a simple linear interpolation here double corr = adCorrect[i] + (adCorrect[i + 1] - adCorrect[i]) * (focalLength - adFocLen[i]) / (adFocLen[i + 1] - adFocLen[i]); return corr.ToString("G3", CultureInfo.InvariantCulture); } } return ""; // should never happen } } #endregion }