Рисование в Pygame с потоками, отображение Поверхность завершается при нажатии на поверхность - PullRequest
0 голосов
/ 30 апреля 2019

Я хочу сделать начальную анимацию, поэтому я подумал об использовании основного потока, который рисует вещи, и множества маленьких потоков, которые отправляют запросы на рисование.

Это потому, что я хочу нарисовать несколько вещей одновременно (например, прямоугольник будет рисоваться прогрессивно, и как только он достигнет 50% от его рисунка, я хочу начать рисовать что-то еще, не здесь, это просто тест)

Итак, я проводил тест, и он работает нормально, однако, когда я пытался перетянуть поверхность, я получаю следующее исключение:

Traceback (most recent call last):
  File "C:\Python27\lib\threading.py", line 801, in __bootstrap_inner
    self.run()
  File "C:\Python27\lib\threading.py", line 754, in run
    self.__target(*self.__args, **self.__kwargs)
  File "C:\checkouts\anim_project\test.py", line 74, in opening_animation_drawer_thread
    drawing_task.draw_yourself(main_window_surface)
  File "C:\checkouts\anim_project\test.py", line 31, in draw_yourself
    main_window_surface.blit(self.surfaces[self.pos_in_list], self.pos_in_screen)
error: display Surface quit

вот программа, для ее запуска вам нужна папка "ball" с несколькими png / jpg внутри, здесь ссылка Google с архивом, который содержит все: https://drive.google.com/open?id=1dW1_kf4trEvH0j_A-1wQ4fW3olK1_7mP

и вот код:

#!/usr/bin/env python
#-*- coding: utf-8 -*-

import time
import pygame
import os

from threading import Thread
from pygame.locals import *
from multiprocessing import Queue
from os import listdir
from os.path import isfile, join

class DrawingTaskLine():
    def __init__(self, start, end, color, thickness):
        self.start = start
        self.end = end
        self.color = color
        self.thickness = thickness

    def draw_yourself(self, main_window_surface):
        pygame.draw.line(main_window_surface, self.color, self.start, self.end,  self.thickness)

class DrawingTaskImage():
    def __init__(self, surfaces, pos_in_list, pos_in_screen):
        self.surfaces = surfaces
        self.pos_in_list = pos_in_list
        self.pos_in_screen = pos_in_screen

    def draw_yourself(self, main_window_surface):
        main_window_surface.blit(self.surfaces[self.pos_in_list], self.pos_in_screen)

def get_all_files_from(my_path):
    file_list_without_path = [f for f in listdir(my_path) if
                              isfile(join(my_path, f)) and 'desktop.ini' not in f]
    return file_list_without_path

def load_all_images_from(my_path, resize_hw=None):
    res = []
    for im_name in get_all_files_from(my_path):
        im_surface = pygame.image.load(os.path.abspath(join(my_path, im_name)))
        if resize_hw is not None:
            im_surface = pygame.transform.scale(im_surface, resize_hw)
        res.append(im_surface)
    return res

commonQueue = Queue()
openingAnimationisRunning = True

main_window_im_width = 900
main_window_im_height = 750
pygame.init()
window = pygame.display.set_mode((main_window_im_width, main_window_im_height))

#making main window
background_color = (50 , 50, 50)
origin_position = (0, 0)
main_window_surface = pygame.Surface((main_window_im_width, main_window_im_height))
main_window_surface.fill(background_color)
window.blit(main_window_surface, origin_position)

print "Loading images....."
ball_size_hw = 100, 100
ball_surfaces = load_all_images_from("ball", ball_size_hw)
print "Loading completed!"

def opening_animation_drawer_thread():
    while openingAnimationisRunning:
        drawing_task = commonQueue.get()

        if drawing_task is None:
            continue

        drawing_task.draw_yourself(main_window_surface)
        window.blit(main_window_surface, origin_position)

def draw_horizonral_lines_animation():
    y = 0
    color_counter = 0
    x_left = 50
    x_right = 300
    colors = [(255,0,0), (0,255,0), (0,0,255)]
    thickness = 1
    while True:
        x1 = (x_left, y)
        x2 = (x_right, y)
        drawing_task = DrawingTaskLine(x1, x2, colors[color_counter], thickness)
        commonQueue.put(drawing_task)
        y += 1
        if (y == main_window_im_height):
            y = 0
            color_counter += 1
        if (color_counter == 3):
            color_counter = 0

def draw_vertical_lines_animation():
    x_start = 350
    x = x_start
    color_counter = 0
    y_top= 400
    y_bottom = 500
    colors = [(255,0,0), (0,255,0), (0,0,255)]
    thickness = 1
    while True:
        y1 = (x, y_top)
        y2 = (x, y_bottom)
        drawing_task = DrawingTaskLine(y1, y2, colors[color_counter], thickness)
        commonQueue.put(drawing_task)
        x += 1
        if (x == main_window_im_width-50):
            x = x_start
            color_counter += 1
        if (color_counter == 3):
            color_counter = 0


def draw_ball_animation():
    time.sleep(5)
    ball_pos = 540, 140
    for n in range(len(ball_surfaces)):
        commonQueue.put(DrawingTaskImage(ball_surfaces, n, ball_pos))
        time.sleep(0.01)


def launch_threaded_animation():
    main_drawer_thread = Thread(target=opening_animation_drawer_thread)
    main_drawer_thread.start()

    draw_horizonral_lines_animation_thread = Thread(target=draw_horizonral_lines_animation)
    draw_horizonral_lines_animation_thread.start()

    draw_vertical_lines_animation_thread = Thread(target=draw_vertical_lines_animation)
    draw_vertical_lines_animation_thread.start()

    draw_ball_animation_thread = Thread(target=draw_ball_animation)
    draw_ball_animation_thread.start()


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

    # do your stuff here once all thread have finished
    pygame.display.flip()

Идея состоит в том, что после завершения всех анимаций они устанавливают логическое значение True, и начинается настоящая игра.

Вы понимаете, почему я не могу пройти поверхность в очереди?

Если я просто отправлю позицию в списке и использую список, значит, она работает.

Спасибо

...