6#include "CLHEP/Units/PhysicalConstants.h"
9#include "G4UImanager.hh"
16#include <unordered_map>
34 size_t startPos = input.find_first_not_of(
" \t");
35 size_t endPos = input.find_last_not_of(
" \t");
38 if (startPos == std::string::npos || endPos == std::string::npos) {
return ""; }
41 return input.substr(startPos, endPos - startPos + 1);
54 while (!s.empty() && std::isspace(
static_cast<unsigned char>(s.front()))) s.remove_prefix(1);
55 while (!s.empty() && std::isspace(
static_cast<unsigned char>(s.back()))) s.remove_suffix(1);
66 result.erase(std::remove(result.begin(), result.end(),
' '), result.end());
76 std::size_t lastSlashPos = path.find_last_of(
'/');
77 if (lastSlashPos == std::string::npos) {
81 return path.substr(lastSlashPos + 1);
90 auto lastSlash = path.find_last_of(
'/');
91 if (lastSlash == std::string::npos)
return ".";
92 return path.substr(0, lastSlash);
95namespace fs = std::filesystem;
104 std::vector<std::string> pvalues;
105 std::stringstream plist(input);
107 while (plist >> tmp) {
109 if (!trimmed.empty()) { pvalues.push_back(trimmed); }
120 const std::string& replacement) {
122 for (
const char& ch : input) {
123 if (toReplace.find(ch) != std::string::npos) { output.append(replacement); }
124 else { output.push_back(ch); }
135 if (from.empty())
return source;
139 size_t findPos = source.find(from, lastPos);
141 while (findPos != string::npos) {
143 newString.append(source, lastPos, findPos - lastPos);
145 lastPos = findPos + from.length();
146 findPos = source.find(from, lastPos);
150 newString += source.substr(lastPos);
161string fillDigits(
const string& word,
const string& c,
int ndigits) {
162 if (c.empty() || ndigits <=
static_cast<int>(word.size()))
return word;
166 int toFill = ndigits -
static_cast<int>(word.size());
167 filled.reserve(ndigits);
169 filled.append(toFill, c[0]);
195static bool parse_double_clocale(std::string_view sv,
double& out) {
198 _locale_t loc = _create_locale(LC_NUMERIC,
"C");
200 out = _strtod_l(tmp.c_str(), &end, loc);
202 return end == tmp.c_str() + tmp.size();
204 locale_t loc = newlocale(LC_NUMERIC_MASK,
"C", (locale_t)0);
206 out = strtod_l(tmp.c_str(), &end, loc);
208 return end == tmp.c_str() + tmp.size();
217 std::cerr <<
FATALERRORL <<
"empty numeric string.\n";
222 if (value.find(
'.') == string::npos) {
223 size_t firstComma = value.find(
',');
224 if (firstComma != string::npos && value.find(
',', firstComma + 1) == string::npos) {
229 const size_t starCount =
static_cast<size_t>(std::count(value.begin(), value.end(),
'*'));
232 if (value.find(
'*') == string::npos) {
235 if (value.find(
'.') == string::npos) {
236 auto firstComma = value.find(
',');
237 if (firstComma != string::npos && value.find(
',', firstComma + 1) == string::npos)
240 if (!parse_double_clocale(value, out)) {
241 std::cerr <<
FATALERRORL <<
"missing '*' before unit or invalid number in <" << v <<
">.\n";
244 if (warnIfNotUnit && out != 0.0) {
245 std::cerr <<
" ! Warning: value " << v <<
" does not contain units." << std::endl;
253 std::cerr <<
FATALERRORL <<
"multiple '*' separators are not allowed in <" << v <<
">.\n";
258 const size_t pos = value.find(
'*');
261 if (left.empty() || right.empty()) {
262 std::cerr <<
FATALERRORL <<
"expected '<number>*<unit>', got <" << v <<
">.\n";
267 if (left.find(
'.') == string::npos) {
268 auto c = left.find(
',');
269 if (c != string::npos && left.find(
',', c + 1) == string::npos)
273 double numeric = 0.0;
274 if (!parse_double_clocale(left, numeric)) {
275 std::cerr <<
FATALERRORL <<
"invalid numeric part before '*' in <" << v <<
">.\n";
287 static const std::unordered_map<string, double> unitConversion = {
289 {
"m", CLHEP::m}, {
"cm", CLHEP::cm}, {
"mm", CLHEP::mm},
290 {
"um", 1E-6 * CLHEP::m}, {
"fm", 1E-15 * CLHEP::m},
291 {
"inch", 2.54 * CLHEP::cm}, {
"inches", 2.54 * CLHEP::cm},
293 {
"deg", CLHEP::deg}, {
"degrees", CLHEP::deg}, {
"arcmin", CLHEP::deg / 60.0},
294 {
"rad", CLHEP::rad}, {
"mrad", CLHEP::mrad},
296 {
"ev", CLHEP::eV}, {
"kev", 1e3 * CLHEP::eV}, {
"mev", CLHEP::MeV}, {
"gev", CLHEP::GeV},
298 {
"t", CLHEP::tesla}, {
"tesla", CLHEP::tesla}, {
"t/m", CLHEP::tesla / CLHEP::m},
299 {
"gauss", CLHEP::gauss}, {
"kilogauss", 1000.0 * CLHEP::gauss},
301 {
"s", CLHEP::s}, {
"ns", CLHEP::ns}, {
"ms", CLHEP::ms}, {
"us", CLHEP::us},
307 if (
auto it = unitConversion.find(unit); it != unitConversion.end()) {
308 return numeric * it->second;
312 auto si_prefix_factor = [](
char p) ->
double {
314 case 'Y':
return 1e24;
315 case 'Z':
return 1e21;
316 case 'E':
return 1e18;
317 case 'P':
return 1e15;
318 case 'T':
return 1e12;
319 case 'G':
return 1e9;
320 case 'M':
return 1e6;
321 case 'k':
return 1e3;
322 case 'h':
return 1e2;
323 case 'd':
return 1e-1;
324 case 'c':
return 1e-2;
325 case 'm':
return 1e-3;
326 case 'u':
return 1e-6;
327 case 'n':
return 1e-9;
328 case 'p':
return 1e-12;
329 case 'f':
return 1e-15;
330 case 'a':
return 1e-18;
331 case 'z':
return 1e-21;
332 case 'y':
return 1e-24;
337 if (unit.size() >= 2) {
338 const double pf = si_prefix_factor(unit.front());
340 const string base = unit.substr(1);
341 if (
auto it2 = unitConversion.find(base); it2 != unitConversion.end()) {
342 return numeric * pf * it2->second;
348 std::cerr <<
GWARNING <<
">" << right <<
"<: unit not recognized for string <" << v <<
">" << std::endl;
354 string gnumber = std::to_string(input) +
"*" + unit;
359 vector<double> output;
360 output.reserve(vstring.size());
362 for (
const auto& s : vstring) { output.push_back(
getG4Number(s, warnIfNotUnit)); }
374 std::ifstream in(filename);
376 std::cerr <<
FATALERRORL <<
"can't open input file " << filename <<
". Check your spelling. " << std::endl;
380 std::stringstream strStream;
381 if (verbosity > 0) { std::cout << std::endl <<
CIRCLEITEM <<
" Loading string from " << filename << std::endl; }
382 strStream << in.rdbuf();
385 string parsedString = strStream.str();
389 while ((nFPos = parsedString.find(commentChars)) != string::npos) {
390 size_t firstNL = parsedString.rfind(
'\n', nFPos);
391 size_t secondNL = parsedString.find(
'\n', nFPos);
392 parsedString.erase(firstNL, secondNL - firstNL);
399 size_t firstpos = input.find(firstDelimiter);
400 size_t secondpos = input.find(secondDelimiter);
402 if (firstpos == string::npos || secondpos == string::npos) {
return ""; }
403 return input.substr(firstpos + firstDelimiter.length(), secondpos - firstpos - firstDelimiter.length());
407 vector<string> pvalues;
410 for (
char ch : input) {
411 if (ch != x[0]) { tmp += ch; }
464 if (stat(path.c_str(), &info) != 0) {
467 return (info.st_mode & S_IFDIR) != 0;
471 for (
const auto& trialLocation : possibleLocations) {
472 string possibleDir = trialLocation +
"/" + dirName;
475 return "UNINITIALIZEDSTRINGQUANTITY";
479bool hasExtension(
const std::string& filename,
const std::vector<std::string>& extensions) {
480 for (
const auto& ext : extensions) {
481 if (filename.size() >= ext.size() &&
482 filename.compare(filename.size() - ext.size(), ext.size(), ext) == 0) {
return true; }
488 vector<string> fileList;
490 DIR* dir = opendir(dirName.c_str());
492 struct dirent* entry;
493 while ((entry = readdir(dir)) !=
nullptr) {
495 string filepath = dirName +
"/" + entry->d_name;
496 if (stat(filepath.c_str(), &info) == 0 && S_ISREG(info.st_mode)) {
497 string filename = entry->d_name;
498 if (
hasExtension(filename, extensions)) { fileList.push_back(filename); }
509 transform(lower.begin(), lower.end(), lower.begin(), ::tolower);
514template <
class KEY,
class VALUE>
515vector<KEY>
getKeys(
const map<KEY, VALUE>& map) {
517 keys.reserve(map.size());
519 for (
const auto& it : map) { keys.push_back(it.first); }
525 static const std::unordered_map<std::string, randomModel> strToEnum = {
532 auto it = strToEnum.find(str);
533 if (it != strToEnum.end()) {
return it->second; }
534 else {
throw std::invalid_argument(
"Invalid string for randomModel: " + str); }
539 if (code.empty())
throw std::invalid_argument(
"empty colour string");
540 if (code.front() ==
'#') code.remove_prefix(1);
541 if (code.size() != 6)
542 throw std::invalid_argument(
"colour must have 6 or 7 hex digits");
544 auto hexNibble = [](
char c) ->
unsigned {
545 if (
'0' <= c && c <=
'9')
return c -
'0';
546 c =
static_cast<char>(std::toupper(
static_cast<unsigned char>(c)));
547 if (
'A' <= c && c <=
'F')
return c -
'A' + 10;
548 throw std::invalid_argument(
"invalid hex digit");
553 for (
int i = 0; i < 6; ++i)
554 rgb = (rgb << 4) | hexNibble(code[i]);
556 auto byteToDouble = [](
unsigned byte) {
return byte / 255.0; };
557 double r = byteToDouble((rgb >> 16) & 0xFF);
558 double g = byteToDouble((rgb >> 8) & 0xFF);
559 double b = byteToDouble(rgb & 0xFF);
561 return {r, g, b, opacity};
565 const std::vector<std::string>& locations,
566 std::string_view filename) {
567 namespace fs = std::filesystem;
569 for (
const auto& loc : locations) {
570 if (loc.empty())
continue;
573 fs::path candidate = (!filename.empty() && fs::is_directory(p))
578 const bool ok = fs::exists(candidate, ec) && fs::is_regular_file(candidate, ec);
579 if (ok)
return candidate.string();
586 if (s.empty())
return true;
588 auto eq = [](std::string_view a, std::string_view b) {
589 if (a.size() != b.size())
return false;
590 for (
size_t i = 0; i < a.size(); ++i)
591 if (std::tolower(
static_cast<unsigned char>(a[i])) != std::tolower(
static_cast<unsigned char>(b[i])))
599 G4UImanager* g4uim = G4UImanager::GetUIpointer();
600 if (g4uim ==
nullptr) {
return; }
601 g4uim->ApplyCommand(command);
Public API for the gutilities namespace.
Common constants and console-formatting macros used across gutilities and related code.
#define GWARNING
Standardized warning label prefix (bold yellow).
#define EC__FILENOTFOUND
Process exit code used when an expected file cannot be opened or found.
#define CIRCLEITEM
Hollow bullet glyph used for list formatting in console logs.
#define EC__G4NUMBERERROR
Process exit code used when parsing a Geant4-style numeric string fails.
#define UNINITIALIZEDSTRINGQUANTITY
Sentinel string representing an uninitialized string quantity.
#define FATALERRORL
Standardized fatal error label prefix (bold red).
string replaceAllStringsWithString(const string &source, const string &from, const string &to)
Replaces all occurrences of a substring with another string.
double getG4Number(const string &v, bool warnIfNotUnit)
Converts a string representation of a number with optional units to a double.
bool hasExtension(const std::string &filename, const std::vector< std::string > &extensions)
Checks if a filename has one of the specified extensions.
vector< double > getG4NumbersFromString(const string &vstring, bool warnIfNotUnit)
Converts a comma-separated string of numbers with units to a vector of doubles.
string removeAllSpacesFromString(const std::string &str)
Removes all spaces from a string.
vector< string > getStringVectorFromStringWithDelimiter(const string &input, const string &x)
Splits a string into a vector of substrings using a specified delimiter.
G4Colour makeG4Colour(std::string_view code, double opacity)
Convert a hex colour string to G4Colour.
randomModel stringToRandomModel(const std::string &str)
Converts a string to a corresponding randomModel enum value.
vector< string > getListOfFilesInDirectory(const string &dirName, const vector< string > &extensions)
Retrieves a list of files with specific extensions from a directory.
vector< double > getG4NumbersFromStringVector(const vector< string > &vstring, bool warnIfNotUnit)
Converts a vector of strings representing numbers with units to a vector of doubles.
string retrieveStringBetweenChars(const string &input, const string &firstDelimiter, const string &secondDelimiter)
Retrieves a substring between two specified delimiters in a string.
string replaceCharInStringWithChars(const std::string &input, const std::string &toReplace, const std::string &replacement)
Replaces all occurrences of specified characters in a string with another string.
string fillDigits(const string &word, const string &c, int ndigits)
Pads a string with a specified character until it reaches a desired length.
bool is_unset(std::string_view s)
Determine whether a string should be treated as "unset".
randomModel
Enumeration of random models.
@ gaussian
Gaussian distribution.
@ uniform
Uniform distribution.
@ sphere
Sphere distribution.
@ cosine
Cosine distribution.
string getDirFromPath(const std::string &path)
Extracts the directory path from a given file path.
string convertToLowercase(const string &str)
Converts a string to lowercase.
bool directoryExists(const std::string &path)
Checks if a directory exists at the given path.
vector< KEY > getKeys(const map< KEY, VALUE > &map)
Retrieves all keys from a map.
string removeLeadingAndTrailingSpacesFromString(const std::string &input)
Removes leading and trailing spaces and tabs from a string.
std::optional< std::string > searchForFileInLocations(const std::vector< std::string > &locations, std::string_view filename)
Search for a regular file across candidate locations.
void apply_uimanager_commands(const std::string &command)
Apply a single Geant4 UI command if a UI manager is available.
string searchForDirInLocations(const string &dirName, const vector< string > &possibleLocations)
Searches for a directory within a list of possible locations.
string parseFileAndRemoveComments(const string &filename, const string &commentChars, int verbosity)
Parses a file and removes all lines containing specified comment characters.
string getFileFromPath(const std::string &path)
Extracts the filename from a given file path.
vector< std::string > getStringVectorFromString(const std::string &input)
Splits a string into a vector of strings using whitespace as delimiters.