#!/usr/bin/env python3

# Filename: A09_vec2d.py

# This Vec2D class is based on the Vec2d class found here:
# http://pygame.org/wiki/2DVectorClass

import math

class Vec2D:
    # Components of the vector can be input as individual arguments or as
    # a tuple pair in one argument.
    def __init__(self, x_or_pair, y = None, int_flag = "not_int"):
        if y == None:
            self.x = x_or_pair[0]
            self.y = x_or_pair[1]
        else:
            self.x = x_or_pair
            self.y = y

        if int_flag == "int":
            self.x = round(self.x)
            self.y = round(self.y)
        else:
            self.x = float(self.x)
            self.y = float(self.y)
        
    # This establishes the format of the output string if you print a vector (an object instance).
    def __str__(self):
        return 'Vec2D(%s, %s)' % (self.x, self.y)

    #============================================================================
    
    # This section defines methods for vector arithmetic and comparison. 
    # Operator overloading is established here.
    
    # Addition
    def add_vector(self, vec_B):
        return Vec2D(self.x + vec_B.x, self.y + vec_B.y)
    def __add__(self, vec_B):
        return Vec2D(self.x + vec_B.x, self.y + vec_B.y)
    
    # Subtraction
    def sub_vector(self, vec_B):
        return Vec2D(self.x - vec_B.x, self.y - vec_B.y)
    def __sub__(self, vec_B):
        return Vec2D(self.x - vec_B.x, self.y - vec_B.y)
    
    # Scaling operations. Note that the operator overloading cases
    # must have the vector preceding the scaling factor!
    # e.g.  vector * 1.2   or   vector / 3.2,  but not    1.2 * vector
    def scale_vector(self, scale_factor):
        return Vec2D( self.x * scale_factor, self.y * scale_factor)
    def __mul__(self, scale_factor):
        return Vec2D( self.x * scale_factor, self.y * scale_factor)
    def __truediv__(self, scale_factor):
        return Vec2D( self.x / scale_factor, self.y / scale_factor)
    
    # Comparisons.
    def equal(self, vec_B):
        return (self.x == vec_B.x) and (self.y == vec_B.y)
    def not_equal(self, vec_B):
        return (self.x != vec_B.x) or (self.y != vec_B.y)
        
    #============================================================================
  
    # Determine the scaler length of the vector. This uses the
    # squareroot operation.
    def length(self):
        return (self.x*self.x + self.y*self.y)**0.5
    
    # Length squared can sometimes be used in substitution for the scaler length
    # of a vector. Length squared is much faster to calculate since there is
    # no squareroot operation.
    def length_squared(self):
        return (self.x*self.x + self.y*self.y)
    
    # Returns a vector in the same direction as the
    # original but with a length of 1 (unit length).
    def normal(self):
        #return self.scale_vector( 1/ self.length())
        return self / self.length()
    
    # Returns a vector in the same direction as the
    # original but with a length equal to the target magnitude.
    def set_magnitude(self, target_magnitude):
        return self.normal() * target_magnitude
    
    # Dot product with another vector.
    def dot(self, vec_B):
        return (self.x * vec_B.x) + (self.y * vec_B.y)
    
    # The vector component of the self vector along the direction of the B vector.
    # (Refer to the drawings in the PDF.)
    def projection_onto(self, vec_B):
        vB_dot_vB = vec_B.dot(vec_B)
        if (vB_dot_vB > 0):
            return vec_B * ( self.dot(vec_B) / vB_dot_vB )        
        else:
            # If vec_B has zero magnitude, return a zero magnitude vector. In this 
            # case, the dot product of the two vectors will be zero. This leaves the 
            # projection undetermined; vec_B * (0/0). It would be appropriate to 
            # return a None value here! But the zero magnitude vector is handy for 
            # dealing with the spring-anchored pucks (because the separation distance 
            # between the puck and the pinning point eventually goes to zero as the 
            # disturbed puck loses energy). So again, it would be better to return a 
            # None here and catch this in the code following the projection. 
            return self * 0
    
    # Rotate 90 degrees counterclockwise.
    def rotate90(self):
        return Vec2D(-self.y, self.x)
    
    # Flip it around (reverse the direction of the vector).
    def rotate180(self):
        return Vec2D(-self.x, -self.y)
    
    # Rotate (change the direction of) the original vector by a specified number of degrees.
    # (Original vector rotated by angle_degrees.)
    def rotated(self, angle_degrees, sameVector=False):
        angle_radians = math.radians(angle_degrees)
        cos = math.cos(angle_radians)
        sin = math.sin(angle_radians)
        # The rotation transformation.
        x = self.x * cos - self.y * sin
        y = self.x * sin + self.y * cos
        if sameVector:
            # Modify the original vector.
            self.x = x
            self.y = y
        else:
            # A new vector object
            return Vec2D(x, y)
        
    # Set the direction of the vector to a specific angle.
    def set_angle(self, angle_degrees):
        self.x = self.length()
        self.y = 0
        return self.rotated(angle_degrees)
    
    # Determine the angle that this vector makes with the x axis. Measure
    # counterclockwise from the x axis.
    def get_angle(self):
        if (self.length_squared() == 0):
            return 0
        return math.degrees(math.atan2(self.y, self.x))
    
    # Determine the angle between two vectors.
    def get_angle_between(self, vec_B):
        cross = self.x*vec_B.y - self.y*vec_B.x    #= ABsin(theta)
        dot   = self.x*vec_B.x + self.y*vec_B.y    #= ABcos(theta)       
        # Use the two parameter input (y,x) form of the arctan. This is safer than
        # taking the arctan of the cross/dot which can be zero in the denominator.
        return math.degrees(math.atan2(cross, dot))  
        
    # Return a tuple containing the two components of the vector.
    def tuple(self):
        return (self.x, self.y)