"""
This module handles the objects types needed by the TwoD model.
"""
#FIXME : allow to set default segments parameters from the launch script (as in 3D).
import numpy as np
import platrock.Common.Debug as Debug
import copy,sys
import platrock.Common.Math as Math
import platrock.Common.TwoDObjects
from platrock.Common import Outputs, PyUtils
from platrock.Common.Utils import ParametersDescriptorsSet
import platrock.Common.BounceModels as BounceModels
[docs]
class Rock(platrock.Common.TwoDObjects.GenericTwoDRock):
"""
A falling rock.
Args:
x (float): initial position along the x axis. Note that the coordinates system used is the one after the terrain is eventually cleaned, shifted and reversed.
height (float): initial height relative to the terrain
Attributes:
vel (:class:`Common.Math.Vector2` [float,float]): velocity along x,z
angVel (float): angular velocity
volume (float): volume
density (float): density
I (float): inertia, autocomputed if not given
pos (:class:`~platrock.Common.Math.Vector2`): position along x,z, autocomputed
radius (float): radius, autocomputed
mass (float): mass, autocomputed
A (float): an intermediate result of the Azzoni Roll, automatically precomputed
v_square (float): an intermediate value computed into the Azzoni Roll
force_roll (bool): a flag set/used into the roll algorithm to handle roll along two colinear segments
is_stopped (boolean): flag triggered when the stopping condition is reached
current_segment (:class:`Segment`): the current segment that is vertically under the rock
flying_direction (int): -1 if the rock is moving towards -x, +1 if the rock is moving towards +x
color (list [float,float,float]): the rock RGB color, each compound being between 0. and 1., autocomputed
out_of_bounds (bool): set to true during the simulation if the rock went out of the terrain
"""
def __init__(self,*args, **kwargs):
super().__init__(*args, **kwargs)
self.A = self.mass/(self.mass+self.I/self.radius**2) # see Azzoni et al. 1995
self.v_square=0 #needed by the rolling aglorithm
self.force_roll=False
[docs]
def update_flying_direction(self):
"""
Deduce and update the :attr:`flying_direction` from the velocity.
"""
self.flying_direction=int(np.sign(self.vel[0]))
Debug.info("Rock direction is set to",self.flying_direction)
[docs]
def move(self,arrival_point,s,segment):
"""
Actually move the rock by updating its :attr:`pos`. The current simulation and arrival segment must be given as they are usually already computed at this stage.
Note:
This method also handles the rock stop condition, it has in charge to set :attr:`is_stopped`.
Args:
arrival_point (:class:`~platrock.Common.Math.Vector2`): the new position along x,z
s (:class:`~TwoD.Simulations.Simulation`): the current simulation, needed to access its output
segment (:class:`Segment`): the segment that is vertically under the rock after the move
"""
self.pos=arrival_point
self.update_current_segment(segment)
[docs]
def fly(self,arrival_point,s,segment):
"""
Apply a fly movement to the rock, it will update the position and the velocity of the rock.
Args:
arrival_point (:class:`~platrock.Common.Math.Vector2`): the new position along x,z
s (:class:`~TwoD.Simulations.Simulation`): the current simulation, needed to access its output
segment (:class:`Segment`): the segment that is vertically under the rock after the move
"""
#update velocity regarding the arrival point :
self.vel[1]=-s.gravity*(arrival_point[0]-self.pos[0])/self.vel[0] + self.vel[1]
Debug.info("Fly to",arrival_point,",new vel is",self.vel)
self.move(arrival_point,s,segment)
[docs]
def roll(self,s,azzoni_roll,arrival_point):
"""
Apply a roll movement to the rock, it will update the position, velocity and angVel of the rock.
Args:
s (:class:`~TwoD.Simulations.Simulation`): the current simulation, needed to access its output
azzoni_roll (:class:`~platrock.Common.BounceModels.Azzoni_Roll`): the instanciated roll model as it contains necessary attributes and methods
arrival_point (:class:`~platrock.Common.Math.Vector2`): the new position along x,z
"""
Debug.info("Roll to ",arrival_point)
if(azzoni_roll.until_stop):
self.vel*=0
self.angVel*=0
self.move(arrival_point,s,self.current_segment)
Debug.info("ROCK STOPPED")
self.is_stopped=True
else:
self.vel=azzoni_roll.get_vel(arrival_point)
self.angVel = Math.Vector1(- self.flying_direction * self.vel.norm()/self.radius)
if self.flying_direction>0 and arrival_point[0]>=self.current_segment.points[1][0] :
next_seg_id=self.current_segment.index+1
elif self.flying_direction<0 and arrival_point[0]<=self.current_segment.points[0][0] :
next_seg_id=self.current_segment.index-1
else: next_seg_id=self.current_segment.index
if next_seg_id<0 or next_seg_id>len(s.terrain.segments)-1 :
self.out_of_bounds=True
self.is_stopped=True
next_segment = self.current_segment
else:
next_segment = s.terrain.segments[next_seg_id]
self.move(arrival_point,s,next_segment)
[docs]
def bounce(self,s,segment,disable_roughness=False):
"""
Apply a bounce from :py:mod:`platrock.Common.BounceModels` to the rock, it will update the velocity and angVel of the rock.
Args:
s (:class:`~TwoD.Simulations.Simulation`): the current simulation, needed to access its output
segment (:class:`Segment`): the segment that is vertically under the rock
disable_roughness (bool): use this to tell the bounce model not to apply the terrain roughness
"""
if(s.override_rebound_params):
bounce_model_number=s.override_bounce_model_number
else:
bounce_model_number=segment.bounce_model_number
bounce_model=s.number_to_bounce_model_instance[bounce_model_number]
bounce_model.run(self,segment,disable_roughness)
return bounce_model.updated_normal
[docs]
class Segment(platrock.Common.TwoDObjects.GenericSegment):
"""
A segment of the terrain. Attributes from all bounce models / rolls will be considered as valid, they are stored in :attr:`valid_input_attrs`, with values : :pyDocPrint:`valid_input_attrs`
Args:
start_point (:class:`~platrock.Common.Math.Vector2`): start point coordinates of the segment (:math:`[x_{start}, z_{start}]`)
end_point (:class:`~platrock.Common.Math.Vector2`): end point coordinates of the segment (:math:`[x_{end}, z_{end}]`)
Attributes:
points (np.ndarray): start and end points in the form :math:`[ [x_{start}, z_{start}], [x_{end}, z_{end}] ]`
index (int): index of the segment (they are continuously and automatically indexed through the terrain)
branch (:class:`~platrock.Common.Math.Vector2`): the vector connecting :attr:`start_point` to :attr:`end_point`
normal (:class:`~platrock.Common.Math.Vector2`): the segment normal vector
slope_gradient (float): the gradient of slope (:math:`\\Delta_z / \\Delta_x`)
slope (float): the slope in radians, CCW
"""
valid_input_attrs=ParametersDescriptorsSet([])
"""Describes the available soil attributes, they are a concatenation of all bounce models parameters."""
for bounce_class in BounceModels.number_to_model_correspondance.values():
valid_input_attrs+=bounce_class.valid_input_attrs
valid_input_attrs+=BounceModels.Toe_Tree_2022.valid_input_attrs
del bounce_class #avoid temp variable to show up in the doc
PyUtils.pyDocPrint(Segment)
[docs]
class Terrain(platrock.Common.TwoDObjects.GenericTwoDTerrain):
"""
A 2D terrain made of segments.
Args:
file (string):
Attributes:
segments (list): the successive :class:`Segment` forming the terrain
rebound_models_available (list): A list of available bounce models numbers regarding the per-segment input parameters given, automatically filled.
forest_available (bool): whether the forest is available in the terrain regarding the per-segment input parameters given, automatically set.
"""
valid_input_attrs=Segment.valid_input_attrs
def __init__(self, *args, **kwargs):
self.rebound_models_available=[]#A list of available rebound models regarding the per-segment input parameters given. Modified by check_segments_parameters_consistency() method.
super().__init__(*args, **kwargs)
[docs]
def check_segments_parameters_consistency(self):
"""
Analyze the segments parameters and checks their consistency/availability. :attr:`forest_available` and :attr:`rebound_models_available` are here.
"""
super().check_segments_forest_parameters_consistency()
s=self.segments[0] #all the segments has the same data, use the first one to list them below
HAS_bounce_model_number=s.bounce_model_number is not None
HAS_roughness=s.roughness is not None
HAS_R_t=s.R_t is not None
HAS_R_n=s.R_n is not None
HAS_v_half=s.v_half is not None
HAS_phi=s.phi is not None
if(HAS_bounce_model_number):
if(HAS_roughness and HAS_R_t and HAS_R_n):
self.rebound_models_available+=[0,1]
if(HAS_v_half and HAS_phi):
self.rebound_models_available+=[2]
bounce_model_numbers=[s.bounce_model_number for s in self.segments]
USE_classical = 0 in bounce_model_numbers
USE_pfeiffer = 1 in bounce_model_numbers
USE_bourrier = 2 in bounce_model_numbers
if( USE_classical and (0 not in self.rebound_models_available)):
raise ValueError('At least one segment "rebound_model" parameter has been set to 0(=Classical) but the corresponding parameters were not specified (roughness, R_n, R_t)')
if( USE_pfeiffer and (1 not in self.rebound_models_available)):
raise ValueError('At least one segment "rebound_model" parameter has been set to 1(=Pfeiffer) but the corresponding parameters were not specified (roughness, R_n, R_t)')
if( USE_bourrier and (2 not in self.rebound_models_available)):
raise ValueError('At least one segment "rebound_model" parameter has been set to 2(=Bourrier) but the corresponding parameters were not specified (roughness, R_t, v_half, phi)')
[docs]
class Checkpoint(platrock.Common.TwoDObjects.GenericTwoDCheckpoint):
pass #nothing to change to the parent class, just declare here for consistency and facilitate eventual future implementations.