Notebook

これは日々の作業を通して学んだことや毎日の生活で気づいたことをを記録しておく備忘録である。

HTML ファイル生成日時: 2025/01/14 09:02:48.430 (台灣標準時)

小惑星二萬個の軌道運動

太陽、水星、金星、地球、火星、木星と小惑星二萬個の軌道運動をアニメーショ ンにしてみたでござる。

アニメーションを作るために使った Python スクリプトは以下の通りにござる。


#!/usr/pkg/bin/python3.9

#
# Time-stamp: <2022/10/12 21:47:21 (CST) daisuke>
#

# importing sys module
import sys

# importing numpy module
import numpy

# importing astropy module
import astropy
import astropy.coordinates
import astropy.time
import astropy.units

# importing astroquery module
import astroquery.jplhorizons

# importing matplotlib module
import matplotlib.animation
import matplotlib.backends.backend_agg
import matplotlib.figure

# output file name prefix
file_prefix = 'solsys_3d_struct'

# output file name extension
file_ext    = 'png'

# units
u_au = astropy.units.au
u_hr = astropy.units.hour

# number of steps to calculate
n_steps = 10000

# number of asteroids to plot
#n_asteroids = 30000
n_asteroids = 20000
#n_asteroids = 16000
#n_asteroids = 10

# step size in hr
step_hr  = 12
step_str = f'{step_hr}h'
step     = step_hr * u_hr

# an empty list for storing asteroids positions
list_asteroids = []

# date/time to start the simulation
t_start_str = f'2020-01-01T00:00:00.000'
    
# time to start the simulation in astropy.time object
t_start = astropy.time.Time (t_start_str, format='isot', scale='utc')

# time to stop the simulation in astropy.time object
t_stop  = t_start + step * n_steps

# an empty list for storing major planets positions
list_major = []

# major body names (Sun, Mercury, Venus, Earth, Mars, Jupiter)
list_names = ['10', '199', '299', '399', '499', '599']

# getting positions of the Sun, Mercury, Venus, Earth, Mars, and Jupiter
# from JPL/Horizons
print (f'Now, getting positions of the Sun and planets...')
for i in list_names:
    print (i)
    query = astroquery.jplhorizons.Horizons (id_type=None, id=f'{i}', \
                                             location='@0', \
                                             epochs={'start': t_start.iso, \
                                                     'stop': t_stop.iso, \
                                                     'step': step_str})
    vec = query.vectors ()
    print (vec)
    x = vec['x']
    y = vec['y']
    z = vec['z']
    list_major.append ( [x, y, z] )
print (f'Finished getting positions of the Sun and planets!')

# getting asteroids positions from JPL/Horizons
print (f'Now, getting asteroids positions...')
for i in range (1, n_asteroids + 1):
    #if (i % 10 == 0):
    #print (f'  now, getting positions of asteroid ({i})...')
    print (f'  now, getting positions of asteroid ({i})...')
    ast_query = astroquery.jplhorizons.Horizons (id_type='smallbody', \
                                                 id=f'{i}', \
                                                 location='@0', \
                                                 epochs={'start': t_start.iso, \
                                                         'stop': t_stop.iso, \
                                                         'step': step_str})
    ast_vec = ast_query.vectors ()
    x = ast_vec['x']
    y = ast_vec['y']
    z = ast_vec['z']
    list_asteroids.append ( [x, y, z] )
print (f'Finished getting asteroids positions...')

# making a fig object using object-oriented interface
fig = matplotlib.figure.Figure (figsize=[15.36, 8.64])
fig.subplots_adjust (left=0.0, right=1.0, bottom=0.0, top=1.0, \
                     wspace=0.0, hspace=0.0)

# making a canvas object
canvas = matplotlib.backends.backend_agg.FigureCanvasAgg (fig)

# making an axes object
#ax = fig.add_subplot (111, projection='3d')
ax = fig.add_axes ( (0, 0, 1, 1), projection='3d')

# an empty list of frames for animation
list_frame = []

# definition of a function for making a sphere
def make_sphere (x_c, y_c, z_c, radius, colour):
    u = numpy.linspace (0, 2 * numpy.pi, 1000)
    v = numpy.linspace (0, numpy.pi, 1000)
    x = radius * numpy.outer (numpy.cos(u), numpy.sin(v)) + x_c
    y = radius * numpy.outer (numpy.sin(u), numpy.sin(v)) + y_c
    z = radius * numpy.outer (numpy.ones(numpy.size(u)), numpy.cos(v)) + z_c
    # plotting the surface
    sphere = ax.plot_surface (x, y, z, color=colour, antialiased=False, \
                               shade=True, rcount=100, ccount=100)
    return (sphere)

# initial value of 'elev' angle
el0 = 90.0

# initial value of 'azim' angle
az0 = -90.0

# initial value of 'dist'
dist0 = 10.0

for i in range (n_steps):
    # clearing previous axes
    ax.cla ()
    
    # time t
    t = t_start + i * 12.0 * u_hr

    # printing positions of the Sun, planets, and asteroids
    if (i % 10 == 0):
        print (f'Now, making a plot for {t}...')

    # settings for plot
    ax.set_xlim3d (-7.5, +7.5)
    ax.set_ylim3d (-7.5, +7.5)
    ax.set_zlim3d (-2.0, +2.0)
    ax.set_box_aspect ( (7.5, 7.5, 2.0) )

    # projection
    ax.set_proj_type ('persp')

    # using black background colour
    #fig.set_facecolor ('black')
    fig.set_facecolor ('white')
    ax.set_facecolor ('black')
    ax.grid (False)
    ax.w_xaxis.set_pane_color ((0.0, 0.0, 0.0, 0.0))
    ax.w_yaxis.set_pane_color ((0.0, 0.0, 0.0, 0.0))
    ax.w_zaxis.set_pane_color ((0.0, 0.0, 0.0, 0.0))

    # camera viewing angle
    if (i < 300):
        el = el0
        az = az0
        dist = dist0
    elif ( (i >= 300) and (i < 1200) ):
        el = el0 - (i - 300) * 0.1
        az = az0
        dist = dist0
    elif ( (i >= 1200) and (i < 1500) ):
        el = 0.0
        az = az0
        dist = dist0
    elif ( (i >= 1500) and (i < 2400) ):
        el = 0.0
        az = az0 - (i - 1500) * 0.1
        dist = dist0
    elif ( (i >= 2400) and (i < 2700) ):
        el = 0.0
        az = -180.0
        dist = dist0
    elif ( (i >= 2700) and (i < 3000) ):
        el = (i - 2700) * 0.1
        az = -180.0
        dist = dist0
    elif ( (i >= 3000) and (i < 3300) ):
        el = 30.0
        az = -180.0
        dist = dist0
    elif ( (i >= 3300) and (i < 3600) ):
        el = 30.0
        az = -180.0
        dist = dist0 - (i - 3300) * 0.01
    elif ( (i >= 3600) and (i < 3900) ):
        el = 30.0 + (i - 3600) * 0.1
        az = -180.0
        dist = dist0 - 3.0
    elif ( (i >= 3900) and (i < 4200) ):
        el = 60.0
        az = -180.0
        dist = dist0 - 3.0
    elif ( (i >= 4200) and (i < 4500) ):
        el = 60.0
        az = -180.0
        dist = dist0 - 3.0 - (i - 4200) * 0.01
    elif ( (i >= 4500) and (i < 5400) ):
        el = 60.0
        az = -180.0 - (i - 4500) * 0.2
        dist = dist0 - 6.0
    elif ( (i >= 5400) and (i < 6000) ):
        el = 60.0
        az = 0.0
        dist = dist0 - 6.0
    elif ( (i >= 6000) and (i < 6150) ):
        el = 60.0 - (i - 6000) * 0.1
        az = 0.0
        dist = dist0 - 6.0
    elif ( (i >= 6150) and (i < 6350) ):
        el = 45.0
        az = 0.0
        dist = dist0 - 6.0 + (i - 6150) * 0.01
    elif ( (i >= 6350) and (i < 6650) ):
        el = 45.0
        az = 0.0
        dist = dist0 - 4.0
    elif ( (i >= 6650) and (i < 6950) ):
        el = 45.0
        az = 0.0 - (i - 6650) * 0.3
        dist = dist0 - 4.0
    elif ( (i >= 6950) and (i < 7250) ):
        el = 45.0
        az = -90.0
        dist = dist0 - 4.0
    elif ( (i >= 7250) and (i < 7550) ):
        el = 45.0 - (i - 7250) * 0.1
        az = -90.0
        dist = dist0 - 4.0
    elif ( (i >= 7550) and (i < 7850) ):
        el = 15.0
        az = -90.0
        dist = dist0 - 4.0
    elif ( (i >= 7850) and (i < 8150) ):
        el = 15.0
        az = -90.0
        dist = dist0 - 4.0 + (i - 7850) * 0.01
    elif ( (i >= 8150) and (i < 8450) ):
        el = 15.0
        az = -90.0
        dist = dist0 - 1.0
    elif ( (i >= 8450) and (i < 9050) ):
        el = 15.0
        az = -90.0
        dist = dist0 - 1.0 - (i - 8450) * 0.01
    elif ( (i >= 9050) and (i < 9350) ):
        el = 15.0
        az = -90.0
        dist = dist0 - 7.0
    elif ( (i >= 9350) and (i < 9650) ):
        el = 15.0
        az = -90.0
        dist = dist0 - 7.0 + (i - 9350) * 0.01
    else:
        el = 15.0
        az = -90.0
        dist = dist0 - 4.0
    
    # plotting the Sun
    sun = make_sphere (list_major[0][0][i], \
                       list_major[0][1][i], \
                       list_major[0][2][i], \
                       0.25, 'orange')

    # plotting Mercury
    mercury = make_sphere (list_major[1][0][i], \
                           list_major[1][1][i], \
                           list_major[1][2][i], \
                           0.05, 'cyan')
    
    # plotting Venus
    venus = make_sphere (list_major[2][0][i], \
                         list_major[2][1][i], \
                         list_major[2][2][i], \
                         0.15, 'gold')

    # plotting Earth
    earth = make_sphere (list_major[3][0][i], \
                         list_major[3][1][i], \
                         list_major[3][2][i], \
                         0.15, 'blue')

    # plotting Mars
    mars = make_sphere (list_major[4][0][i], \
                        list_major[4][1][i], \
                        list_major[4][2][i], \
                        0.15, 'red')

    # plotting Jupiter
    jupiter = make_sphere (list_major[5][0][i], \
                           list_major[5][1][i], \
                           list_major[5][2][i], \
                           0.15, 'bisque')

    # plotting asteroids
    for j in range (0, n_asteroids):
        ax.scatter (list_asteroids[j][0][i], \
                    list_asteroids[j][1][i], \
                    list_asteroids[j][2][i], \
                    s=1.0, \
                    color='saddlebrown', \
                    alpha=0.5)
    
    # title
    title = ax.text2D (0.5, 0.75, f'Inner Solar System', \
                       color='white', \
                       horizontalalignment='center', \
                       transform=ax.transAxes)

    # plotting the time
    time = ax.text2D (0.20, 0.23, f'Date/Time: {t} (UTC)', \
                      color='white', \
                      horizontalalignment='center', \
                      transform=ax.transAxes)

    # plotting the author name
    author = ax.text2D (0.85, 0.23, f'animation generated by Daisuke', \
                        color='white', \
                        horizontalalignment='center', \
                        transform=ax.transAxes)

    # viewing angles of camera
    ax.view_init (elev=el, azim=az)
    ax.dist = dist

    # image file
    file_image = f'{file_prefix}_{i:06d}.{file_ext}'
    fig.savefig (file_image, dpi=225)

一萬枚の画像から動画を作るためには、以下のコマンドを実行したでござる。


% ffmpeg5 -f image2 -start_number 0 -framerate 30 -i solsys_3d_struct_%06dc.png -an -vcodec libx264 -pix_fmt yuv420p -threads 12 solsys_3d_struct_no_audio.mp4

音楽をのせるには以下のようにしたでござる。


% ffmpeg5 -i solsys_3d_struct_no_audio.mp4 -i audio/fragments.f251.webm -c:v copy solsys_3d_struct_with_audio.mp4



Frequently accessed files

  1. Computer___Python/20220518_0.html
  2. Computer___Network/20230726_00.html
  3. Misc___Taiwan/20240207_00.html
  4. Computer___Network/20230516_00.html
  5. Computer___FreeBSD/20220621_0.html
  6. Computer___Python/20220715_0.html
  7. Computer___Network/20230508_00.html
  8. Food___Taiwan/20220429_0.html
  9. Computer___Network/20240130_00.html
  10. Computer___Network/20240416_00.html
  11. Computer___Python/20220410_0.html
  12. Computer___NetBSD/20220817_3.html
  13. Computer___NetBSD/20230119_00.html
  14. Computer___Python/20221013_0.html
  15. Computer___Debian/20210223_1.html
  16. Computer___Python/20210124_0.html
  17. Computer___NetBSD/20220428_0.html
  18. Computer___NetBSD/20220818_1.html
  19. Computer___TeX/20231107_00.html
  20. Computer___NetBSD/20240101_02.html
  21. Computer___Python/20240101_00.html
  22. Science___Math/20220420_0.html
  23. Computer___Network/20220413_1.html
  24. Computer___TeX/20230503_00.html
  25. Computer___NetBSD/20220808_0.html
  26. Science___Astronomy/20220503_0.html
  27. Computer___NetBSD/20230515_00.html
  28. Computer___NetBSD/20210127_0.html
  29. Misc___Japan/20240610_00.html
  30. Computer___Python/20230717_01.html


HTML file generated by Kinoshita Daisuke.