Версия для печати темы

Нажмите сюда для просмотра этой темы в обычном формате

Форум на CrossPlatform.RU _ Qt Система рисования. Печать _ Урок 11 Попробуем выстрелить;Баг отрисовки - как исправить?

Автор: I_Wanna_Know 28.8.2014, 16:40

Доброго времени суток уважаемые форумчане!

Я новичок в Qt, изучаю фреймворк по урокам с вашего портала(за что отдельное спасибо старавшимся) в общем то в http://doc.crossplatform.ru/qt/4.3.5/tutorial-t11.html , при компиляции его с исходников (всё корректно скопипаcтенно опечатки исправлены) происходит некорректное отображение снаряда он по всей траектории полёта меняет свой размер(становится то больше то меньше), не на всём пути траектории отрисовывается и внезапно рывками появляется дальше по траектории и как бы "мерцает", сначала думал что это проблемы буфферизации, но почитав доки понял вроде как с Qt4+ двойная буферизация в виджетах включена по умолчанию ...в общем не знаю что и думать, помогите пожалуйста исправить этот баг или хотя бы подскажите как в рамках этого урока по другому переписать виджет.


Автор: FireBlack 30.8.2014, 12:27

Проще всего данный баг лечится простой заменой update(region); на update(); в функции CannonField::moveShot().

Кстати, чтобы избавиться от эффекта антиалиазинга (при отрисовки орудия) стоит добавить painter.setRenderHint(QPainter::Antialiasing, true); после QPainter painter(this); в функции CannonField::paintEvent(QPaintEvent *).


Автор: I_Wanna_Know 31.8.2014, 17:06

Цитата(FireBlack @ 30.8.2014, 13:27) *
Проще всего данный баг лечится простой заменой update(region); на update(); в функции CannonField::moveShot().


Огромное спасибо! Помогло! Эм если не затруднит не могли бы вы подробно объяснить почему так происходит?
Кстати иногда есть ещё такой слегка едва заметный шлейф типо тень от снаряда т.е это вроде бы(не уверен) отрисовка снаряда на предидущей позиции когда снаряд уже в следующей точке как бы это поправить?

Цитата(FireBlack @ 30.8.2014, 13:27) *
Кстати, чтобы избавиться от эффекта антиалиазинга (при отрисовки орудия) стоит добавить painter.setRenderHint(QPainter::Antialiasing, true); после QPainter painter(this); в функции CannonField::paintEvent(QPaintEvent *).

Ещё раз спасибо, что откликнулись, но об этом я знал и антиалиазинг сразу же включил :)

Автор: FireBlack 1.9.2014, 5:25

Цитата(I_Wanna_Know @ 31.8.2014, 17:06) *
Эм если не затруднит не могли бы вы подробно объяснить почему так происходит?

Точно не скажу, но предполагаю что region неполностью охватывает облать, необходимую для перерисовки. Свою теорию подтверждаю следующим кодом, где увеличиваю region на 10 пикселей в ширину и в высоту:
void CannonField::moveShot()
{
    QRect region = shotRect();
    ++timerCount;

    QRect shotR = shotRect();
    if (shotR.x() > width() || shotR.y() > height()) {
        autoShootTimer->stop();
    } else {
        region = region.united(shotR);
        const int margin = 5;
        region.adjust(-margin, -margin, margin, margin);
    }
    update(region);
}

Хотя это не решение, а костыль, т.к. он не учитывает скорость снаряда и если ее увеличить - мерцание опять появится.
Вообще, приведенный в примере код, должен теоретически работать корректно, может быть данные проблемы связаны с конкретной реализацией?! Неплохо бы проверить на Linux'е данный пример.

Цитата(I_Wanna_Know @ 31.8.2014, 17:06) *
Кстати иногда есть ещё такой слегка едва заметный шлейф типо тень от снаряда т.е это вроде бы(не уверен) отрисовка снаряда на предидущей позиции когда снаряд уже в следующей точке как бы это поправить?

Думаю это не более чем оптическая иллюзия из-за высокой скорости снаряда. Можете сами проверить, заменив autoShootTimer->start(5); на autoShootTimer->start(50); в функции CannonField::shoot().

Автор: Litkevich Yuriy 1.9.2014, 9:39

Цитата(FireBlack @ 1.9.2014, 8:25) *
но предполагаю что region неполностью охватывает облать, необходимую для перерисовки.
...
т.к. он не учитывает скорость снаряда
почти верно. Функция update() не перересовывает виджет, а только планирует его перерисовку (всего виджета или только его части). А вот сама перерисовка произойдёт через неопределённое время, поэтой причине пытаться учесть скорость не получиться. А ограничить перерисовку некоторой областью хочется, чтобы сократить затраты времени на неё.
По этому реальный выход из ситуации - эксперементально подобрать область виджета, в которой (одновременно) только что был, и в которой может оказаться снаряд.

Автор: I_Wanna_Know 1.9.2014, 11:19

Цитата(FireBlack @ 1.9.2014, 6:25) *
Точно не скажу, но предполагаю что region неполностью охватывает облать, необходимую для перерисовки

Я тоже о чём то подобном думал, но
Цитата(FireBlack @ 1.9.2014, 6:25) *
он не учитывает скорость снаряда и если ее увеличить - мерцание опять появится.

Учитывая это мне начинает казаться, что это не из-за того что он не всю область охватывает, а в том как работает сама функция update и учитывая
Цитата(Litkevich Yuriy @ 1.9.2014, 10:39) *
Функция update() не перересовывает виджет, а только планирует его перерисовку (всего виджета или только его части). А вот сама перерисовка произойдёт через неопределённое время,

И вызовется PaintEvent это написано в описании этой функции, но я то думал что это как раз таки для повышения производительности и более быстрой отрисовки оказалось что чуть ли не наоборот...
Цитата(Litkevich Yuriy @ 1.9.2014, 10:39) *
поэтой причине пытаться учесть скорость не получиться. А ограничить перерисовку некоторой областью хочется, чтобы сократить затраты времени на неё.
По этому реальный выход из ситуации - эксперементально подобрать область виджета, в которой (одновременно) только что был, и в которой может оказаться снаряд.


А вот это уже полный геморой как то это не красиво чтоли....
Кстати Litkevich Yuriy быть может вы знаете более надёжный и быстрый способ отрисовки 2D графики в Qt ? Вообще с помощью чего можно переделать этот урок?!

З.Ы, жаль на этом форуме нет функционала редактирования или удаления своих постов :-(

Автор: FireBlack 1.9.2014, 21:36

Цитата(I_Wanna_Know @ 1.9.2014, 12:19) *
... может вы знаете более надёжный и быстрый способ отрисовки 2D графики в Qt ? Вообще с помощью чего можно переделать этот урок?!

Хоть, вопрос адресован и не мне, постараюсь ответить: можно использовать сочетание http://qt-project.org/doc/qt-5/qgraphicsscene.html и http://qt-project.org/doc/qt-5/qgraphicsview.html. В документации есть множество http://qt-project.org/doc/qt-5/examples-graphicsview.html. В данном случае о головной боли с перерисовкой можно вообще забыть т.к. все это реализовано автоматически. Необходимо только перемещать снаряд изменяя его координаты.

Цитата(I_Wanna_Know @ 1.9.2014, 12:19) *
З.Ы, жаль на этом форуме нет функционала редактирования или удаления своих постов :-(

Подобный функционал скорее всего есть, но новичкам он не доступен.

Автор: Litkevich Yuriy 3.9.2014, 11:21

Цитата(I_Wanna_Know @ 1.9.2014, 14:19) *
но я то думал что это как раз таки для повышения производительности и более быстрой отрисовки
первая часть верна, в целом ПО работает быстрее, за счёт того, что рисование происходит не часто.


Цитата(I_Wanna_Know @ 1.9.2014, 14:19) *
надёжный и быстрый способ отрисовки 2D графики в Qt
т.н. в Qt система Arthur, т.е. использование QPainter в обработчике события paintEvent - это самый быстрый способ в Qt что-либо нарисовать, но он низкоуровневый.
В целом задача со снарядом решаема, только ручной работы много (можно запомнить область в которой снаряд был, а затем в обработчике рисования стереть только её, и т.п.)
Примеры по низкоуровневому рисованию смотрите в демке: %QTDIR%\demos\affine

Более удобный (но и более медленный) - http://doc.crossplatform.ru/qt/4.7.x/graphicsview.html


Цитата(FireBlack @ 2.9.2014, 0:36) *
Подобный функционал скорее всего есть, но новичкам он не доступен.
через 20 сообщений появится

Автор: I_Wanna_Know 5.9.2014, 23:12

Спасибо всем за ответы пожалуй я действительно буду пытаться постигать QGraphicsScene и QGraphicsView, мне он кажутся по цивильнее
И ещё, чтобы не плодить темы спрошу тут:
Подскажите пожалуйста, возможно ли "скрестить" Qt+Visual Studio чтобы можно было пользоваться Visual Studio дебажа Qt проект и использовать инструмент визуалки "карта кода"(очень удобная вещь), если да то как?

Автор: Litkevich Yuriy 6.9.2014, 10:14

Цитата(I_Wanna_Know @ 6.9.2014, 2:12) *
И ещё, чтобы не плодить темы спрошу тут:
Правило: в одной теме решается только одна проблема

Автор: Litkevich Yuriy 18.10.2014, 17:51

Тему разделил: http://www.forum.crossplatform.ru/index.php?showtopic=10187

Форум Invision Power Board (http://www.invisionboard.com)
© Invision Power Services (http://www.invisionpower.com)