From 0525bc463f53487a7de8409697fabec28471a0bd Mon Sep 17 00:00:00 2001
From: emile <hanemile@protonmail.com>
Date: Mon, 15 Oct 2018 16:07:22 +0200
Subject: for detailed information about the changes, see
 https://git.darknebu.la/GalaxySimulator/Source/pulls/1

---
 Source              | Bin 0 -> 3302760 bytes
 csv/csv.go          |  56 +++++--------------------
 draw/draw.go        |  69 ++++++++++++++++---------------
 file/file.go        |  31 ++++++++++++++
 forces/forces.go    | 117 ++++++++++++++++++++++++----------------------------
 main.go             |  31 +++++++-------
 out_0.png           | Bin 0 -> 3367531 bytes
 structs/star.go     |  29 +++++++++++++
 structs/structs.go  |  23 -----------
 structs/vector2D.go |  63 ++++++++++++++++++++++++++++
 10 files changed, 239 insertions(+), 180 deletions(-)
 create mode 100755 Source
 create mode 100644 file/file.go
 create mode 100644 out_0.png
 create mode 100644 structs/star.go
 delete mode 100644 structs/structs.go
 create mode 100644 structs/vector2D.go

diff --git a/Source b/Source
new file mode 100755
index 0000000..b35d91d
Binary files /dev/null and b/Source differ
diff --git a/csv/csv.go b/csv/csv.go
index 7a0fa9f..aca131b 100644
--- a/csv/csv.go
+++ b/csv/csv.go
@@ -1,75 +1,39 @@
 package csv
 
 import (
+	"../file"
 	"../structs"
-	"encoding/csv"
-	"log"
-	"math/rand"
-	"os"
+	"git.darknebu.la/bit/logplus"
 	"strconv"
-	"time"
 )
 
-// openStar2DCSV opens the file at the given path and reads its content. It then returns the content inform of a slice
-// of slices
-func openStar2DCSV(path string) [][]string {
-
-	// Open the file located at the given path
-	b, err := os.Open(path)
-
-	// Handle errors
-	if err != nil {
-		log.Printf("openStar2DsCSV Panic! (cannot read file from %s)", path)
-	}
-
-	// Close the file afre reading it's content
-	defer b.Close()
-
-	// Parse the files conten usin a csv-reader
-	lines, err := csv.NewReader(b).ReadAll()
-
-	// Handle errors
-	if err != nil {
-		log.Println("openStar2DsCSV Panic! (cannot read the files content)")
-	}
-
-	return lines
-}
-
 // Import gets a file, a starting line, an ending line and  a struct. It then adds the content of the file to the struct
 // For finding the length of the .csv, you can use the following command in linux:
 // $ cat <csv> | wc -l
 func Import(path string, start int, end int, slice []structs.Star2D) []structs.Star2D {
-	lines := openStar2DCSV(path)
-
-	// seed the random number generator
-	rand.Seed(time.Now().UTC().UnixNano())
+	f, _ := file.Open(path)
+	lines, _ := f.ReadCSV()
 
-	// iterate over all the lines in the given range
 	for linenr, line := range lines[start:end] {
 		x, errx := strconv.ParseFloat(line[0], 64)
 		y, erry := strconv.ParseFloat(line[1], 64)
 
 		// Handle errors
 		if errx != nil {
-			log.Printf("error reading value from csv in line nr. %d (%s)", linenr, errx)
+			logplus.LogFError("error reading value from csv in line nr. %d (%s)", linenr, errx)
 		}
 		if erry != nil {
-			log.Printf("error reading value from csv in line nr. %d (%s)", linenr, erry)
+			logplus.LogFError("error reading value from csv in line nr. %d (%s)", linenr, erry)
 		}
 
-		// TODO: Export the code below to its own function
-		var mass float64 = float64(10000 + rand.Intn(100000-10000))
-
 		// Create a temporary star for assembling the star
-		tempStar2D := structs.Star2D{
-			structs.Coord{X: x, Y: y},
-			structs.Force{X: 0, Y: 0},
-			mass,
+		tempStar := structs.Star2D{
+			C: structs.Vec2{X: x, Y: y},
+			M: 50000,
 		}
 
 		// Add the Temporary star to the slice
-		slice = append(slice, tempStar2D)
+		slice = append(slice, tempStar)
 
 	}
 
diff --git a/draw/draw.go b/draw/draw.go
index 1823493..a5d5f0d 100644
--- a/draw/draw.go
+++ b/draw/draw.go
@@ -33,41 +33,32 @@ func saveImage(dc *gg.Context, path string) {
 	dc.SavePNG(path)
 }
 
-// drawStar2D draws the given stars to the given context
-func drawStar2D(dc *gg.Context, star structs.Star2D) {
-	// the radius can be any value inbetween 1e4 and 1e5
-
-	// Define the default star radius as 1 and change it according to the stars mass
-	radius := 1
-	switch {
-	case star.Mass < 10000:
-		radius = 1
-	case star.Mass < 50000 && star.Mass > 10000:
-		radius = 2
-	case star.Mass < 100000 && star.Mass > 50000:
-		radius = 3
+// drawStar draws the given stars to the given context
+func drawStar(dc *gg.Context, star structs.Star2D) {
+
+	// set radius of the star to 2
+	R := 2.0
+
+	// if the star has a greater mass than the default 5E4, it's radius is increased to 5
+	if star.M > 5E4 {
+		R = 5
 	}
 
-	// Draw the star
-	dc.DrawPoint(star.C.X/50, star.C.Y/50, float64(radius))
+	// draw the star / point
+	dc.DrawPoint(star.C.X/50, star.C.Y/50, R)
 	dc.Fill()
 	dc.Stroke()
 }
 
-// vectorLength calculates the length of the given vector
-func vectorLength(force structs.Force) float64 {
-	return math.Sqrt(math.Pow(force.X, 2) + math.Pow(force.Y, 2))
-}
-
-func drawForce(dc *gg.Context, star structs.Star2D) {
-	// controll the length of the vector
+func drawVelocity(dc *gg.Context, star structs.Star2D) {
+	// scaling factor for a better view of the velocity difference
 	var scalingFactor float64 = 15
 
 	// Move the "cursor" to the start position of the vector
 	dc.MoveTo(star.C.X/50, star.C.Y/50)
 
 	// calculate the length of the vector
-	vecLength := vectorLength(star.F)
+	vecLength := star.V.GetLength()
 
 	// Use a sigmoid function to generate useful values for coloring the vectors according to their
 	// strength
@@ -76,31 +67,31 @@ func drawForce(dc *gg.Context, star structs.Star2D) {
 	// Set the color to a blue / red
 	dc.SetRGB(val, 0, 1-val)
 
-	// trace the Vector
-	FxUnit := star.F.X / math.Abs(vecLength)
-	FyUnit := star.F.Y / math.Abs(vecLength)
+	// calculate the direction vector
+	FUnit := (&star.V).Divide(vecLength)
 
-	dc.LineTo(star.C.X/50+(FxUnit*scalingFactor), star.C.Y/50+(FyUnit*scalingFactor))
+	// set end-position of the vector line
+	dc.LineTo(star.C.X/50+(FUnit.X*scalingFactor), star.C.Y/50+(FUnit.Y*scalingFactor))
 
-	// css
+	// set line width
 	dc.SetLineWidth(3)
 
 	// And finally: DRAW (stroke) the vector
 	dc.Stroke()
 }
 
-// drawStar2Ds draws all the stars in the given slice to the given context
-func drawStars2D(dc *gg.Context, slice []structs.Star2D) {
-	// draw all the forces in the given slice
+// drawStars draws all the stars in the given slice to the given context
+func drawStars(dc *gg.Context, slice []structs.Star2D) {
+	// draw all the velocity in the given slice
 	for _, star := range slice {
-		drawForce(dc, star)
+		drawVelocity(dc, star)
 	}
 
 	dc.SetRGB(1, 1, 1)
 
 	// draw all the stars in the given slice
 	for _, star := range slice {
-		drawStar2D(dc, star)
+		drawStar(dc, star)
 	}
 }
 
@@ -111,7 +102,17 @@ func Slice(slice []structs.Star2D, path string) {
 	dc := initializePlot()
 
 	// draw all the stars in the given slice
-	drawStars2D(dc, slice)
+	drawStars(dc, slice)
+
+	dc.SetRGB(1, 1, 1)
+
+	// drawing the 4 big stars as bigger white dots
+	//dc.DrawCircle(600, 600, 5)
+	//dc.DrawCircle(-600, 600, 5)
+	//dc.DrawCircle(-600, 0, 5)
+	//dc.DrawCircle(600, -600, 5)
+
+	dc.Fill()
 
 	// save the plot to the given path
 	saveImage(dc, path)
diff --git a/file/file.go b/file/file.go
new file mode 100644
index 0000000..ce6ac9c
--- /dev/null
+++ b/file/file.go
@@ -0,0 +1,31 @@
+package file
+
+import (
+	"encoding/csv"
+	"git.darknebu.la/bit/logplus"
+	"os"
+)
+
+type File struct {
+	f *os.File
+}
+
+func Open(path string) (File, error) {
+	file, err := os.Open(path)
+	if err != nil {
+		logplus.LogFError("openStarsCSV Panic! (cannot read file from %s)", path)
+	}
+	return File{f: file}, err
+}
+
+func (file *File) ReadCSV() ([][]string, error) {
+	lines, err := csv.NewReader(file.f).ReadAll()
+	if err != nil {
+		logplus.LogError("openStarsCSV Panic! (cannot read the files content)")
+	}
+	return lines, err
+}
+
+func (file *File) Close() error {
+	return file.f.Close()
+}
diff --git a/forces/forces.go b/forces/forces.go
index 7763f87..a994334 100644
--- a/forces/forces.go
+++ b/forces/forces.go
@@ -1,87 +1,81 @@
 package forces
 
 import (
-	"../llog"
 	"../structs"
 	"fmt"
+	"git.darknebu.la/bit/logplus"
 	"gopkg.in/cheggaaa/pb.v1"
 	"math"
 )
 
 // forces_acting calculates the force inbetween the two given stars s1 and s2
 // The function return the force
-func forceActing(s1 structs.Star2D, s2 structs.Star2D) structs.Force {
+func accelerationActing(s1 structs.Star2D, s2 structs.Star2D) structs.Vec2 {
+
 	// Gravitational constant
-	var G float64 = 6.674e-11
+	var G = 6.674E-11
 
-	// Distance between the stars
-	var r21 = math.Sqrt(math.Pow(s2.C.X-s1.C.X, 2) + math.Pow(s2.C.Y-s1.C.Y, 2))
+	// the vector from star s1 to star s2
+	var r12 = s2.C.Subtract(s1.C)
 
-	// Unit vector pointing from s1 to s2
-	rhat := structs.Force{s2.C.X - s1.C.X, s2.C.Y - s1.C.Y}
+	// the distance between the stars
+	var deltaR = r12.GetLength()
 
-	// Calculate how strong the star is affected
-	var FScalar = G * (s1.Mass * s2.Mass) / math.Pow(math.Abs(r21), 2)
+	// the scalar acceleration of star s1 by star s2
+	scalarA := G * (s2.M) / math.Pow(deltaR, 2)
 
-	// Calculate the overall force by combining the scalar and the vector
-	var Fx = FScalar * rhat.X
-	var Fy = FScalar * rhat.Y
+	// calculate the direction vector of the vector from s1 to s2
+	directionVectorA := r12.GetDirVector()
 
-	// Pack the forces in a force structur
-	F := structs.Force{Fx, Fy}
+	// the vector acceleration of scalarA
+	A := directionVectorA.Multiply(scalarA)
 
-	return F
+	return A
 }
 
-// forces calculates the forces acting in between a given star and all the other stars in a given array.
-func forces(stars_arr []structs.Star2D, nr int) structs.Force {
+// accelerations calculates the acceleration acting in between a given star and all the other stars in a given array.
+func accelerations(stars_arr []structs.Star2D, nr int) structs.Vec2 {
 
-	var force structs.Force
+	var a = /*acceleration*/ structs.Vec2{}
 	// Iterate over all the stars in the stars_arr
 	for index := range stars_arr {
 
 		// If the current star is not the star itself
 		if index != nr {
 
-			// generate a new force and add it to the overall force of the star
-			fa := forceActing(stars_arr[nr], stars_arr[index])
-			stars_arr[nr].F.X += fa.X
-			stars_arr[nr].F.Y += fa.Y
+			// calculate the acceleration and add it to the overall acceleration of the star
+			aa := accelerationActing(stars_arr[nr], stars_arr[index])
+			a = a.Add(aa)
 
-			force.X += fa.X
-			force.Y += fa.Y
 		}
 	}
 
-	return force
+	return a
 }
 
-// forcesThread calculates the forces acting on a given amount of stars in a given range for a given slice of stars
+// accelerationThread calculates the acceleration acting on a given amount of stars in a given range for a given slice of stars
 // as a go-routine
-func forcesThread(starSlice []structs.Star2D, localRangeStar2Dt int, localRangeEnd int, channel chan structs.Star2D) {
+func accelerationThread(starSlice []structs.Star2D, localRangeStart int, localRangeEnd int, channel chan structs.Star2D) {
 
 	// iterate over the given range
-	for index := localRangeStar2Dt; index < localRangeEnd; index++ {
+	for index := localRangeStart; index < localRangeEnd; index++ {
 
-		// Calculate the force acting inbetween the given star and all other stars
-		var force = forces(starSlice, index)
+		// Calculate the acceleration acting inbetween the given star and all other stars
+		var a = accelerations(starSlice, index)
 
 		// create a new star
-		newStar2D := structs.Star2D{
-			structs.Coord{starSlice[index].C.X, starSlice[index].C.Y},
-			structs.Force{force.X, force.Y},
-			starSlice[index].Mass,
-		}
+		newStar := starSlice[index].Copy()
+		newStar.AccelerateVelocity(a, 1)
 
-		// push the new Star2D into the channel
-		channel <- newStar2D
+		// push the new Star into the channel
+		channel <- newStar
 	}
 }
 
-// CalcAllForces calculates all the forces acting inbetween all the stars in the given starSlice slice and
-// returns a "new" slice contaning the forces
-func CalcAllForces(starSlice []structs.Star2D, threads int) []structs.Star2D {
-	// create a channel for bundling the stars generaten in the go-routines
+// CalcAllAccelerations calculates all the accelerations acting in between all the stars in the given starSlice slice and
+// returns a "new" slice containing the stars with their new velocities
+func CalcAllAccelerations(starSlice []structs.Star2D, threads int) []structs.Star2D {
+	// create a channel for bundling the stars generated in the go-routines
 	channel := make(chan structs.Star2D, 1000)
 
 	sliceLength := len(starSlice)
@@ -93,43 +87,47 @@ func CalcAllForces(starSlice []structs.Star2D, threads int) []structs.Star2D {
 	// generate a new slice for storing the stars
 	var newSlice []structs.Star2D
 
-	llog.Good(fmt.Sprintf("Starting %d workers, each processing %d stars", threads, localRangeLen))
+	logplus.LogNeutral(fmt.Sprintf("Starting %d workers, each processing %d stars", threads, localRangeLen))
 
 	// start n go threads
 	for i := 0; i < threads; i++ {
 
 		// define the local range
-		localRangeStar2Dt := i * localRangeLen
+		localRangeStart := i * localRangeLen
 		localRangeEnd := (i * localRangeLen) + localRangeLen
 
-		// calculate the forces for all the stars in the given slice in the given range and return them using the
+		// fmt.Printf("starting worker nr. %d, processing %d stars\n", i, localRangeEnd-localRangeStart)
+
+		// calculate the accelerations for all the stars in the given slice in the given range and return them using the
 		// given channel
-		go forcesThread(starSlice, localRangeStar2Dt, localRangeEnd, channel)
+		go accelerationThread(starSlice, localRangeStart, localRangeEnd, channel)
 	}
 
 	// Handle errors (10004 stars, but 1250 stars per thread, so 4 stars are not calculate and block the queue)
 	if sliceLength > localRangeLen {
 
 		// Calculate the amount of stars and their range
-		remainingStar2Ds := sliceLength - (localRangeLen * threads)
+		remainingStars := sliceLength - (localRangeLen * threads)
 		localRangeEnd := ((threads - 1) * localRangeLen) + localRangeLen
 
 		// Run the Thread
-		// go forcesThread(starSlice, localRangeEnd, localRangeEnd+remainingStar2Ds, channel)
-		forcesThread(starSlice, localRangeEnd, localRangeEnd+remainingStar2Ds, channel)
+		// go accelerationThread(starSlice, localRangeEnd, localRangeEnd+remainingStars, channel)
+		accelerationThread(starSlice, localRangeEnd, localRangeEnd+remainingStars, channel)
 	}
 
 	// Initialize a new progress bar
-	bar := pb.New(len(starSlice)).Prefix("Star2Ds: ")
+	bar := pb.New(len(starSlice)).Prefix("Stars: ")
+
+	bar.Start()
 
 	// iterate over the amount of stars
 	for i := 0; i < sliceLength; i++ {
 
 		// block until a star is finisehd
-		var newStar2D structs.Star2D = <-channel
+		var newStar = <-channel
 
 		// append the star from the channel to the newSlice for returning in the end
-		newSlice = append(newSlice, newStar2D)
+		newSlice = append(newSlice, newStar)
 
 		// increment the progress bar and the counter
 		bar.Increment()
@@ -141,26 +139,19 @@ func CalcAllForces(starSlice []structs.Star2D, threads int) []structs.Star2D {
 }
 
 // Calculate the new positions of the stars using the
-func NextTimestep(starSlice []structs.Star2D, deltat int) []structs.Star2D {
+func NextTimestep(starSlice []structs.Star2D, deltat float64) []structs.Star2D {
 	// create a new slice for storing the "new" stars
 	var newSlice []structs.Star2D
 
 	// iterate over all the stars in the old slice
 	for index := range starSlice {
 
-		// calculate the new position
-		newX := starSlice[index].C.X + starSlice[index].F.X*float64(deltat)
-		newY := starSlice[index].C.Y + starSlice[index].F.Y*float64(deltat)
-
-		// assemble the new star
-		newStar2D := structs.Star2D{
-			C:    structs.Coord{X: newX, Y: newY},
-			F:    structs.Force{},
-			Mass: starSlice[index].Mass,
-		}
+		// move the star with it's velocity for time deltat
+		newStar := starSlice[index].Copy()
+		newStar.Move(deltat)
 
 		// append the new star to the newSlice
-		newSlice = append(newSlice, newStar2D)
+		newSlice = append(newSlice, newStar)
 	}
 
 	return newSlice
diff --git a/main.go b/main.go
index 0ad2e69..7aa389f 100644
--- a/main.go
+++ b/main.go
@@ -7,37 +7,40 @@ import (
 	"./structs"
 	"fmt"
 	"git.darknebu.la/bit/logplus"
+	"math"
 )
 
 func main() {
+	logplus.SetLogLevel(logplus.LevelAll)
 	var threads int = 8
+	var path1 string = "out_0.png"
 	var frames int = 1
-	var stars int = 800
 
 	// the slice starsSlice stores the star structures
-	starsSlice := []structs.Star2D{}
+	starsSlice := []structs.Star2D{
+		//		{C: structs.Vec2{X:  30000,  Y: 30000},M: 5E8},
+		//		{C: structs.Vec2{X: -30000,  Y: 30000},M: 5E8},
+		//		{C: structs.Vec2{X: -30000           },M: 5E8},
+		//		{C: structs.Vec2{X:  30000, Y: -30000},M: 5E8},
+		{C: structs.Vec2{}, M: 5E8},
+	}
 
-	logplus.LogPositive("Opening the csv")
-	starsSlice = csv.Import("data/U_ALL.csv", 0, stars, starsSlice)
+	logplus.LogNeutral("Opening the csv")
+	starsSlice = csv.Import("data/U_ALL.csv", 0, 25000, starsSlice)
 
 	// Simulate the position of the stars after a specific time
 	for i := 0; i < frames; i++ {
-
-		// Print the iterator
 		logplus.LogPositive("--- --- --- --- ---")
 		logplus.LogPositive(fmt.Sprintf("Frames %d/%d", i, frames))
 
-		// Calculate the position of stars after one timestep
-		logplus.LogPositive("Calculate the new Star positions")
-		starsSlice = forces.NextTimestep(starsSlice, 500000)
+		logplus.LogNeutral("Calculate the new Star positions")
+		starsSlice = forces.NextTimestep(starsSlice, 25*math.Pow(10, 4+7))
 
-		// Calculate all the forces acting inside of the galaxy
-		logplus.LogPositive("Calculate the acting forces")
-		starsSlice = forces.CalcAllForces(starsSlice, threads)
+		logplus.LogNeutral("Calculate the acting accelerations")
+		starsSlice = forces.CalcAllAccelerations(starsSlice, threads)
 
-		// Save the slice
 		outputName := fmt.Sprintf("out_%d.png", i+1)
-		logplus.LogPositive(fmt.Sprintf("draw the slice and save it to %s\n", outputName))
+		logplus.LogNeutral(fmt.Sprintf("draw the slice and save it to %s\n", outputName))
 		draw.Slice(starsSlice, outputName)
 	}
 }
diff --git a/out_0.png b/out_0.png
new file mode 100644
index 0000000..830315d
Binary files /dev/null and b/out_0.png differ
diff --git a/structs/star.go b/structs/star.go
new file mode 100644
index 0000000..72afc00
--- /dev/null
+++ b/structs/star.go
@@ -0,0 +1,29 @@
+package structs
+
+import "C"
+
+type Star2D struct {
+	C Vec2    // coordinates of the star
+	V Vec2    // velocity    of the star
+	M float64 // mass        of the star
+}
+
+func (s *Star2D) Copy() Star2D {
+	return Star2D{s.C.Copy(), s.V.Copy(), s.M}
+}
+
+// accelerate the star with the acceleration a for the time t
+func (s *Star2D) AccelerateVelocity(a Vec2, t float64) {
+	s.V = s.V.Add(a.Multiply(t))
+}
+
+// move the star with it's velocity for the time t
+func (s *Star2D) Move(t float64) {
+	s.C = s.C.Add(s.V.Multiply(t))
+}
+
+// accelerate and move the star with it's velocity and the acceleration a for the time t
+func (s *Star2D) Accelerate(a Vec2, t float64) {
+	s.AccelerateVelocity(a, t)
+	s.Move(t)
+}
diff --git a/structs/structs.go b/structs/structs.go
deleted file mode 100644
index ed70a84..0000000
--- a/structs/structs.go
+++ /dev/null
@@ -1,23 +0,0 @@
-package structs
-
-/*
-	The structs package defines structs that are used to store information that is used for processing
-	star related data.
-*/
-
-// Force struct soring a force vector
-type Force struct {
-	X, Y float64
-}
-
-// Coord struct storing coordinates
-type Coord struct {
-	X, Y float64
-}
-
-// Star struct storing all necessary star information
-type Star2D struct {
-	C    Coord
-	F    Force
-	Mass float64
-}
diff --git a/structs/vector2D.go b/structs/vector2D.go
new file mode 100644
index 0000000..47e457f
--- /dev/null
+++ b/structs/vector2D.go
@@ -0,0 +1,63 @@
+package structs
+
+import (
+	"math"
+)
+
+type Vec2 struct {
+	X, Y float64
+}
+
+// creates a copy of the vector
+func (v *Vec2) Copy() Vec2 {
+	return Vec2{v.X, v.Y}
+}
+
+func (v *Vec2) Split() (x float64, y float64) {
+	return v.X, v.Y
+}
+
+// changes the length of the vector to the length l
+func (v *Vec2) SetLength(l float64) {
+	var k = l / v.GetLength()
+	var newV = v.Multiply(k)
+	//	v = newV
+	v.X, v.Y = newV.Split()
+}
+
+// changes the length of the vector to the length 1
+func (v *Vec2) SetLengthOne() {
+	v.SetLength(1)
+}
+
+// returns the direction Vector of this vector. This means a copy of this vector with a length of 1
+func (v *Vec2) GetDirVector() Vec2 {
+	var dirV = v.Copy()
+	dirV.SetLengthOne()
+	return dirV
+}
+
+// returns the length of the vector
+func (v *Vec2) GetLength() float64 {
+	return math.Sqrt(math.Pow(v.X, 2) + math.Pow(v.Y, 2))
+}
+
+// returns the product of the vector and a scalar s
+func (v *Vec2) Multiply(s float64) Vec2 {
+	return Vec2{v.X * s, v.Y * s}
+}
+
+// returns the quotient of the vector and a scalar s
+func (v *Vec2) Divide(s float64) Vec2 {
+	return Vec2{v.X / s, v.Y / s}
+}
+
+// returns the sum of this vector and the vector v2
+func (v1 *Vec2) Add(v2 Vec2) Vec2 {
+	return Vec2{v1.X + v2.X, v1.Y + v2.Y}
+}
+
+// returns the difference of this vector minus the vector v2
+func (v1 *Vec2) Subtract(v2 Vec2) Vec2 {
+	return Vec2{v1.X - v2.X, v1.Y - v2.Y}
+}
-- 
cgit 1.4.1