package draw import ( "../structs" "github.com/fogleman/gg" "math" ) // initializePlot generates a new plot and returns the plot context func initializePlot() *gg.Context { // Define the image size const imageWidth = 8192 const imageHeight = 8192 // Initialize the new context dc := gg.NewContext(imageWidth, imageHeight) // Set the background black dc.SetRGB(0, 0, 0) dc.Clear() // Invert the Y axis (positive values are on the top and right) dc.InvertY() // Set the coordinate midpoint to the middle of the image dc.Translate(imageWidth/2, imageHeight/2) return dc } // saveImages saves the given context to a png at the given path func saveImage(dc *gg.Context, path string) { dc.SavePNG(path) } // drawStar draws the given stars to the given context func drawStar(dc *gg.Context, star structs.Star) { // 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 } // Draw the star dc.DrawPoint(star.C.X/50, star.C.Y/50, float64(radius)) 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.Star) { // controll the length of the vector 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) // Use a sigmoid function to generate useful values for coloring the vectors according to their // strength var val = 1.0 / (1.0 + math.Exp(-vecLength*scalingFactor/2)) // 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) dc.LineTo(star.C.X/50+(FxUnit*scalingFactor), star.C.Y/50+(FyUnit*scalingFactor)) // dc.LineTo(star.C.X/100 + (star.F.X * scalingFactor), star.C.Y/100 + (star.F.Y * scalingFactor)) // // css dc.SetLineWidth(3) // And finally: DRAW (stroke) the vector dc.Stroke() } // drawStars draws all the stars in the given slice to the given context func drawStars(dc *gg.Context, slice []structs.Star) { // draw all the forces in the given slice for _, star := range slice { drawForce(dc, star) } dc.SetRGB(1, 1, 1) // draw all the stars in the given slice for _, star := range slice { drawStar(dc, star) } } // Slice draws the stars and the forces acting on them and saves the result to the given path func Slice(slice []structs.Star, path string) { // initialize the plot dc := initializePlot() // draw all the stars in the given 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) }