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.
293 lines
5.9 KiB
293 lines
5.9 KiB
14 years ago
|
/****************************************************************
|
||
|
**
|
||
|
** Implementation CannonField class, Qt tutorial 14
|
||
|
**
|
||
|
****************************************************************/
|
||
|
|
||
|
#include "cannon.h"
|
||
|
#include <qtimer.h>
|
||
|
#include <qpainter.h>
|
||
|
#include <qpixmap.h>
|
||
|
#include <qdatetime.h>
|
||
|
|
||
|
#include <math.h>
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
|
||
|
CannonField::CannonField( QWidget *parent, const char *name )
|
||
|
: QWidget( parent, name )
|
||
|
{
|
||
|
ang = 45;
|
||
|
f = 0;
|
||
|
timerCount = 0;
|
||
|
autoShootTimer = new QTimer( this, "movement handler" );
|
||
|
connect( autoShootTimer, SIGNAL(timeout()),
|
||
|
this, SLOT(moveShot()) );
|
||
|
shoot_ang = 0;
|
||
|
shoot_f = 0;
|
||
|
target = QPoint( 0, 0 );
|
||
|
gameEnded = FALSE;
|
||
|
barrelPressed = FALSE;
|
||
|
setPalette( QPalette( QColor( 250, 250, 200) ) );
|
||
|
newTarget();
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::setAngle( int degrees )
|
||
|
{
|
||
|
if ( degrees < 5 )
|
||
|
degrees = 5;
|
||
|
if ( degrees > 70 )
|
||
|
degrees = 70;
|
||
|
if ( ang == degrees )
|
||
|
return;
|
||
|
ang = degrees;
|
||
|
repaint( cannonRect(), FALSE );
|
||
|
emit angleChanged( ang );
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::setForce( int newton )
|
||
|
{
|
||
|
if ( newton < 0 )
|
||
|
newton = 0;
|
||
|
if ( f == newton )
|
||
|
return;
|
||
|
f = newton;
|
||
|
emit forceChanged( f );
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::shoot()
|
||
|
{
|
||
|
if ( isShooting() )
|
||
|
return;
|
||
|
timerCount = 0;
|
||
|
shoot_ang = ang;
|
||
|
shoot_f = f;
|
||
|
autoShootTimer->start( 50 );
|
||
|
emit canShoot( FALSE );
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::newTarget()
|
||
|
{
|
||
|
static bool first_time = TRUE;
|
||
|
if ( first_time ) {
|
||
|
first_time = FALSE;
|
||
|
QTime midnight( 0, 0, 0 );
|
||
|
srand( midnight.secsTo(QTime::currentTime()) );
|
||
|
}
|
||
|
QRegion r( targetRect() );
|
||
|
target = QPoint( 200 + rand() % 190,
|
||
|
10 + rand() % 255 );
|
||
|
repaint( r.unite( targetRect() ) );
|
||
|
}
|
||
|
|
||
|
void CannonField::setGameOver()
|
||
|
{
|
||
|
if ( gameEnded )
|
||
|
return;
|
||
|
if ( isShooting() )
|
||
|
autoShootTimer->stop();
|
||
|
gameEnded = TRUE;
|
||
|
repaint();
|
||
|
}
|
||
|
|
||
|
void CannonField::restartGame()
|
||
|
{
|
||
|
if ( isShooting() )
|
||
|
autoShootTimer->stop();
|
||
|
gameEnded = FALSE;
|
||
|
repaint();
|
||
|
emit canShoot( TRUE );
|
||
|
}
|
||
|
|
||
|
void CannonField::moveShot()
|
||
|
{
|
||
|
QRegion r( shotRect() );
|
||
|
timerCount++;
|
||
|
|
||
|
QRect shotR = shotRect();
|
||
|
|
||
|
if ( shotR.intersects( targetRect() ) ) {
|
||
|
autoShootTimer->stop();
|
||
|
emit hit();
|
||
|
emit canShoot( TRUE );
|
||
|
} else if ( shotR.x() > width() || shotR.y() > height() ||
|
||
|
shotR.intersects(barrierRect()) ) {
|
||
|
autoShootTimer->stop();
|
||
|
emit missed();
|
||
|
emit canShoot( TRUE );
|
||
|
} else {
|
||
|
r = r.unite( QRegion( shotR ) );
|
||
|
}
|
||
|
|
||
|
repaint( r );
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::mousePressEvent( QMouseEvent *e )
|
||
|
{
|
||
|
if ( e->button() != LeftButton )
|
||
|
return;
|
||
|
if ( barrelHit( e->pos() ) )
|
||
|
barrelPressed = TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::mouseMoveEvent( QMouseEvent *e )
|
||
|
{
|
||
|
if ( !barrelPressed )
|
||
|
return;
|
||
|
QPoint pnt = e->pos();
|
||
|
if ( pnt.x() <= 0 )
|
||
|
pnt.setX( 1 );
|
||
|
if ( pnt.y() >= height() )
|
||
|
pnt.setY( height() - 1 );
|
||
|
double rad = atan(((double)rect().bottom()-pnt.y())/pnt.x());
|
||
|
setAngle( qRound ( rad*180/3.14159265 ) );
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::mouseReleaseEvent( QMouseEvent *e )
|
||
|
{
|
||
|
if ( e->button() == LeftButton )
|
||
|
barrelPressed = FALSE;
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::paintEvent( QPaintEvent *e )
|
||
|
{
|
||
|
QRect updateR = e->rect();
|
||
|
QPainter p( this );
|
||
|
|
||
|
if ( gameEnded ) {
|
||
|
p.setPen( black );
|
||
|
p.setFont( QFont( "Courier", 48, QFont::Bold ) );
|
||
|
p.drawText( rect(), AlignCenter, "Game Over" );
|
||
|
}
|
||
|
if ( updateR.intersects( cannonRect() ) )
|
||
|
paintCannon( &p );
|
||
|
if ( updateR.intersects( barrierRect() ) )
|
||
|
paintBarrier( &p );
|
||
|
if ( isShooting() && updateR.intersects( shotRect() ) )
|
||
|
paintShot( &p );
|
||
|
if ( !gameEnded && updateR.intersects( targetRect() ) )
|
||
|
paintTarget( &p );
|
||
|
}
|
||
|
|
||
|
void CannonField::paintShot( QPainter *p )
|
||
|
{
|
||
|
p->setBrush( black );
|
||
|
p->setPen( NoPen );
|
||
|
p->drawRect( shotRect() );
|
||
|
}
|
||
|
|
||
|
|
||
|
void CannonField::paintTarget( QPainter *p )
|
||
|
{
|
||
|
p->setBrush( red );
|
||
|
p->setPen( black );
|
||
|
p->drawRect( targetRect() );
|
||
|
}
|
||
|
|
||
|
void CannonField::paintBarrier( QPainter *p )
|
||
|
{
|
||
|
p->setBrush( yellow );
|
||
|
p->setPen( black );
|
||
|
p->drawRect( barrierRect() );
|
||
|
}
|
||
|
|
||
|
const QRect barrelRect(33, -4, 15, 8);
|
||
|
|
||
|
void CannonField::paintCannon( QPainter *p )
|
||
|
{
|
||
|
QRect cr = cannonRect();
|
||
|
QPixmap pix( cr.size() );
|
||
|
pix.fill( this, cr.topLeft() );
|
||
|
|
||
|
QPainter tmp( &pix );
|
||
|
tmp.setBrush( blue );
|
||
|
tmp.setPen( NoPen );
|
||
|
|
||
|
tmp.translate( 0, pix.height() - 1 );
|
||
|
tmp.drawPie( QRect( -35,-35, 70, 70 ), 0, 90*16 );
|
||
|
tmp.rotate( -ang );
|
||
|
tmp.drawRect( barrelRect );
|
||
|
tmp.end();
|
||
|
|
||
|
p->drawPixmap( cr.topLeft(), pix );
|
||
|
}
|
||
|
|
||
|
|
||
|
QRect CannonField::cannonRect() const
|
||
|
{
|
||
|
QRect r( 0, 0, 50, 50 );
|
||
|
r.moveBottomLeft( rect().bottomLeft() );
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
|
||
|
QRect CannonField::shotRect() const
|
||
|
{
|
||
|
const double gravity = 4;
|
||
|
|
||
|
double time = timerCount / 4.0;
|
||
|
double velocity = shoot_f;
|
||
|
double radians = shoot_ang*3.14159265/180;
|
||
|
|
||
|
double velx = velocity*cos( radians );
|
||
|
double vely = velocity*sin( radians );
|
||
|
double x0 = ( barrelRect.right() + 5 )*cos(radians);
|
||
|
double y0 = ( barrelRect.right() + 5 )*sin(radians);
|
||
|
double x = x0 + velx*time;
|
||
|
double y = y0 + vely*time - 0.5*gravity*time*time;
|
||
|
|
||
|
QRect r = QRect( 0, 0, 6, 6 );
|
||
|
r.moveCenter( QPoint( qRound(x), height() - 1 - qRound(y) ) );
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
|
||
|
QRect CannonField::targetRect() const
|
||
|
{
|
||
|
QRect r( 0, 0, 20, 10 );
|
||
|
r.moveCenter( QPoint(target.x(),height() - 1 - target.y()) );
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
|
||
|
QRect CannonField::barrierRect() const
|
||
|
{
|
||
|
return QRect( 145, height() - 100, 15, 100 );
|
||
|
}
|
||
|
|
||
|
|
||
|
bool CannonField::barrelHit( const QPoint &p ) const
|
||
|
{
|
||
|
QWMatrix mtx;
|
||
|
mtx.translate( 0, height() - 1 );
|
||
|
mtx.rotate( -ang );
|
||
|
mtx = mtx.invert();
|
||
|
return barrelRect.contains( mtx.map(p) );
|
||
|
}
|
||
|
|
||
|
|
||
|
bool CannonField::isShooting() const
|
||
|
{
|
||
|
return autoShootTimer->isActive();
|
||
|
}
|
||
|
|
||
|
|
||
|
QSize CannonField::sizeHint() const
|
||
|
{
|
||
|
return QSize( 400, 300 );
|
||
|
}
|
||
|
|
||
|
|
||
|
QSizePolicy CannonField::sizePolicy() const
|
||
|
{
|
||
|
return QSizePolicy( QSizePolicy::Expanding, QSizePolicy::Expanding );
|
||
|
}
|