aboutsummaryrefslogtreecommitdiff
path: root/Занимательное программирование/1/2_ideal_gas/unit1.pas
blob: 44489cab0b53fd858c3b16ed9da57aecb02dc397 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
unit Unit1;

{$mode objfpc}{$H+}

interface

uses
  Classes, SysUtils, Forms, Controls, Graphics, Dialogs, ExtCtrls, StdCtrls,
  Math;

type

  Molecule = record
    X, Y: Integer;
    Vx, Vy: Integer;
  end;

  { TForm1 }

  TForm1 = class(TForm)
    StartStopBtn: TButton;
    Screen: TPaintBox;
    procedure StartStopBtnClick(Sender: TObject);
    procedure ScreenPaint(Sender: TObject);
  private
    const R: Integer = 10; { Радиус молекулы }
      V: Integer = 7; { Максимальная скорость молекулы }
      N = 30; { Количество молекул }
    var Mol: array[1..N] of Molecule; { Массив молекул }
  public

  end;

var
  Form1: TForm1;
  IsRunning: Boolean = false;

implementation

{$R *.lfm}

{ TForm1 }

procedure TForm1.ScreenPaint(Sender: TObject);
  var i: Integer;
begin
  if not IsRunning then Exit;

  for i := 1 to N do { Цикл по всем молекулам }
  begin
    Screen.Canvas.Pen.Color := clBtnFace; { Стираем молекулу }
    Screen.Canvas.Ellipse(Mol[i].X - R, Mol[i].Y - R,
      Mol[i].X + R, Mol[i].Y + R);

    Mol[i].X := Mol[i].X + Mol[i].Vx; { Сдвигаем на новую позицию }
    Mol[i].Y := Mol[i].Y + Mol[i]. Vy;

    { Определяем, не вышла ли молекула за границы аквариума }
    if Mol[i].X > Screen.Width - R then
    begin
      Mol[i].X := Screen.Width - R;
      Mol[i].Vx := -Mol[i].Vx;
    end;
    if Mol[i].X < R then
    begin
      Mol[i].X := R;
      Mol[i].Vx := -Mol[i].Vx;
    end;
    if Mol[i].Y > Screen.Height - R then
    begin
      Mol[i].Y := Screen.Height - R;
      Mol[i].Vy := -Mol[i].Vy;
    end;
    if Mol[i].Y < R then
    begin
      Mol[i].Y := R;
      Mol[i].Vy := -Mol[i].Vy;
    end;

    Screen.Canvas.Pen.Color := clBlue; { Рисуем молекулу на новой }
    Screen.Canvas.Ellipse(Mol[i].X - R, Mol[i].Y - R, { позиции }
      Mol[i].X + R, Mol[i].Y + R);
  end;
end;

procedure TForm1.StartStopBtnClick(Sender: TObject);
  var angle: Real; { Угол, задающий изначальное направление полета }
    i: Integer; { Счетчик цикла }
    CurV: Integer; { Выбранная случайная скорость молекулы }
begin
  if IsRunning then
  begin
    IsRunning := false;
    StartStopBtn.Caption := 'Пуск';
    Exit;
  end;

  StartStopBtn.Caption := 'Стоп';
  IsRunning := true;

  Randomize;

  for i := 1 to N do { Цикл по всем молекулам }
  begin
    Mol[i].X := RandomRange(R, Screen.Width - R); { Выбор начального }
    Mol[i].Y := RandomRange(R, Screen.Height - R); { положения молекулы }
    angle := Random(360) * Pi / 180; { и ее направления }

    CurV := RandomRange(1, V); { Выбор скорости молекулы (1 - V) }
    Mol[i].Vx := Round(CurV * Sin(angle)); { Получение составляющих }
    Mol[i].Vy := Round(CurV * Cos(angle)); { скорости молекулы }
  end;

  while IsRunning do { Основной цикл }
  begin
    Screen.Invalidate;
    Sleep(10); { Пауза на 10 миллисекунд }
    Application.ProcessMessages;
    if Application.Terminated then Break;
  end;

end;

end.