有人可以向我展示一个使用Vector2.as_polar()
获取距离和方位角的示例,然后将该信息重新插入Vector2.from_polar()
以获取笛卡尔坐标吗?
(0, 0)
假设我们想要 和 之间的距离和方位角(15, 10)
,然后将该信息插入Vector2.from_polar()
以再次生成(15, 10)
。
我会准确地解释我在做什么,但对我和其他人来说,只看一个工作示例可能会更容易。
有人可以向我展示一个使用Vector2.as_polar()
获取距离和方位角的示例,然后将该信息重新插入Vector2.from_polar()
以获取笛卡尔坐标吗?
(0, 0)
假设我们想要 和 之间的距离和方位角(15, 10)
,然后将该信息插入Vector2.from_polar()
以再次生成(15, 10)
。
我会准确地解释我在做什么,但对我和其他人来说,只看一个工作示例可能会更容易。
您可以将矢量定义为半径和角度 - 极坐标。
让我们使用屏幕中心的一个点向量作为我们极坐标系的极点。
pole = Vector2(screen_rect.center)
要借助该from_polar
方法设置向量的坐标,首先定义一个向量,然后传递一个由半径和角度组成的元组:
vec = Vector2()
# This updates the cartesian coordinates of vec.
vec.from_polar((90, 60)) # 90 pixels long, rotated 60 degrees.
然后使用pygame.draw.line
绘制它(添加vec
以pole
获取目标坐标):
pg.draw.line(screen, (200, 90, 20), pole, pole+vec, 3)
作为 的演示as_polar
,计算到鼠标位置的向量并将极坐标解包到变量r
中phi
,我们稍后会渲染这些变量以查看该向量的当前半径和角度。
r, phi = (mouse_pos-pole).as_polar()
请注意,pygame 的 y 轴是翻转的,因此负角向上,正角向下。
import sys
import pygame as pg
from pygame.math import Vector2
def main():
screen = pg.display.set_mode((640, 480))
screen_rect = screen.get_rect()
font = pg.font.Font(None, 30)
color = pg.Color(150, 250, 100)
clock = pg.time.Clock()
pole = Vector2(screen_rect.center)
# A zero vector.
vec = Vector2()
# To set the coordinates of `vec` we can pass the
# polar coordinates (radius, angle) to `from_polar`.
vec.from_polar((90, 60))
print(vec) # Now take a look at the updated cartesian coordinates.
done = False
while not done:
for event in pg.event.get():
if event.type == pg.QUIT:
done = True
mouse_pos = pg.mouse.get_pos()
# `as_polar` gives us the radius and angle of the vector that
# points to the mouse. Unpack it into the `r` and `phi` variables.
r, phi = (mouse_pos-pole).as_polar()
screen.fill((30, 30, 30))
# Draw the line to the mouse pos.
pg.draw.line(screen, color, pole, mouse_pos, 3)
# Add the vec to the pole to get the target coordinates.
pg.draw.line(screen, (200, 90, 20), pole, pole+vec, 3)
# Render the radius and angle.
txt = font.render('r: {:.1f}'.format(r), True, color)
screen.blit(txt, (30, 30))
txt = font.render('phi: {:.1f}'.format(phi), True, color)
screen.blit(txt, (30, 50))
pg.display.flip()
clock.tick(30)
if __name__ == '__main__':
pg.init()
main()
pg.quit()
sys.exit()
这是另一个跟随鼠标的精灵示例(按“w”或“s”加速)。我将self.speed
作为半径和as_polar
作为参数返回的角度传递给以from_polar
设置vel
玩家精灵的ocity。
import sys
import pygame as pg
class Player(pg.sprite.Sprite):
def __init__(self, pos):
super().__init__()
self.image = pg.Surface((50, 30), pg.SRCALPHA)
color = pg.Color('dodgerblue1')
pg.draw.polygon(self.image, color, ((1, 1), (49, 15), (1, 29)))
self.orig_img = self.image
self.rect = self.image.get_rect(center=pos)
self.pos = pg.math.Vector2(pos)
self.vel = pg.math.Vector2(0, 0)
self.speed = 0
def update(self):
self.rotate()
self.pos += self.vel
self.rect.center = self.pos
def rotate(self):
_, angle = (pg.mouse.get_pos()-self.pos).as_polar()
self.vel.from_polar((self.speed, angle))
self.image = pg.transform.rotozoom(self.orig_img, -angle, 1)
self.rect = self.image.get_rect(center=self.rect.center)
def main():
screen = pg.display.set_mode((640, 480))
clock = pg.time.Clock()
all_sprites = pg.sprite.Group()
player = Player((300, 200))
all_sprites.add(player)
done = False
while not done:
for event in pg.event.get():
if event.type == pg.QUIT:
done = True
keys = pg.key.get_pressed()
if keys[pg.K_w]:
player.speed += .2
elif keys[pg.K_s]:
player.speed -= .2
all_sprites.update()
screen.fill((30, 30, 30))
all_sprites.draw(screen)
pg.display.flip()
clock.tick(30)
if __name__ == '__main__':
pg.init()
main()
pg.quit()
sys.exit()