【78】8台球【源码】

[复制链接]

该用户从未签到

2380

主题

2433

帖子

9139

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
9139
QQ
跳转到指定楼层
楼主
发表于 2017-6-13 11:20:39 | 只看该作者 |只看大图 回帖奖励 |正序浏览 |阅读模式

想要查看内容赶紧注册登陆吧!

您需要 登录 才可以下载或查看,没有帐号?立即注册

x


#include "StdAfx.h"
#include ".\ballrender.h"
//
// float  Sta_friction = 0.003f; //球的滑动摩擦系数
// float  Rou_friction = 0.00022f; //球的滚动摩擦系数
// float  Xuan_friction = 0.00014f;  //球的旋切摩擦系数
//
// float  Sta_friction = 0.00299f; //球的滑动摩擦系数
// float  Rou_friction = 0.00030f; //球的滚动摩擦系数
// float  Xuan_friction = 0.0002f;  //球的旋切摩擦系数
float  dt=1.0f ;
float  Sta_friction = 0.003f; //球的滑动摩擦系数
float  Rou_friction = 0.00022f; //球的滚动摩擦系数
float  Xuan_friction = 0.00014f;  //球的旋切摩擦系数
// float  Sta_friction = 1.003f; //球的滑动摩擦系数
// float  Rou_friction = 0.00025f; //球的滚动摩擦系数
// float  Xuan_friction = 0.00019f;  //球的旋切摩擦系数




Vector3 light;
Vector3  Pos[128][128];
BYTE mapX[128][128];
float diff [128][128];
Vector3 ver[32][32];
short FJC [128][128];

HTEXTURE        texLight=0;
HTEXTURE        texShadow=0;
HTEXTURE        texBall=0;

CHgeSprite*        pSpriteShadow=NULL;
CHgeSprite*        pSprite=NULL;
CHgeSprite*     pSpriteLight=NULL;

CBallRender::CBallRender(void)
{   
    Iz =M*banjin*banjin*2/5;
    SENSTA_FRICTION = Sta_friction * 1.f + Sta_friction * M / Iz * banjin * banjin;  //滑动摩擦最小检测阀值
    SENROU_FRICTION = Rou_friction * 1.f;  //滚动摩擦最小检测阀值

    BALL_State=4;     //球静止
    BALL_bEnable=true ;  //    是否有效
    BALL_V=Vector3(0);
    BALL_Vang=Vector3(0);
    BALL_Orientation=matrix3_identity();   
    BALL_bIsRedraw =false;   
    BALL_bIsInit =false;
    tex = NULL;
    offsetx = 0;
    offsety = 0;
    banJin = 16;
    zhiJin = 32;
    binhole = false;
    pSpriteLight = NULL;
    m_pIGameFrameView = NULL;
}

CBallRender::~CBallRender(void)
{   
//    m_GraphHelper->Free();
}

//应用力
void CBallRender::ApplyForce(Vector3 force,Vector3 colpos)
{
    float  dt=1.0f ;
    apply_impulse (force* dt, colpos);
}

void CBallRender::apply_impulse(Vector3 impulse ,Vector3 colpos)//摩擦力
{
    Vector3 Ra , vatmp ;

    BALL_V+= (impulse/ M);//线速度
    if(impulse.Z!=0.0&&abs(impulse.Z)>1)
    {
        BALL_V.X /= impulse.Z;
        BALL_V.Y /= impulse.Z;
    }
    BALL_V.Z = 0 ; //不能够向下

    Ra= colpos- Ball_Pos;

    vatmp = cross(Ra, impulse) / Iz    ;
    BALL_Vang+=vatmp;//角速度

    static Vector3 temp = BALL_V;
    TRACE("BALL_Vang.X=%f,BALL_Vang.Y=%f,BALL_Vang.Z=%f",BALL_Vang.X,BALL_Vang.Y,BALL_Vang.Z);
    TRACE("BALL_V.X=%f,BALL_V.Y=%f,BALL_V.Z=%f",BALL_V.X,BALL_V.Y,BALL_V.Z);
    TRACE("增量:x = %f,y = %f,z = %f",BALL_V.X - temp.X,BALL_V.Y - temp.Y,BALL_V.Z-temp.Z);
    temp = BALL_V;
}

void CBallRender::Reset(int _ballNo,Vector3 _ballPos,bool _enable)
{
    Ball_Pos =_ballPos;
    BallNo =_ballNo;
    BALL_bEnable =_enable;
}

void CBallRender::Init(int _ballNo,Vector3 _ballPos)
{
    HGE* pHge = NULL;
    if (NULL != m_pIGameFrameView)
        pHge = m_pIGameFrameView->GetHge();
   
//    ASSERT(NULL != pHge);
    if (NULL == pHge) return;

    Ball_Pos =_ballPos;//Vector3(rand()%700+20,rand()%500+20,0);   
    BallNo =_ballNo;   
    if (tex)
    {
        BALL_bEnable = true;
        return;
    }

    //tex = m_GraphHelper->Texture_Create(zhiJin,zhiJin);
    tex =  pHge->Texture_Load("TaiQiu\\pic\\mask1.bmp");
    Vector3  ran = NormaliseVector(Vector3(ranged_random(.1f,2),ranged_random(.1f,2),ranged_random(.1f,2)));
    BALL_Orientation = rotation_matrix (ranged_random(1,800), ran);    // 设置每个球的随机方位

    HTEXTURE  texMask;
    texMask = pHge->Texture_Load(("TaiQiu\\pic\\mask1.bmp"));
    if (texBall==0)
    {   
        texLight = pHge->Texture_Load(("TaiQiu\\pic\\TEX_Specular25.bmp"));   
        texBall = pHge->Texture_Load(("TaiQiu\\pic\\TEX_BALL.bmp"));
        texShadow = pHge->Texture_Create(zhiJin,zhiJin);
        //pSpriteShadow2 = new CHgeSprite (_T("TaiQiu\\pic\\Ball_Shadow.png"));

        pSpriteShadow = new CHgeSprite(texShadow,0,0,zhiJin,zhiJin);
        pSpriteShadow->SetHotSpot(16,16);
        pSpriteShadow->SetColor(0xff000000);

        DWORD *ptex = pHge->Texture_Lock(tex,false);
        DWORD *ptexmask = pHge->Texture_Lock(texMask);
        DWORD *ptexlight = pHge->Texture_Lock(texLight,false);
        DWORD *ptexShadow = pHge->Texture_Lock(texShadow,false);
        for(int i=0;i<zhiJin;i++)
        {
            for(int j=0;j<zhiJin;j++)
            {
                DWORD col = ptexmask[i*zhiJin+j];
                BYTE *p = (BYTE*)&col;
                BYTE c = ((int)*(p)+6*(int)*(p+1)+3*(int)*(p+2))/10;    //设置透明度        
                col = ptex[i*zhiJin+j];
                ptex[i*zhiJin+j] = SETA(ptex[i*zhiJin+j],c);
                //ptexlight[i*zhiJin+j] = SETA(ptexlight[i*zhiJin+j],c);
                col = ptex[i*zhiJin+j];
                //ptex[i*zhiJin+j] = SETG(ptex[i*zhiJin+j],0XFF);//设置绿色?
                    //ptex[i*zhiJin+j] = SETR(ptex[i*zhiJin+j],100);//设置绿色?
                    //ptex[i*zhiJin+j] = SETB(ptex[i*zhiJin+j],165);//设置绿色?
                p =(BYTE*)&ptexShadow[i*zhiJin+j];
                *(p+3)=c/2;
            }
        }
    /*    for(int i=0;i<25;i++)
        {
            for(int j=0;j<25;j++)
            {
                DWORD col = ptexlight[i*25+j];
                BYTE c = 0;
                if(col==0xff000000)
                    c = 0;
                else c = 0xff;
                ptexlight[i*25+j] = SETA(ptexlight[i*25+j],c);
            }
        }*/
        pHge->Texture_Unlock(tex);
        pHge->Texture_Unlock(texMask);
        pHge->Texture_Unlock(texLight);
        pHge->Texture_Unlock(texShadow);   

        light = Vector3(800/2,520/2,-500);

        float X,Y,Z;
        //法线预处理   
        for(int i=0;i<zhiJin;i++)
        {
            for(int j=0;j<zhiJin;j++)
            {
                mapX[j] =0;
                X =i-banJin+0.5;
                Y =j-banJin+0.5;
                ver[j].X =X;//法线
                ver[j].Y =Y;
                Z = banJin*banJin-(X*X + Y*Y);        
                if (Z>=0)
                {
                    ver[j].Z = -sqrtf(Z);
                    mapX[j] = 1;
                }
            }
        }
    }

    DWORD *ptex = pHge->Texture_Lock(tex,false);
    DWORD *ptexmask = pHge->Texture_Lock(texMask);        
    for(int i=0;i<zhiJin;i++)
    {
        for(int j=0;j<zhiJin;j++)
        {
            DWORD col =ptexmask[i*zhiJin+j];
            BYTE*p =(BYTE*)&col;
            BYTE c=((int)*(p)+6*(int)*(p+1)+3*(int)*(p+2))/10;
        /*    DWORD temp = GETA(col);        
            if(col==0xff000000)
                temp = 0;
            else temp = c;*/
            col =ptex[i*zhiJin+j];            
            ptex[i*zhiJin+j] =SETA(ptex[i*zhiJin+j],c);            
            col =ptex[i*zhiJin+j];
        //    ptex[i*zhiJin+j] =SETG(ptex[i*zhiJin+j],0xFF);        
            //    ptex[i*zhiJin+j] = SETR(ptex[i*zhiJin+j],100);//设置绿色?
            //    ptex[i*zhiJin+j] = SETB(ptex[i*zhiJin+j],165);//设置绿色?
        }
    }
    pHge->Texture_Unlock(tex);
    pHge->Texture_Unlock(texMask);
    pHge->Texture_Free(texMask);

    pSprite = new CHgeSprite(tex,0,0,zhiJin,zhiJin);
    pSprite->SetHotSpot(banJin,banJin);   
    pSprite->SetZ(0.6f);

    pSpriteLight = new CHgeSprite(texLight,0,0,25,25);
    pSpriteLight->SetHotSpot(banjin,banjin);   
    pSpriteLight->SetZ(0.6f);

    //int height = m_GraphHelper->Texture_GetHeight(tex,true);
    //CString str;
    //str.Format("zhiJin=%d,texHeight=%d,spriteHeight=%d,spriteWidth=%d",zhiJin,height,pSprite->GetHeight(),pSprite->GetHeight());
    //AfxMessageBox(str);
}


void CBallRender::Moveball()
{
    if    (BALL_bEnable==false) return;

    if (BALL_V.X==0&&BALL_V.Y==0&&BALL_V.Z==0
        &&BALL_Vang.X==0&&BALL_Vang.Y==0&&BALL_Vang.Z==0)
    {
        return ;
    }
    else
    {
        BALL_bIsRedraw =true;
    }

    Vector3 tool_ver, tra_ver, Normaliz;
    tra_ver = cross(BALL_Vang, Vector3(0, 0, banjin));
    tool_ver = (BALL_V+tra_ver) ;   
   
    if ((tra_ver.mag() < SENROU_FRICTION) && (Vector3(BALL_V.X, BALL_V.Y).mag() < SENROU_FRICTION))
    {
        BALL_State = 3;   //水平方向静止 灰
        BALL_V = Vector3(0, 0, 0);
        BALL_Vang.X = 0;     BALL_Vang.Y = 0 ;
    }  
    else if (tool_ver.mag() < SENSTA_FRICTION)
    {
        TRACE("滚动摩擦");
        BALL_State = 2;   //滚动摩擦 绿
        Normaliz = NormaliseVector(BALL_V);
        apply_impulse (Normaliz*(-M * Rou_friction), Ball_Pos);
        BALL_Vang.X = BALL_V.Y / banjin;
        BALL_Vang.Y = -BALL_V.X / banjin;
    }
    else
    {
        TRACE("滑动摩擦");
        BALL_State = 1 ;//滑动摩擦 红
        Normaliz = NormaliseVector(tool_ver);   
        apply_impulse ( Normaliz*(-M * Sta_friction ), (Ball_Pos+Vector3(0, 0, banjin))    );
    }

    //旋切摩擦
    if (fabs(BALL_Vang.Z) < Xuan_friction)
        BALL_Vang.Z = 0 ;
    else
    {
        if (BALL_Vang.Z > 0)
            BALL_Vang.Z  -=  Xuan_friction ;
        else
            BALL_Vang.Z +=  Xuan_friction;
    }

    if ((BALL_State == 3) && (BALL_Vang.Z == 0))  BALL_State = 4; //完全静止 黑

    Ball_Pos+= (BALL_V* dt);
    Matrix3 Mtmp ;
     Vector3 & rot = BALL_Vang;

    Mtmp=dt * Matrix3(      0,  rot[2], -rot[1],
                      -rot[2],       0,  rot[0],
                       rot[1], -rot[0],       0 ) * BALL_Orientation;
   
    BALL_Orientation += Mtmp;
    BALL_Orientation.orthonormalise();
}

void CBallRender::Move()
{        
    Moveball();        
}
void CBallRender::UpdateBallTex()
{
    HGE* pHge = NULL;
    if (NULL != m_pIGameFrameView)
        pHge = m_pIGameFrameView->GetHge();

    ASSERT(NULL != pHge);
    if (NULL == pHge) return;

    int i,j;
    float IncX , IncY,lng;
    float X , Y , XYmag2;


    IncX = Ball_Pos.X - (INT)Ball_Pos.X;
    IncY = Ball_Pos.Y  - (INT)Ball_Pos.Y;
    if(IncX!=0||IncY!=0)
    {
        int i=0;
        i++;
    }

    //    法线预处理
    for (i=0;i<33;i++)
    {
        for (j=0;j<33;j++)
        {


            mapX[j] = 0 ;
            X =  i - IncX - (banJin-0.5);
            Y = j - IncY - (banJin-0.5);
            Pos[j].X =X ;
            Pos[j].Y =Y ;

            XYmag2=(X*X + Y*Y);
            lng = banJin*banJin - XYmag2 ;
            if (lng > 0    )
            {
                Pos[j].Z = -sqrtf(lng)    ;
                mapX[j] = 1 ;
            }


            //    边缘反锯齿
            lng = banJin - sqrtf(XYmag2) ;
            if (fabs(lng) < 0.4714f)
                CALEFJC (X, Y, i, j );
        }
    }
    //'计算漫反射
    Vector3 lightdir;
    lightdir = light-Ball_Pos;
    lightdir.normalise();

    for (i=0;i<zhiJin+1;i++)
    {
        for (j=0;j<zhiJin+1;j++)
        {

            if (mapX[j] != 0)
            {
                diff[j] = dot(Pos[j], lightdir) / banJin    ;
                if (diff[j] < 0) diff[j] = 0 ;
                diff[j] = 0.3 + 0.7 * diff[j]    ;
            }

        }
    }

    Matrix3 Mtmp ;
    Mtmp= transpose (BALL_Orientation);  //反矩阵

    Vector3 pos;
    DWORD *ptex = pHge->Texture_Lock(tex,false);
    DWORD *ptexBall = pHge->Texture_Lock(texBall);  
   
    int tempNo = BallNo-14;
    int offx =BallNo%4;
    int offy =BallNo/4;
    int mPitch =64*16;
    DWORD * inoff =ptexBall+offy*64*4*64+offx*64;   
    float tempNum = 31.5;
    int ballDraw_X, ballDraw_Y;
    ballDraw_X = (INT)Ball_Pos.X - banJin;
    ballDraw_Y = (INT)Ball_Pos.Y - banJin; // '左下角坐标
    for(int i=0;i<zhiJin;i++)
    {
        for(int j=0;j<zhiJin;j++)
        {
            //转换点
            pos= Mtmp *Pos[j];            
            if (mapX[j]!=0)
            {
                if (pos.Z >0)
                {   
                    BYTE ca =GETA(ptex[j*zhiJin+i]);
                    GetFitColor((tempNum-pos.X),(tempNum+pos.Y),inoff,mPitch,&ptex[j*zhiJin+i]);
                    ptex[j*zhiJin+i] =SETA(ptex[j*zhiJin+i],ca);
                }
                else
                {
                    BYTE ca =GETA(ptex[j*zhiJin+i]);                    
                    GetFitColor((tempNum-pos.X),(tempNum-pos.Y),inoff,mPitch,&ptex[j*zhiJin+i]);
                    ptex[j*zhiJin+i] =SETA(ptex[j*zhiJin+i],ca);
                }

                MIX_multself1((BYTE*)&ptex[j*zhiJin+i],(INT)(diff[j]*255));  //  '漫反射混合

               
                BYTE*P= ((BYTE*)&ptex[j*zhiJin+i]);

                if (mapX[j] == 2)
                {
                    //与地面混合
                    MIX_APLC((BYTE*)&ptex[j*zhiJin+i],P, FJC[j]);
                }
            }            
        }
    }
    pHge->Texture_Unlock(tex);
    pHge->Texture_Unlock(texBall);
}


void CBallRender::Render()
{   
    if (BALL_bEnable==false) {
        return ;
    }
    if (BALL_bIsInit==false)
    {
        UpdateBallTex();
        BALL_bIsInit =true;
    }
    else if (BALL_bIsRedraw) {
        UpdateBallTex();
        BALL_bIsRedraw =false;
    }
    float PntX , PntY;
    float H, dx, dy;
    H = -light.Z;
    dx = Ball_Pos.X - light.X;
    dy = Ball_Pos.Y - light.Y;
    PntX = dx * (5 + H) / H + light.X;
    PntY = dy * (5 + H) / H + light.Y;
   
    //画阴影
    pSpriteShadow->SetBlendMode(BLEND_DEFAULT);
    pSpriteShadow->SetTexture(texShadow);
    //pSpriteShadow->Render(PntX+offsetx,PntY+offsety);
    pSpriteShadow->SetColor(ARGB(255,255,255,255));
    pSpriteShadow->RenderEx(PntX+offsetx,PntY+offsety,0,0.9,0.9);
    pSpriteShadow->SetColor(ARGB(125,0,0,0));
    pSpriteShadow->RenderEx(Ball_Pos.X+offsetx,Ball_Pos.Y+offsety,0,0.88,0.88);


    //画球
    pSprite->SetBlendMode(BLEND_DEFAULT);
    pSprite->SetTexture(tex);
//    pSprite->Render(Ball_Pos.X+offsetx,Ball_Pos.Y+offsety);
    pSprite->RenderEx(Ball_Pos.X+offsetx,Ball_Pos.Y+offsety,0,0.8,0.8);
    //
    ////画高亮
    //pSpriteLight->SetTexture(texLight);
    //pSpriteLight->SetBlendMode(BLEND_MODE_COLORMUL );   
    //pSpriteLight->Render(Ball_Pos.X+offsetx,Ball_Pos.Y+offsety);
     

}
void CBallRender::RenderEx()
{   
    if (BALL_bEnable==false) {
        return ;
    }
    if (BALL_bIsInit==false)
    {
        UpdateBallTex();
        BALL_bIsInit =true;
    }
    else if (BALL_bIsRedraw) {
        UpdateBallTex();
        BALL_bIsRedraw =false;
    }
    float PntX , PntY;
    float H, dx, dy;
    H = -light.Z;
    dx = Ball_Pos.X - light.X;
    dy = Ball_Pos.Y - light.Y;
    PntX = dx * (banJin + H) / H + light.X;
    PntY = dy * (banJin + H) / H + light.Y;
    ////
    ////画阴影
    //pSpriteShadow->SetBlendMode(BLEND_MODE_DEFAULT);
    //pSpriteShadow->SetTexture(texShadow);
    //pSpriteShadow->Render(PntX+offsetx,PntY+offsety);
    //画球
    pSprite->SetBlendMode(BLEND_DEFAULT);
    pSprite->SetTexture(tex);
    pSprite->RenderEx(Ball_Pos.X+offsetx,Ball_Pos.Y+offsety,3.1415926,0.8,0.8);
////    画高亮
    //pSprite->SetTexture(texLight);
    //pSprite->SetBlendMode(BLEND_MODE_COLORMUL );   
    //pSprite->Render(Ball_Pos.X+offsetx,Ball_Pos.Y+offsety);

}

bool  CBallRender:: IsStop()
{
    return BALL_State == 4;
}
bool  CBallRender:: MouseMoveIn(float x,float y)
{
    x = x - offsetx;
    y = y - offsety;
    float xLength = x - this->Ball_Pos.X;
    float yLength = y - Ball_Pos.Y;
    float length = sqrt(xLength*xLength + yLength*yLength);
    return length<banjin-1;
}

void  CBallRender::CALEFJC    (FLOAT Xf, FLOAT Yf ,INT Ni ,INT Nj)
{
    int i,j;
    Vector3 Toolver(0,0,0);
    FLOAT Xs, Ys;
    int Numin=0;// '落在里面

    for (i=0;i<3;i++)
    {
        for (j=0;j<3;j++)
        {

            Xs = Xf + (FLOAT)i *0.33333f  - 0.33333f    ;
            Ys = Yf + (FLOAT)j * 0.33333f - 0.33333f    ;
            if (sqrtf(Xs* Xs + Ys*Ys) < banJin)
            {
                Toolver += CALEPOS(Xs, Ys);
                Numin = Numin + 1 ;
            }

        }
    }

    if (Numin > 0)
    {
        Toolver.normalise();
        Toolver *= banJin ;
        Pos[Ni][Nj] = Toolver ;
        mapX[Ni][Nj] = 2    ;
        FJC[Ni][Nj] = Numin;
    }

}

void CBallRender::SetView(IUnknownEx* pIUnknownEx)
{
     if (NULL != pIUnknownEx)
     {
         m_pIGameFrameView = /*GET_OBJECTPTR_INTERFACE*/QUERY_OBJECT_PTR_INTERFACE(pIUnknownEx, ITQView);
         ASSERT(m_pIGameFrameView != NULL);
     }
}
下载地址(回复可见):
游客,如果您要查看本帖隐藏内容请回复


解压密码默认为:www.woaidaima.com


分享到:  QQ好友和群QQ好友和群
收藏收藏
回复

使用道具 举报

该用户从未签到

0

主题

186

帖子

192

积分

邀请会员

积分
192
6#
发表于 2023-2-24 10:26:54 | 只看该作者
谢谢谢谢谢谢谢
回复 支持 反对

使用道具 举报

该用户从未签到

3

主题

61

帖子

208

积分

邀请会员

积分
208
5#
发表于 2022-12-8 14:22:35 | 只看该作者
谢谢谢谢谢谢谢谢谢谢谢谢谢谢谢谢谢谢谢
刚刚回家看看
回复 支持 反对

使用道具 举报

该用户从未签到

0

主题

48

帖子

49

积分

邀请会员

积分
49
地板
发表于 2022-4-14 11:06:30 | 只看该作者
感謝分享
回复

使用道具 举报

该用户从未签到

0

主题

127

帖子

1415

积分

邀请会员

积分
1415
QQ
板凳
发表于 2019-6-26 02:49:13 | 只看该作者
既然你诚信诚意的推荐了,那我就回复了!
阿斯
回复 支持 反对

使用道具 举报

该用户从未签到

0

主题

147

帖子

173

积分

新手上路

Rank: 1

积分
173
沙发
发表于 2019-4-26 15:08:23 | 只看该作者
既然你诚信诚意的推荐了,那我就回复了!
回复 支持 反对

使用道具 举报

快速回复高级模式
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表