見出し画像

力の釣り合いと合力の理解を深めるシミュレーションゲーム開発


構造力学は、物体や構造物に作用する力を分析し、バランスや安定性を評価する学問である。

建築や機械工学など、多くの分野で重要な役割を果たしているが、その基本的な概念である「力の釣り合い」や「合力」は、数式や静的な図面だけでは直感的に理解しにくいことがある。

こうした課題を克服するために、インタラクティブな要素を取り入れたシミュレーションゲームが有効となるので開発した。

具体的には、ユーザーが力の大きさや方向を操作しながら、力の合力がゼロになる状態、つまり「釣り合いの状態」を探すゲームを設計することで、構造力学の原理を体感的に学ぶことができる。

このゲームの主旨は、力の釣り合いを視覚的かつインタラクティブに体験させることで、物理学や力学の理解を促す点にある。

力の釣り合いとは、複数の力が作用しているときに、物体が静止している状態、すなわち力の合計がゼロになる状態を意味する。このシミュレーションでは、画面上に2つの力(ベクトル)を表示し、ユーザーがその大きさ角度を操作して力の合力を調整する。

さらに、合力(残りの力)がゼロに近づくほど、画面上に緑色の矢印として表示される合力が短くなる。この視覚的フィードバックは、ユーザーが力のつり合いに向かって試行錯誤する過程をサポートし、単に理論を学ぶだけでなく、感覚的に「力の釣り合い」の状態を理解できるように設計されている。

ユーザーは、キーボード入力を通じてそれぞれの力のパラメータ(大きさや角度)を調整することができる。この操作は、力の成分分解の概念を裏側で支えている。例えば、1つの力を 、角度を としたとき、その力は水平成分 と鉛直成分 に分解される。これを2つの力について合計し、最終的に得られる「合力」の大きさと方向を計算する。

ユーザーが入力を行うたびにこれらの計算が画面上で即座に反映され、ベクトルがリアルタイムに動くことで、学習者は「力がどのように釣り合うのか」を動的に観察し、理解を深めることができる。

また、ゲームの最終的な目標は、合力がしきい値(例えば0.5kN)未満になるように調整することである。

この条件を達成すると、「釣り合い達成!クリア!」というメッセージが表示される。こうしたフィードバックを通じて、学習者は力の釣り合いの達成感を得られると同時に、繰り返し挑戦することでベクトル操作や成分分解の理解が自然と身につく仕組みになっている。

このゲームは、力の釣り合いという抽象的な概念を視覚化し、シンプルな操作で直感的に学べることが最大の特徴である。数式や静止図だけでは得られない「動き」や「変化」を通じて、力がどのように作用し、合力がどのように減少するのかを観察できる。

学習者自身が試行錯誤しながら理解を深めるプロセスは、アクティブラーニングの一環としても非常に有効である。

結論として、このシミュレーションゲームは、構造力学の基礎概念である「力のつり合い」や「合力の計算」を楽しく体験しながら学ぶための優れた教育ツールとなる。遊びながら自然に理論を学ぶことができるため、初心者や学生にとって、力学の理解を助ける強力なサポートとなるだろう。

python ソースコード

import pygame
import math
import sys

# Pygame初期設定
pygame.init()

# 画面設定
WIDTH, HEIGHT = 800, 600
screen = pygame.display.set_mode((WIDTH, HEIGHT))
pygame.display.set_caption("力のつり合いゲーム")

# 色の定義
WHITE = (255, 255, 255)
BLACK = (0, 0, 0)
RED = (255, 0, 0)
BLUE = (0, 0, 255)
GREEN = (0, 255, 0)

# スケールと原点
SCALE = 50  # 力のスケーリング係数
ORIGIN = (WIDTH // 2, HEIGHT // 2)

# フォント設定
FONT = pygame.font.Font(None, 36)

def draw_vector(screen, color, start, magnitude, angle_deg, label=None):
    """ベクトルを描画する関数"""
    angle_rad = math.radians(angle_deg)
    end_x = start[0] + magnitude * SCALE * math.cos(angle_rad)
    end_y = start[1] - magnitude * SCALE * math.sin(angle_rad)  # Y座標は反転
    pygame.draw.line(screen, color, start, (end_x, end_y), 5)
    pygame.draw.circle(screen, color, (int(end_x), int(end_y)), 5)
    if label:
        text = FONT.render(label, True, BLACK)
        screen.blit(text, (end_x + 10, end_y - 20))

def calculate_resultant(force1, force2):
    """2つの力の合力を計算"""
    fx = force1["magnitude"] * math.cos(math.radians(force1["angle"])) + \
         force2["magnitude"] * math.cos(math.radians(force2["angle"]))
    fy = force1["magnitude"] * math.sin(math.radians(force1["angle"])) + \
         force2["magnitude"] * math.sin(math.radians(force2["angle"]))
    magnitude = math.sqrt(fx**2 + fy**2)
    angle = math.degrees(math.atan2(fy, fx))
    return {"magnitude": magnitude, "angle": angle}

def main():
    clock = pygame.time.Clock()

    # 力の初期設定
    force1 = {"magnitude": 4, "angle": 90, "color": RED, "label": "F1"}
    force2 = {"magnitude": 4, "angle": 45, "color": BLUE, "label": "F2"}

    running = True
    while running:
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                running = False

        # キーボード入力で力を調整
        keys = pygame.key.get_pressed()
        if keys[pygame.K_UP]:
            force1["magnitude"] += 0.1
        if keys[pygame.K_DOWN]:
            force1["magnitude"] -= 0.1
        if keys[pygame.K_LEFT]:
            force1["angle"] += 1
        if keys[pygame.K_RIGHT]:
            force1["angle"] -= 1

        if keys[pygame.K_w]:
            force2["magnitude"] += 0.1
        if keys[pygame.K_s]:
            force2["magnitude"] -= 0.1
        if keys[pygame.K_a]:
            force2["angle"] += 1
        if keys[pygame.K_d]:
            force2["angle"] -= 1

        # 合力の計算
        resultant = calculate_resultant(force1, force2)

        # 画面の描画
        screen.fill(WHITE)
        pygame.draw.line(screen, BLACK, (0, ORIGIN[1]), (WIDTH, ORIGIN[1]), 2)  # X軸
        pygame.draw.line(screen, BLACK, (ORIGIN[0], 0), (ORIGIN[0], HEIGHT), 2)  # Y軸

        # 力の描画
        draw_vector(screen, force1["color"], ORIGIN, force1["magnitude"], force1["angle"], force1["label"])
        draw_vector(screen, force2["color"], ORIGIN, force2["magnitude"], force2["angle"], force2["label"])

        # 合力の描画(緑色)
        draw_vector(screen, GREEN, ORIGIN, resultant["magnitude"], resultant["angle"], "Resultant")

        # 合力の大きさを表示
        text = FONT.render(f"Resultant Force: {resultant['magnitude']:.2f} kN", True, BLACK)
        screen.blit(text, (20, 20))

        # 勝利条件(合力が0.5未満)
        if resultant["magnitude"] < 0.5:
            win_text = FONT.render("釣り合い達成!クリア!", True, GREEN)
            screen.blit(win_text, (WIDTH // 2 - 100, HEIGHT // 2))

        pygame.display.flip()
        clock.tick(60)

    pygame.quit()
    sys.exit()

if __name__ == "__main__":
    main()

ゲームの特徴

1. 力の調整
• 矢印キー で 力1 の大きさと角度を調整。
• WASDキー で 力2 の大きさと角度を調整。
2. 合力の表示
• 合力が画面上に 緑色の矢印 として表示されます。
• 合力の大きさは画面上部にリアルタイムで表示されます。
3. 勝利条件
• 合力の大きさが 0.5未満 になると「クリア!」のメッセージが表示されます。

 操作方法:
• 矢印キー: 力1 の調整(大きさ・角度)。
• W, A, S, D キー: 力2 の調整(大きさ・角度)。
目標:
• 2つの力を調整して 合力がゼロ になるようにしましょう。





いいなと思ったら応援しよう!