如何将physicsBody添加到JSTileMap生成的切片并将其更改为一个physicsBody

时间:2016-11-10 13:59:38

标签: ios objective-c sprite-kit jstilemap

我使用JSTileMap绘制一个级别。我改变了一点,将SKPhysicsBody添加到每个瓷砖,但有时当我对主角施加一些冲动并且他击中墙/地面/天花板时他表现得很奇怪。他从表面反射的方式与物理原理相反。我认为这是因为玩家遇到两个物理体(例如地面的两个物理体)连接的点。 SKPhysicsBody类提供了一种方法,用于从不同的物理体+(SKPhysicsBody *)bodyWithBodies:(NSArray *)体创建一个物理体;我可以使用这种方法从类似于物理体的物体中创建一个物理体吗?

这是JSTileMap中添加物理实体的方法:

+(id) layerWithTilesetInfo:(NSArray*)tilesets layerInfo:(TMXLayerInfo*)layerInfo mapInfo:(JSTileMap*)mapInfo
{
    TMXLayer* layer = [TMXLayer node];
  layer.map = mapInfo;

    layer.tilesByColumnRow = [NSMutableDictionary dictionary];
    // basic properties from layerInfo
    layer.layerInfo = layerInfo;
    layer.layerInfo.layer = layer;
    layer.mapTileSize = mapInfo.tileSize;
    layer.alpha = layerInfo.opacity;
    layer.position = layerInfo.offset;

    // recalc the offset if we are isometriic
    if (mapInfo.orientation == OrientationStyle_Isometric)
    {
        layer.position = CGPointMake((layer.mapTileSize.width / 2.0) * (layer.position.x - layer.position.y),
                                     (layer.mapTileSize.height / 2.0) * (-layer.position.x - layer.position.y));
    }

    NSMutableDictionary* layerNodes = [NSMutableDictionary dictionaryWithCapacity:tilesets.count];

    //MY CODE
    NSMutableArray *arrayOfSprites = [[NSMutableArray alloc] init];
    SKNode *theSprite;
    //----||----

    // loop through the tiles
    for (NSInteger col = 0; col < layerInfo.layerGridSize.width; col++)
    {
        for (NSInteger row = 0; row < layerInfo.layerGridSize.height; row++)
        {
            // get the gID
            NSInteger gID = layerInfo.tiles[col + (NSInteger)(row * layerInfo.layerGridSize.width)];

            // mask off the flip bits and remember their result.
            bool flipX = (gID & kTileHorizontalFlag) != 0;
            bool flipY = (gID & kTileVerticalFlag) != 0;
            bool flipDiag = (gID & kTileDiagonalFlag) != 0;
            gID = gID & kFlippedMask;

            // skip 0 GIDs
            if (!gID)
                continue;

            // get the tileset for the passed gID.  This will allow us to support multiple tilesets!
            TMXTilesetInfo* tilesetInfo = [mapInfo tilesetInfoForGid:gID];
            [layer.tileInfo addObject:tilesetInfo];

            if (tilesetInfo)    // should never be nil?
            {
                SKTexture* texture = [tilesetInfo textureForGid:gID];
                SKSpriteNode* sprite = [SKSpriteNode spriteNodeWithTexture:texture];

                sprite.name = [NSString stringWithFormat:@"%ld",(long)(col + row * layerInfo.layerGridSize.width)];

                // make sure it's in the right position.
                if (mapInfo.orientation == OrientationStyle_Isometric)
                {
                    sprite.position = CGPointMake((layer.mapTileSize.width / 2.0) * (layerInfo.layerGridSize.width + col - row - 1),
                                                  (layer.mapTileSize.height / 2.0) * ((layerInfo.layerGridSize.height * 2 - col - row) - 2) );
                }
                else
                {
                    sprite.position = CGPointMake(col * layer.mapTileSize.width + layer.mapTileSize.width/2.0,
                                                  (mapInfo.mapSize.height * (tilesetInfo.tileSize.height)) - ((row + 1) * layer.mapTileSize.height) + layer.mapTileSize.height/2.0);
                }

                // flip sprites if necessary
                if(flipDiag)
                {
                    if(flipX)
                        sprite.zRotation = -M_PI_2;
                    else if(flipY)
                        sprite.zRotation = M_PI_2;
                }
                else
                {
                    if(flipY)
                        sprite.yScale *= -1;
                    if(flipX)
                        sprite.xScale *= -1;
                }

                // add sprite to correct node for this tileset
                SKNode* layerNode = layerNodes[tilesetInfo.name];
                if (!layerNode) {
                    layerNode = [[SKNode alloc] init];
                    layerNodes[tilesetInfo.name] = layerNode;
                }

                //adding physicsbody to every tile
                //sprite.physicsBody = [SKPhysicsBody bodyWithRectangleOfSize:CGSizeMake(sprite.frame.size.width, sprite.frame.size.height)];
                //sprite.physicsBody.dynamic = NO;
                //sprite.physicsBody.categoryBitMask = mapCategory;

                //[arrayOfSprites addObject:sprite.physicsBody];

                [layerNode addChild:sprite];
                NSUInteger indexes[] = {col, row};
                NSIndexPath *indexPath = [NSIndexPath indexPathWithIndexes:indexes length:2];
                [layer.tilesByColumnRow setObject:sprite forKey:indexPath];

                //MY CODE
                sprite.physicsBody = [SKPhysicsBody bodyWithRectangleOfSize:sprite.frame.size];
                sprite.physicsBody.dynamic = NO;
                [arrayOfSprites addObject:sprite.physicsBody];
                //-----||------

#ifdef DEBUG
//              CGRect textRect = [texture textureRect];
//              NSLog(@"atlasNum %2d (%2d,%2d), gid (%d,%d), rect (%f, %f, %f, %f) sprite.pos (%3.2f,%3.2f) flipx%2d flipy%2d flipDiag%2d", gID+1, row, col, [tilesetInfo rowFromGid:gID], [tilesetInfo colFromGid:gID], textRect.origin.x, textRect.origin.y, textRect.size.width, textRect.size.height, sprite.position.x, sprite.position.y, flipX, flipY, flipDiag);
#endif

            }
        }
    }

    //MY CODE
    NSArray *array = [arrayOfSprites copy];
    theSprite = [SKNode node];
    theSprite.physicsBody = [SKPhysicsBody bodyWithBodies:array];
    theSprite.physicsBody.dynamic = NO;
    theSprite.position = CGPointMake(layer.position.x+16, layer.position.y+16);
    [layer addChild:theSprite];
    //-----||------

    // add nodes for any tilesets that were used in this layer
    for (SKNode* layerNode in layerNodes.allValues) {
        if (layerNode.children.count > 0) {
            [layer addChild:layerNode];
        }
    }

    [layer calculateAccumulatedFrame];

    return layer;
}

将物理主体添加到每个磁贴并将这些物理主体添加到NSMutableArray之后,我将此NSMutableArray的副本分配给NSArray并尝试使用它创建一个物理主体:

//MY CODE
        NSArray *array = [arrayOfSprites copy];
        theSprite = [SKNode node];
        theSprite.physicsBody = [SKPhysicsBody bodyWithBodies:array];
        theSprite.physicsBody.dynamic = NO;
        theSprite.position = CGPointMake(layer.position.x+16, layer.position.y+16);
        [layer addChild:theSprite];
        //-----||------

结果,添加了一个具有一个图块的高度和宽度的物理主体。

1 个答案:

答案 0 :(得分:1)

如果要使用[SKPhysicsBody bodyWithBodies:array],则需要确保数组中的所有实体都与父节点相关。

sprite.physicsBody = [SKPhysicsBody bodyWithRectangleOfSize:sprite.frame.size];表示您的物理体位置相对于精灵节点。您需要相对于父节点的主体。

我知道如何做到的唯一方法是使用center:

sprite.physicsBody = [SKPhysicsBody bodyWithRectangleOfSize:sprite.frame.size center:sprite.position];

这应该将SKPhysicsBody放在sprite添加到父节点的位置。