Source code for TwoD.Objects

"""
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.