2228 {
2229
2232
2234 if (!pScene) {
2236 G4warn <<
"ERROR: No current scene. Please create one." <<
G4endl;
2237 }
2238 return;
2239 } else {
2243 << "ERROR: Scene has no extent. Add volumes or use \"/vis/scene/add/extent\"."
2245 }
2246 return;
2247 }
2248 }
2249
2250 G4double userLength, red, green, blue, xmid, ymid, zmid;
2251 G4String userLengthUnit, direction, placement, positionUnit;
2252 std::istringstream is (newValue);
2253 is >> userLength >> userLengthUnit >> direction
2254 >> red >> green >> blue
2255 >> placement
2256 >> xmid >> ymid >> zmid >> positionUnit;
2257
2259 const G4VisExtent& sceneExtent = pScene->
GetExtent();
2260 if (userLengthUnit == "auto") {
2262 const G4double intLog10Length = std::floor(std::log10(lengthMax));
2263 length = std::pow(10,intLog10Length);
2264 if (5.*length < lengthMax) length *= 5.;
2265 else if (2.*length < lengthMax) length *= 2.;
2266 } else {
2268 }
2269 G4String annotation =
G4BestUnit(length,
"Length");
2270
2272 xmid *= unit; ymid *= unit; zmid *= unit;
2273
2274 Scale::Direction scaleDirection (Scale::x);
2275 if (direction[0] == 'y') scaleDirection = Scale::y;
2276 if (direction[0] == 'z') scaleDirection = Scale::z;
2277
2279 if (!pViewer) {
2282 "ERROR: G4VisCommandSceneAddScale::SetNewValue: no viewer."
2283 "\n Auto direction needs a viewer."
2285 }
2286 return;
2287 }
2288
2293
2294 if (direction == "auto") {
2295 if (std::abs(vp.
x()) > std::abs(vp.
y()) &&
2296 std::abs(vp.
x()) > std::abs(vp.
z())) {
2297 if (std::abs(up.
y()) > std::abs(up.
z())) scaleDirection = Scale::z;
2298 else scaleDirection = Scale::y;
2299 }
2300 else if (std::abs(vp.
y()) > std::abs(vp.
x()) &&
2301 std::abs(vp.
y()) > std::abs(vp.
z())) {
2302 if (std::abs(up.
x()) > std::abs(up.
z())) scaleDirection = Scale::z;
2303 else scaleDirection = Scale::x;
2304 }
2305 else if (std::abs(vp.
z()) > std::abs(vp.
x()) &&
2306 std::abs(vp.
z()) > std::abs(vp.
y())) {
2307 if (std::abs(up.
y()) > std::abs(up.
x())) scaleDirection = Scale::x;
2308 else scaleDirection = Scale::y;
2309 }
2310 }
2311
2312 G4bool autoPlacing =
false;
if (placement ==
"auto") autoPlacing =
true;
2313
2314
2315
2316 const G4double halfLength(length / 2.);
2318 const G4double freeLengthFraction (1. + 2. * comfort);
2319
2326
2327
2330 worried = true;
2333 "WARNING: Existing scene does not yet have any extent."
2334 "\n Maybe you have not yet added any geometrical object."
2336 }
2337 }
2338
2339
2341 switch (scaleDirection) {
2342 case Scale::x:
2343 if (freeLengthFraction * (xmax - xmin) < length) room = false;
2344 break;
2345 case Scale::y:
2346 if (freeLengthFraction * (ymax - ymin) < length) room = false;
2347 break;
2348 case Scale::z:
2349 if (freeLengthFraction * (zmax - zmin) < length) room = false;
2350 break;
2351 }
2352 if (!room) {
2353 worried = true;
2356 "WARNING: Not enough room in existing scene. Maybe scale is too long."
2358 }
2359 }
2360 if (worried) {
2363 "WARNING: The scale you have asked for is bigger than the existing"
2364 "\n scene. Maybe you have added it too soon. It is recommended that"
2365 "\n you add the scale last so that it can be correctly auto-positioned"
2366 "\n so as not to be obscured by any existing object and so that the"
2367 "\n view parameters can be correctly recalculated."
2369 }
2370 }
2371
2372
2373
2374
2375
2376
2377
2378
2379
2380
2381
2382
2383
2384
2385
2386
2387
2388
2389
2390
2391
2392
2393
2394
2395 G4double sxmid(xmid), symid(ymid), szmid(zmid);
2396 if (autoPlacing) {
2397
2398
2399 const G4double xComfort = comfort * (xmax - xmin);
2400 const G4double yComfort = comfort * (ymax - ymin);
2401 const G4double zComfort = comfort * (zmax - zmin);
2402 switch (scaleDirection) {
2403 case Scale::x:
2405 sxmid = xmax + xComfort;
2406 symid = ymin - yComfort;
2407 szmid = zmin - zComfort;
2408 } else {
2409 sxmid = xmin - xComfort;
2410 symid = ymin - yComfort;
2411 szmid = zmax + zComfort;
2412 }
2413 break;
2414 case Scale::y:
2416 sxmid = xmin - xComfort;
2417 symid = ymax + yComfort;
2418 szmid = zmin - zComfort;
2419 } else {
2420 sxmid = xmax + xComfort;
2421 symid = ymin - yComfort;
2422 szmid = zmin - zComfort;
2423 }
2424 break;
2425 case Scale::z:
2427 sxmid = xmax + xComfort;
2428 symid = ymin - yComfort;
2429 szmid = zmax + zComfort;
2430 } else {
2431 sxmid = xmin - xComfort;
2432 symid = ymin - yComfort;
2433 szmid = zmax + zComfort;
2434 }
2435 break;
2436 }
2437 }
2438
2442 G4VisExtent scaleExtent(-h,h,-t,t,-t,t);
2443 switch (scaleDirection) {
2444 case Scale::x:
2445 break;
2446 case Scale::y:
2448 break;
2449 case Scale::z:
2451 break;
2452 }
2454 scaleExtent = scaleExtent.Transform(transform);
2455
2456 G4Colour colour(red, green, blue);
2457 if (direction == "auto") {
2458 switch (scaleDirection) {
2459 case Scale::x:
2461 break;
2462 case Scale::y:
2464 break;
2465 case Scale::z:
2467 break;
2468 }
2469 }
2470 G4VisAttributes visAttr(colour);
2471
2472 Scale* scale = new Scale
2473 (visAttr, length, transform,
2475 G4VModel* model = new G4CallbackModel<Scale>(scale);
2480
2481 const G4String& currentSceneName = pScene -> GetName ();
2482 G4bool successful = pScene -> AddRunDurationModel (model, warn);
2483 if (successful) {
2485 G4cout <<
"Scale of " << annotation
2486 << " added to scene \"" << currentSceneName << "\".";
2488 G4cout <<
"\n with extent " << scaleExtent
2491 }
2493 }
2494 }
2496
2498}
HepGeom::Vector3D< G4double > G4Vector3D
G4GLOB_DLL std::ostream G4cout
const G4VisExtent & GetExtent() const
static G4double ValueOf(const char *unitName)
void SetType(const G4String &)
void SetGlobalDescription(const G4String &)
void SetGlobalTag(const G4String &)
void SetExtent(const G4VisExtent &)
const G4ViewParameters & GetViewParameters() const
static G4double fCurrentTextSize
void G4VisCommandsSceneAddUnsuccessful(G4VisManager::Verbosity verbosity)
void CheckSceneAndNotifyHandlers(G4Scene *=nullptr)
static G4VisManager * fpVisManager
const G4Vector3D & GetViewpointDirection() const
const G4Vector3D & GetUpVector() const
G4double GetExtentRadius() const