-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch '3d' into resurrect-circular-cushion-collisions
- Loading branch information
Showing
28 changed files
with
2,022 additions
and
1,797 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Large diffs are not rendered by default.
Oops, something went wrong.
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,59 +1,101 @@ | ||
"""Resolve events""" | ||
|
||
import inspect | ||
|
||
import attrs | ||
|
||
from pooltool.physics.resolve.ball_ball import ( | ||
BallBallCollisionStrategy, | ||
BallBallModel, | ||
get_ball_ball_model, | ||
ball_ball_models, | ||
) | ||
from pooltool.physics.resolve.ball_cushion import ( | ||
BallCCushionCollisionStrategy, | ||
BallCCushionModel, | ||
BallLCushionCollisionStrategy, | ||
BallLCushionModel, | ||
get_ball_circ_cushion_model, | ||
get_ball_lin_cushion_model, | ||
ball_ccushion_models, | ||
ball_lcushion_models, | ||
) | ||
from pooltool.physics.resolve.ball_pocket import ( | ||
BallPocketModel, | ||
BallPocketStrategy, | ||
get_ball_pocket_model, | ||
ball_pocket_models, | ||
) | ||
from pooltool.physics.resolve.resolver import ( | ||
RESOLVER_CONFIG_PATH, | ||
RESOLVER_PATH, | ||
Resolver, | ||
ResolverConfig, | ||
) | ||
from pooltool.physics.resolve.stick_ball import ( | ||
StickBallCollisionStrategy, | ||
StickBallModel, | ||
get_stick_ball_model, | ||
stick_ball_models, | ||
) | ||
from pooltool.physics.resolve.transition import ( | ||
BallTransitionModel, | ||
BallTransitionStrategy, | ||
get_transition_model, | ||
ball_transition_models, | ||
) | ||
|
||
|
||
def _display_model(cls, model): | ||
fp = inspect.getfile(cls) | ||
print(f" {model.value} ({fp})") | ||
|
||
if not attrs.has(cls): | ||
raise TypeError(f"{cls.__name__} is not an attrs class.") | ||
|
||
indent = 4 | ||
indent_str = " " * indent | ||
|
||
for field in attrs.fields(cls): | ||
if field.name == "model": | ||
continue | ||
|
||
default_val = field.default | ||
if default_val is attrs.NOTHING: | ||
default_val = None | ||
|
||
print( | ||
f"{indent_str} - {field.name}: " | ||
f"type={field.type}, default={default_val}" | ||
) | ||
|
||
|
||
def display_models(): | ||
print("\nball_ball models:") | ||
for model in BallBallModel: | ||
_display_model(ball_ball_models[model], model) | ||
print("\nball_linear_cushion models:") | ||
for model in BallLCushionModel: | ||
_display_model(ball_lcushion_models[model], model) | ||
print("\nball_circular_cushion models:") | ||
for model in BallCCushionModel: | ||
_display_model(ball_ccushion_models[model], model) | ||
print("\nstick_ball models:") | ||
for model in StickBallModel: | ||
_display_model(stick_ball_models[model], model) | ||
print("\nball_pocket models:") | ||
for model in BallPocketModel: | ||
_display_model(ball_pocket_models[model], model) | ||
print("\nball_transition models:") | ||
for model in BallTransitionModel: | ||
_display_model(ball_transition_models[model], model) | ||
|
||
|
||
__all__ = [ | ||
"Resolver", | ||
"RESOLVER_CONFIG_PATH", | ||
"ResolverConfig", | ||
"RESOLVER_PATH", | ||
"BallBallCollisionStrategy", | ||
"BallBallModel", | ||
"get_ball_ball_model", | ||
"BallCCushionCollisionStrategy", | ||
"BallCCushionModel", | ||
"BallLCushionCollisionStrategy", | ||
"BallLCushionModel", | ||
"get_ball_circ_cushion_model", | ||
"get_ball_lin_cushion_model", | ||
"BallPocketModel", | ||
"BallPocketStrategy", | ||
"get_ball_pocket_model", | ||
"StickBallCollisionStrategy", | ||
"StickBallModel", | ||
"get_stick_ball_model", | ||
"BallTransitionModel", | ||
"BallTransitionStrategy", | ||
"get_transition_model", | ||
] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,63 +1,31 @@ | ||
"""Models for ball-ball collisions.""" | ||
|
||
from typing import Dict, Optional, Type | ||
from typing import Dict, Tuple, Type, cast | ||
|
||
import attrs | ||
|
||
from pooltool.physics.resolve.ball_ball.core import BallBallCollisionStrategy | ||
from pooltool.physics.resolve.ball_ball.frictional_inelastic import FrictionalInelastic | ||
from pooltool.physics.resolve.ball_ball.frictional_mathavan import FrictionalMathavan | ||
from pooltool.physics.resolve.ball_ball.frictionless_elastic import FrictionlessElastic | ||
from pooltool.physics.resolve.types import ModelArgs | ||
from pooltool.utils.strenum import StrEnum, auto | ||
|
||
|
||
class BallBallModel(StrEnum): | ||
"""An Enum for different ball-ball collision models | ||
from pooltool.physics.resolve.models import BallBallModel | ||
|
||
Attributes: | ||
FRICTIONLESS_ELASTIC: | ||
Frictionless, instantaneous, elastic, equal mass collision | ||
(:class:`FrictionlessElastic`). | ||
""" | ||
_ball_ball_model_registry: Tuple[Type[BallBallCollisionStrategy], ...] = ( | ||
FrictionlessElastic, | ||
FrictionalMathavan, | ||
FrictionalInelastic, | ||
) | ||
|
||
FRICTIONLESS_ELASTIC = auto() | ||
FRICTIONAL_INELASTIC = auto() | ||
FRICTIONAL_MATHAVAN = auto() | ||
|
||
|
||
_ball_ball_models: Dict[BallBallModel, Type[BallBallCollisionStrategy]] = { | ||
BallBallModel.FRICTIONLESS_ELASTIC: FrictionlessElastic, | ||
BallBallModel.FRICTIONAL_INELASTIC: FrictionalInelastic, | ||
BallBallModel.FRICTIONAL_MATHAVAN: FrictionalMathavan, | ||
ball_ball_models: Dict[BallBallModel, Type[BallBallCollisionStrategy]] = { | ||
cast(BallBallModel, attrs.fields_dict(cls)["model"].default): cls | ||
for cls in _ball_ball_model_registry | ||
} | ||
|
||
|
||
def get_ball_ball_model( | ||
model: Optional[BallBallModel] = None, params: ModelArgs = {} | ||
) -> BallBallCollisionStrategy: | ||
"""Returns a ball-ball collision model | ||
Args: | ||
model: | ||
An Enum specifying the desired model. If not passed, | ||
:class:`FrictionalMathavan` is passed with empty params. | ||
params: | ||
A mapping of parameters accepted by the model. | ||
Returns: | ||
An instantiated model that satisfies the :class:`BallBallCollisionStrategy` | ||
protocol. | ||
""" | ||
|
||
if model is None: | ||
return FrictionlessElastic() | ||
|
||
return _ball_ball_models[model](**params) | ||
|
||
|
||
__all__ = [ | ||
"BallBallModel", | ||
"get_ball_ball_model", | ||
"FrictionalMathavan", | ||
"FrictionalInelastic", | ||
"FrictionlessElastic", | ||
"ball_ball_models", | ||
] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,67 @@ | ||
import math | ||
from typing import Dict, Protocol, Type | ||
|
||
import attrs | ||
import numpy as np | ||
|
||
import pooltool.ptmath as ptmath | ||
from pooltool.objects.ball.datatypes import Ball | ||
from pooltool.physics.utils import surface_velocity | ||
from pooltool.utils.strenum import StrEnum, auto | ||
|
||
|
||
class BallBallFrictionModel(StrEnum): | ||
"""An Enum for different ball-ball friction models""" | ||
|
||
AVERAGE = auto() | ||
ALCIATORE = auto() | ||
|
||
|
||
class BallBallFrictionStrategy(Protocol): | ||
"""Ball-ball friction models must satisfy this protocol""" | ||
|
||
def calculate_friction(self, ball1: Ball, ball2: Ball) -> float: | ||
"""This method calculates ball-ball friction""" | ||
... | ||
|
||
|
||
@attrs.define | ||
class AlciatoreBallBallFriction: | ||
"""Friction fit curve u_b = a + b * exp(-c * v_rel) used in David Alciatore's TP A-14""" | ||
|
||
a: float = 9.951e-3 | ||
b: float = 0.108 | ||
c: float = 1.088 | ||
|
||
model: BallBallFrictionModel = attrs.field( | ||
default=BallBallFrictionModel.ALCIATORE, init=False, repr=False | ||
) | ||
|
||
def calculate_friction(self, ball1: Ball, ball2: Ball) -> float: | ||
unit_x = np.array([1.0, 0.0, 0.0]) | ||
v1_c = surface_velocity(ball1.state.rvw, unit_x, ball1.params.R) - np.array( | ||
[ball1.state.rvw[1][0], 0, 0] | ||
) | ||
v2_c = surface_velocity(ball2.state.rvw, -unit_x, ball2.params.R) - np.array( | ||
[ball2.state.rvw[1][0], 0, 0] | ||
) | ||
relative_surface_speed = ptmath.norm3d(v1_c - v2_c) | ||
return self.a + self.b * math.exp(-self.c * relative_surface_speed) | ||
|
||
|
||
@attrs.define | ||
class AverageBallBallFriction: | ||
model: BallBallFrictionModel = attrs.field( | ||
default=BallBallFrictionModel.AVERAGE, init=False, repr=False | ||
) | ||
|
||
def calculate_friction(self, ball1: Ball, ball2: Ball) -> float: | ||
return (ball1.params.u_b + ball2.params.u_b) / 2 | ||
|
||
|
||
ball_ball_friction_models: Dict[ | ||
BallBallFrictionModel, Type[BallBallFrictionStrategy] | ||
] = { | ||
BallBallFrictionModel.AVERAGE: AverageBallBallFriction, | ||
BallBallFrictionModel.ALCIATORE: AlciatoreBallBallFriction, | ||
} |
Oops, something went wrong.