Skip to content
Snippets Groups Projects
tileset.rs 10.4 KiB
Newer Older
alexdevteam's avatar
alexdevteam committed
use std::collections::HashMap;
use std::io::Read;
alexdevteam's avatar
alexdevteam committed
use std::path::{Path, PathBuf};
alexdevteam's avatar
alexdevteam committed

use xml::attribute::OwnedAttribute;
use xml::reader::XmlEvent;
use xml::EventReader;

use crate::error::TiledError;
use crate::image::Image;
alexdevteam's avatar
alexdevteam committed
use crate::properties::{parse_properties, Properties};
alexdevteam's avatar
alexdevteam committed
use crate::tile::Tile;
use crate::{util::*, Gid};
alexdevteam's avatar
alexdevteam committed

/// A tileset, usually the tilesheet image.
#[derive(Debug, PartialEq, Clone)]
pub struct Tileset {
    pub name: String,
    pub tile_width: u32,
    pub tile_height: u32,
    pub spacing: u32,
    pub margin: u32,
    pub tilecount: u32,
alexdevteam's avatar
alexdevteam committed

    /// A tileset can either:
    /// * have a single spritesheet `image` in `tileset` ("regular" tileset);
    /// * have zero images in `tileset` and one `image` per `tile` ("image collection" tileset).
    ///
alexdevteam's avatar
alexdevteam committed
    /// --------
    /// - Source: [tiled issue #2117](https://github.com/mapeditor/tiled/issues/2117)
    /// - Source: [`columns` documentation](https://doc.mapeditor.org/en/stable/reference/tmx-map-format/#tileset)
    pub image: Option<Image>,

    /// All the tiles present in this tileset, indexed by their local IDs.
    pub tiles: HashMap<u32, Tile>,

    /// The custom properties of the tileset.
alexdevteam's avatar
alexdevteam committed
    pub properties: Properties,
alexdevteam's avatar
alexdevteam committed

    /// Where this tileset was loaded from.
    /// If fully embedded, this will return `None`.
alexdevteam's avatar
alexdevteam committed
    pub source: Option<PathBuf>,
alexdevteam's avatar
alexdevteam committed
}

pub(crate) enum EmbeddedParseResultType {
    ExternalReference { tileset_path: PathBuf },
    Embedded { tileset: Tileset },
}

pub(crate) struct EmbeddedParseResult {
    pub first_gid: Gid,
    pub result_type: EmbeddedParseResultType,
}

alexdevteam's avatar
alexdevteam committed
/// Internal structure for holding mid-parse information.
struct TilesetProperties {
    spacing: Option<u32>,
    margin: Option<u32>,
    tilecount: u32,
alexdevteam's avatar
alexdevteam committed
    columns: Option<u32>,
    name: String,
    tile_width: u32,
    tile_height: u32,
    /// The path all non-absolute paths are relative to.
alexdevteam's avatar
alexdevteam committed
    path_relative_to: Option<PathBuf>,
    source: Option<PathBuf>,
}

alexdevteam's avatar
alexdevteam committed
impl Tileset {
alexdevteam's avatar
alexdevteam committed
    /// Parse a buffer hopefully containing the contents of a Tiled tileset.
    pub fn parse<R: Read>(reader: R) -> Result<Self, TiledError> {
        Tileset::new_external(reader, None)
alexdevteam's avatar
alexdevteam committed
    }

    /// Parse a buffer hopefully containing the contents of a Tiled tileset.
    pub fn parse_with_path<R: Read>(reader: R, path: impl AsRef<Path>) -> Result<Self, TiledError> {
        Tileset::new_external(reader, Some(path.as_ref()))
alexdevteam's avatar
alexdevteam committed
    }

    pub fn get_tile(&self, id: u32) -> Option<&Tile> {
        self.tiles.get(&id)
alexdevteam's avatar
alexdevteam committed
    }
alexdevteam's avatar
alexdevteam committed

impl Tileset {
    pub(crate) fn new_external<R: Read>(file: R, path: Option<&Path>) -> Result<Self, TiledError> {
alexdevteam's avatar
alexdevteam committed
        let mut tileset_parser = EventReader::new(file);
        loop {
            match tileset_parser
                .next()
                .map_err(TiledError::XmlDecodingError)?
            {
                XmlEvent::StartElement {
                    name, attributes, ..
                } => {
                    if name.local_name == "tileset" {
                        return Self::parse_external_tileset(
                            &mut tileset_parser.into_iter(),
alexdevteam's avatar
alexdevteam committed
                            &attributes,
                            path,
                        );
                    }
                }
                XmlEvent::EndDocument => {
                    return Err(TiledError::PrematureEnd(
                        "Tileset Document ended before map was parsed".to_string(),
                    ))
                }
                _ => {}
            }
        }
    }

    pub(crate) fn parse_xml_in_map(
        parser: &mut impl Iterator<Item = XmlEventResult>,
        attrs: Vec<OwnedAttribute>,
        map_path: &Path,
    ) -> Result<EmbeddedParseResult, TiledError> {
        let path_relative_to = map_path.parent();
        Tileset::parse_xml_embedded(parser, &attrs, path_relative_to).or_else(|err| {
            if matches!(err, TiledError::MalformedAttributes(_)) {
                Tileset::parse_xml_reference(&attrs, path_relative_to)
            } else {
                Err(err)
            }
        })
    }

    /// Returns both the tileset and its first gid in the corresponding map.
    fn parse_xml_embedded(
        parser: &mut impl Iterator<Item = XmlEventResult>,
alexdevteam's avatar
alexdevteam committed
        attrs: &Vec<OwnedAttribute>,
alexdevteam's avatar
alexdevteam committed
        path_relative_to: Option<&Path>,
    ) -> Result<EmbeddedParseResult, TiledError> {
        let ((spacing, margin, columns, name), (tilecount, first_gid, tile_width, tile_height)) = get_attrs!(
alexdevteam's avatar
alexdevteam committed
           attrs,
           optionals: [
                ("spacing", spacing, |v:String| v.parse().ok()),
                ("margin", margin, |v:String| v.parse().ok()),
                ("columns", columns, |v:String| v.parse().ok()),
                ("name", name, |v| Some(v)),
alexdevteam's avatar
alexdevteam committed
            ],
           required: [
                ("tilecount", tilecount, |v:String| v.parse().ok()),
                ("firstgid", first_gid, |v:String| v.parse().ok().map(|n| Gid(n))),
alexdevteam's avatar
alexdevteam committed
                ("tilewidth", width, |v:String| v.parse().ok()),
                ("tileheight", height, |v:String| v.parse().ok()),
            ],
            TiledError::MalformedAttributes("tileset must have a firstgid, name tile width and height with correct types".to_string())
        );

alexdevteam's avatar
alexdevteam committed
        Self::finish_parsing_xml(
            parser,
            TilesetProperties {
                spacing,
                margin,
                name: name.unwrap_or_default(),
alexdevteam's avatar
alexdevteam committed
                path_relative_to: path_relative_to.map(Path::to_owned),
                columns,
                tilecount,
                tile_height,
                tile_width,
                source: None,
        .map(|tileset| EmbeddedParseResult {
            first_gid,
            result_type: EmbeddedParseResultType::Embedded { tileset },
        })
alexdevteam's avatar
alexdevteam committed
    }

alexdevteam's avatar
alexdevteam committed
    fn parse_xml_reference(
alexdevteam's avatar
alexdevteam committed
        attrs: &Vec<OwnedAttribute>,
alexdevteam's avatar
alexdevteam committed
        path_relative_to: Option<&Path>,
    ) -> Result<EmbeddedParseResult, TiledError> {
alexdevteam's avatar
alexdevteam committed
        let ((), (first_gid, source)) = get_attrs!(
            attrs,
            optionals: [],
            required: [
                ("firstgid", first_gid, |v:String| v.parse().ok().map(|n| Gid(n))),
alexdevteam's avatar
alexdevteam committed
                ("source", name, |v| Some(v)),
            ],
alexdevteam's avatar
alexdevteam committed
            TiledError::MalformedAttributes("Tileset reference must have a firstgid and source with correct types".to_string())
alexdevteam's avatar
alexdevteam committed
        );

alexdevteam's avatar
alexdevteam committed
        let tileset_path = path_relative_to
            .ok_or(TiledError::SourceRequired {
                object_to_parse: "Tileset".to_string(),
            })?
            .join(source);

        Ok(EmbeddedParseResult {
            first_gid,
            result_type: EmbeddedParseResultType::ExternalReference { tileset_path },
        })
alexdevteam's avatar
alexdevteam committed
    }

    fn parse_external_tileset(
        parser: &mut impl Iterator<Item = XmlEventResult>,
alexdevteam's avatar
alexdevteam committed
        attrs: &Vec<OwnedAttribute>,
alexdevteam's avatar
alexdevteam committed
        path: Option<&Path>,
alexdevteam's avatar
alexdevteam committed
    ) -> Result<Tileset, TiledError> {
        let ((spacing, margin, columns, name), (tilecount, tile_width, tile_height)) = get_attrs!(
alexdevteam's avatar
alexdevteam committed
            attrs,
            optionals: [
                ("spacing", spacing, |v:String| v.parse().ok()),
                ("margin", margin, |v:String| v.parse().ok()),
                ("columns", columns, |v:String| v.parse().ok()),
                ("name", name, |v| Some(v)),
alexdevteam's avatar
alexdevteam committed
            ],
            required: [
                ("tilecount", tilecount, |v:String| v.parse().ok()),
alexdevteam's avatar
alexdevteam committed
                ("tilewidth", width, |v:String| v.parse().ok()),
                ("tileheight", height, |v:String| v.parse().ok()),
            ],
            TiledError::MalformedAttributes("tileset must have a name, tile width and height with correct types".to_string())
alexdevteam's avatar
alexdevteam committed
        );

alexdevteam's avatar
alexdevteam committed
        let source_path = path.and_then(|p| p.parent().map(Path::to_owned));
alexdevteam's avatar
alexdevteam committed

alexdevteam's avatar
alexdevteam committed
        Self::finish_parsing_xml(
            parser,
            TilesetProperties {
                spacing,
                margin,
                name: name.unwrap_or_default(),
alexdevteam's avatar
alexdevteam committed
                path_relative_to: source_path,
                columns,
                tilecount,
                tile_height,
                tile_width,
                source: path.map(Path::to_owned),
            },
        )
    }

    fn finish_parsing_xml(
        parser: &mut impl Iterator<Item = XmlEventResult>,
alexdevteam's avatar
alexdevteam committed
        prop: TilesetProperties,
    ) -> Result<Tileset, TiledError> {
        let mut tiles = HashMap::with_capacity(prop.tilecount as usize);
alexdevteam's avatar
alexdevteam committed
        let mut properties = HashMap::new();
alexdevteam's avatar
alexdevteam committed
        parse_tag!(parser, "tileset", {
            "image" => |attrs| {
alexdevteam's avatar
alexdevteam committed
                image = Some(Image::new(parser, attrs, prop.path_relative_to.as_ref().ok_or(TiledError::SourceRequired{object_to_parse: "Image".to_string()})?)?);
alexdevteam's avatar
alexdevteam committed
                Ok(())
            },
            "properties" => |_| {
                properties = parse_properties(parser)?;
                Ok(())
            },
                let (id, tile) = Tile::new(parser, attrs, prop.path_relative_to.as_ref().and_then(|p| Some(p.as_path())))?;
                tiles.insert(id, tile);
alexdevteam's avatar
alexdevteam committed
        });

        // A tileset is considered an image collection tileset if there is no image attribute (because its tiles do).
        let is_image_collection_tileset = image.is_none();

        if !is_image_collection_tileset {
            for tile_id in 0..prop.tilecount {
                tiles.entry(tile_id).or_default();
            }
        }
        let margin = prop.margin.unwrap_or(0);
        let spacing = prop.spacing.unwrap_or(0);
alexdevteam's avatar
alexdevteam committed
        let columns = prop
            .columns
            .map(Ok)
            .unwrap_or_else(|| Self::calculate_columns(&image, prop.tile_width, margin, spacing))?;
alexdevteam's avatar
alexdevteam committed
        Ok(Tileset {
alexdevteam's avatar
alexdevteam committed
            name: prop.name,
            tile_width: prop.tile_width,
            tile_height: prop.tile_height,
            spacing,
            margin,
alexdevteam's avatar
alexdevteam committed
            tilecount: prop.tilecount,
alexdevteam's avatar
alexdevteam committed
            tiles,
alexdevteam's avatar
alexdevteam committed
            properties,
alexdevteam's avatar
alexdevteam committed
            source: prop.source,
alexdevteam's avatar
alexdevteam committed
        })
    }
alexdevteam's avatar
alexdevteam committed

    fn calculate_columns(
        image: &Option<Image>,
        tile_width: u32,
        margin: u32,
        spacing: u32,
    ) -> Result<u32, TiledError> {
        image
            .as_ref()
            .ok_or(TiledError::MalformedAttributes(
                "No <image> nor columns attribute in <tileset>".to_string(),
            ))
            .and_then(|image| Ok((image.width as u32 - margin + spacing) / (tile_width + spacing)))
    }
alexdevteam's avatar
alexdevteam committed
}