Волны в антеннах
https://vk.com/pobeditelvp
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
264 lines
9.3 KiB
264 lines
9.3 KiB
<!-- Copyright 2023 by Alexei Bezborodov <AlexeiBv+mirocod_wave_refraction@narod.ru> --> |
|
<!-- public domain --> |
|
|
|
<html lang="ru"> |
|
|
|
<head> |
|
<meta charset="UTF-8"> |
|
<title>Переотражение волн</title> |
|
|
|
<!-- |
|
Загрузка скриптов из интернета |
|
<script type="text/javascript" src="https://cdn.jsdelivr.net/npm/brython@3.8.9/brython.min.js"></script> |
|
<script type="text/javascript" src="https://cdn.jsdelivr.net/npm/brython@3.8.9/brython_stdlib.js"></script> |
|
--> |
|
|
|
<!-- |
|
Загрузка скриптов из локальной папки (рядом с файлом html) |
|
--> |
|
<script type="text/javascript" src="brython.min.js"></script> |
|
<script type="text/javascript" src="brython_stdlib.js"></script> |
|
|
|
|
|
</head> |
|
|
|
<body onload="brython()"> |
|
|
|
<details open> |
|
<summary>Расчётные данные</summary> |
|
<div> |
|
<p> |
|
<input type="range" id="max_step_range" for="max_step" name="volume" min="10" max="1000" value="100" |
|
oninput='document.getElementById("max_step").value = document.getElementById("max_step_range").value; recalculate();'> |
|
<input id = "max_step" type="number" value="100" step="1" |
|
oninput='recalculate();'/> |
|
<label for="max_step" alt="Количество расчётных точек на всю длину среды">Точность расчёта</label> |
|
</p> |
|
|
|
<p> |
|
<input type="checkbox" id="right_refraction" checked |
|
oninput='recalculate();'/> |
|
<label for="right_refraction" alt="">Отражения и переотражения справа</label> |
|
</p> |
|
|
|
<p> |
|
<input type="range" id="max_right_refraction_range" for="max_step" name="volume" min="0" max="100" value="3" |
|
oninput='document.getElementById("max_right_refraction").value = document.getElementById("max_right_refraction_range").value; recalculate();'> |
|
<input id = "max_right_refraction" type="number" value="3" step="1" |
|
oninput='recalculate();'/> |
|
<label for="max_right_refraction" alt="">Количество переотражений справа</label> |
|
</p> |
|
|
|
<p> |
|
<input type="checkbox" id="left_refraction" checked |
|
oninput='recalculate();'/> |
|
<label for="left_refraction" alt="">Отражения и переотражения слева</label> |
|
</p> |
|
|
|
<p> |
|
<input type="range" id="max_left_refraction_range" for="max_step" name="volume" min="0" max="100" value="3" |
|
oninput='document.getElementById("max_left_refraction").value = document.getElementById("max_left_refraction_range").value; recalculate();'> |
|
<input id = "max_left_refraction" type="number" value="3" step="1" |
|
oninput='recalculate();'/> |
|
<label for="max_left_refraction" alt="">Количество переотражений слева</label> |
|
</p> |
|
|
|
<p> |
|
<input type="range" id="length_range" for="omega" name="volume" min="0" max="200" value="50" |
|
oninput='document.getElementById("length").value = document.getElementById("length_range").value / 100; recalculate();'> |
|
<input id = "length" type="number" value="0.5" step=".01" |
|
oninput='recalculate();'/> |
|
<label for="length" alt="">Длина среды (в длинах волн)</label> |
|
</p> |
|
|
|
<p> |
|
<input type="checkbox" id="invert_phase" checked |
|
oninput='recalculate();'/> |
|
<label for="invert_phase" alt="">Инвертирование волны при отражении</label> |
|
</p> |
|
|
|
<p> |
|
<input type="range" id="k_refraction_range" for="omega" name="volume" min="0" max="100" value="60" |
|
oninput='document.getElementById("k_refraction").value = document.getElementById("k_refraction_range").value / 100; recalculate();'> |
|
<input id = "k_refraction" type="number" value="0.6" step=".01" |
|
oninput='recalculate();'/> |
|
<label for="k_refraction" alt="">Коэффициент отражения</label> |
|
</p> |
|
|
|
<p> |
|
<input type="range" id="omega_range" for="omega" name="volume" min="-1000" max="1000" value="0" |
|
oninput='document.getElementById("omega").value = document.getElementById("omega_range").value / 100; recalculate();'> |
|
<input id = "omega" type="number" value="0" step=".01" |
|
oninput='recalculate();'/> |
|
<label for="omega" alt="">Начальная фаза главной функции</label> |
|
</p> |
|
|
|
<button id="recalculate">Пересчитать</button> |
|
</div> |
|
</details> |
|
|
|
<h1 class="text-center">Графики волн</h1> |
|
<canvas id="draw-board" width="1024" height="500"></canvas> |
|
|
|
<script type="text/python"> |
|
|
|
from browser import document, html, window |
|
import random, math |
|
|
|
canvas = document["draw-board"] |
|
ctx = canvas.getContext("2d") |
|
|
|
x_border = 10 |
|
y_border = 10 |
|
|
|
# Чёрный экран |
|
def DrawBlackScreen(): |
|
ctx.fillStyle = "black" |
|
ctx.fillRect(0, 0, canvas.width, canvas.height) |
|
|
|
def DrawLine(x1, y1, x2, y2, lineWidth, style): |
|
ctx.strokeStyle = style |
|
ctx.lineWidth = lineWidth |
|
ctx.beginPath() |
|
ctx.moveTo(x1, y1) |
|
ctx.lineTo(x2, y2) |
|
ctx.stroke() |
|
return |
|
|
|
def DrawAxesScreen(): |
|
ctx.fillRect(0, 0, canvas.width, canvas.height) |
|
x1 = x_border |
|
x2 = canvas.width - x1 |
|
y1 = y_border |
|
y2 = canvas.height - y1 |
|
ym = canvas.height/2 |
|
style = 'red' |
|
lw = 2 |
|
DrawLine(x1, ym, x2, ym, lw, style) |
|
DrawLine(x1, y1, x1, y2, lw, style) |
|
DrawLine(x2, y1, x2, y2, lw, style) |
|
|
|
|
|
def max_step_range(event): |
|
#document["max_step"].value = document["max_step_range"].value |
|
document["max_step"].value = 103 |
|
|
|
def base_func(x, omega): |
|
return math.sin(x + omega) |
|
|
|
def recalculate(): |
|
max_step = int(document["max_step"].value) |
|
r = document["invert_phase"].checked |
|
if r: |
|
r = -1 |
|
else: |
|
r = 1 |
|
k_refraction = float(document["k_refraction"].value) |
|
omega = float(document["omega"].value) |
|
|
|
right_refraction = document["right_refraction"].checked |
|
max_right_refraction = int(document["max_right_refraction"].value) |
|
|
|
left_refraction = document["left_refraction"].checked |
|
max_left_refraction = int(document["max_left_refraction"].value) |
|
|
|
length = float(document["length"].value) |
|
|
|
DrawBlackScreen() |
|
DrawAxesScreen() |
|
|
|
right_refraction_last_values = [0.0] * max_right_refraction; |
|
left_refraction_last_values = [0.0] * max_left_refraction; |
|
|
|
result = 0.0 |
|
result_last_value = result |
|
|
|
base_func_value = 0.0 |
|
base_func_last_value = result |
|
|
|
pi = math.pi |
|
|
|
x1 = x_border |
|
x2 = canvas.width - x1 |
|
y1 = y_border |
|
y2 = canvas.height - y1 |
|
ym = canvas.height/2 |
|
style = 'white' |
|
style_right = '#FF0000' |
|
style_left = '#00FF00' |
|
style_result = '#00FFFF' |
|
line_width_scale = 4.0 |
|
y_scale = 0.3 |
|
|
|
cur_x = 0 |
|
prev_x = cur_x |
|
|
|
for i in range(max_step): |
|
result = 0 |
|
x = i / max_step * 2.0 * pi * length + omega |
|
|
|
cur_x = x1 + (x2 - x1) * i / max_step |
|
|
|
omega1 = 0.0 * omega |
|
|
|
base_func_value = base_func(x, omega1) |
|
if i > 0: |
|
base_y = ym - (ym - y1) * y_scale * base_func_value |
|
base_y_last = ym - (ym - y1) * y_scale * base_func_last_value |
|
|
|
DrawLine(prev_x, base_y_last, cur_x, base_y, 1.0 * line_width_scale, style) |
|
base_func_last_value = base_func_value |
|
result += base_func_value |
|
|
|
for r_i in range(max_right_refraction * int(right_refraction) ): |
|
cur_k = k_refraction ** (r_i + 1) |
|
cur_r = r ** (r_i + 1) |
|
cur_znak = (-1) ** r_i |
|
func_value = cur_k * cur_r * base_func( ((r_i + 1) + (r_i + 1) % 2) * length * 2 * pi - cur_znak * x, omega1) |
|
if i > 0: |
|
base_y = ym - (ym - y1) * y_scale * func_value |
|
base_y_last = ym - (ym - y1) * y_scale * right_refraction_last_values[r_i] |
|
|
|
DrawLine(prev_x, base_y_last, cur_x, base_y, cur_k * line_width_scale, style_right) |
|
right_refraction_last_values[r_i] = func_value |
|
result += func_value |
|
|
|
for r_i in range(max_left_refraction * int(left_refraction) ): |
|
cur_k = k_refraction ** (r_i + 1) |
|
cur_r = r ** (r_i + 1) |
|
cur_znak = (-1) ** r_i |
|
func_value = cur_k * cur_r * base_func( -((r_i + 1) - (r_i + 1) % 2) * length * 2 * pi - cur_znak * x, omega1) |
|
if i > 0: |
|
base_y = ym - (ym - y1) * y_scale * func_value |
|
base_y_last = ym - (ym - y1) * y_scale * left_refraction_last_values[r_i] |
|
|
|
DrawLine(prev_x, base_y_last, cur_x, base_y, cur_k * line_width_scale, style_left) |
|
left_refraction_last_values[r_i] = func_value |
|
result += func_value |
|
|
|
if i > 0: |
|
base_y = ym - (ym - y1) * y_scale * result |
|
base_y_last = ym - (ym - y1) * y_scale * result_last_value |
|
|
|
DrawLine(prev_x, base_y_last, cur_x, base_y, 2.0 * line_width_scale, style_result) |
|
result_last_value = result |
|
|
|
|
|
prev_x = cur_x |
|
|
|
def recalculate_event(event): |
|
recalculate() |
|
|
|
|
|
window.recalculate = recalculate |
|
document["recalculate"].bind("click", recalculate_event) |
|
|
|
#document["omega"].bind("onchange", recalculate) |
|
|
|
|
|
|
|
</script> |
|
|
|
</body> |
|
|
|
</html>
|
|
|