Where communities thrive


  • Join over 1.5M+ people
  • Join over 100K+ communities
  • Free without limits
  • Create your own community
People
Repo info
Activity
vblanco20-1
@vblanco20-1
@Sheepyhead i solved that problem on my own with some evil hackery
struct ContextWrapper{
pub ctx: *mut Context
}

impl ContextWrapper{
    fn get(&self) -> &mut Context
    {
        unsafe{
            &mut *self.ctx
        }
    }
}
unsafe impl Sync for ContextWrapper{}
unsafe impl Send for ContextWrapper{}
Dmitriy
@dpogretskiy
and my demo is a bit messy, but there's your answer, https://github.com/dpogretskiy/specs-ggez-showcase/blob/master/src/game.rs#L138
you just run the rendering system outside of specs dispatcher and can put whatever you want into it
also you run in whatever thread you want and it works
Troels Jessen
@Sheepyhead
Oh my god, thank you so much Dmitriy, I can't believe I didn't think of that! I'm gonna go implement that right now!
vblanco20-1
@vblanco20-1
in mine, i run it as a specs system
not outside of it
Troels Jessen
@Sheepyhead
I actually prefer it the other way around, it also saves me from having to separate my draw system so I can run it in the render loop
Dmitriy presented precisely what I was trying to do, but I appreciate your input too!
Troels Jessen
@Sheepyhead
Hmm seems like there are some things I can't really understand out of your code @dpogretskiy , I'm unsure where the run_now function on RenderingSystem comes from, and I'm not sure what World.res is supposed to be
Troels Jessen
@Sheepyhead
Also I've failed harshly in trying to get the project to compile so I can't use RLS to see where it's coming from either :sweat_smile:
Dmitriy
@dpogretskiy
It's a pretty dated code, that used old versions of specs and ggez
that's probably what you want, it runs the system, but now it uses &mut World instead of res whatever it was at the date apparently
Troels Jessen
@Sheepyhead
I actually found that trait but for some reason RLS was still complaining. I got it to work now though! Thank you so much!! :D
Mark
@markvandieren

Hey, does anyone have experience using the saveload module with bincode? First issue: the Serializer struct that needs to be passed to the serialize function is private. I bypassed that by cloning the repository and making it public, but that's not really a nice solution.

Second issue, I get this error: Bincode can only encode sequences and maps that have a knowable size ahead of time

So bincode can't be used with saveload?

Jochen Görtler
@grtlr
Hi everybody! I'm new to specs and have a question: If I create entities with create_iter, is it guaranteed that these entities will be joined later on in the same order as they are created? Thank you very much!
Dmitriy
@dpogretskiy
absolutely no @grtlr :)
altho you can just put them in the sorted set as you go through, and then do your thing, not perfect, but if it will cause issues you'd probably find a 'less naive' solution
Jochen Görtler
@grtlr
@dpogretskiy Thank you for the quick reply, I'll go ahead and do it that way!
Arnold Loubriat
@DataTriny

Hello there, I'm coming back to creating a text-based game using Specs and I can't figure out a good way to handle input. In my mind, a command (like attack) is composed of selectors (CreatureSelector and WeaponSelector in this case) that are responsible for associating a slice of input to something in the world. Those selectors must run in a given order defined by each command, and they all have very different needs in terms of storage reading. The most practical way I've found is to create custom system traits and my own dispatcher like so:

pub trait Command<'a> {
    type Data: SystemData<'a>;
    fn get_selectors(&self) -> &[Box<dyn ExecuteSelector>];
    fn run(&self, args: Vec<Selectable>, game: &Game, data: Self::Data) -> CommandResult;
    fn setup(&self, world: &mut World);
}
pub trait ExecuteCommand<'a> {
    fn get_selectors(&self) -> &[Box<Selector>];
    fn run(&self, args: Vec<Selectable>, game: &Game, world: &'a World) -> CommandResult;
    fn setup(&self, world: &mut World);
}
impl<'a, T> ExecuteCommand<'a> for T
    where T: Command<'a>
{ ... }
pub trait Selector<'a> {
    type Data: SystemData<'a>;
    fn parse(&self, input: &str, game: &Game, data: Self::Data) -> ParsingResult<(usize, Selectable)>;
    fn setup(&self, res: &mut World);
}
pub trait ExecuteSelector<'a> { ... }
impl<'a, T> ExecuteSelector<'a> for T
    where T: Selector<'a>
{ ... }
pub struct CommandDispatcher<'a> {
    commands: Vec<Box<dyn ExecuteCommand<'a>>>
}
impl<'a> CommandDispatcher<'a> {
    pub fn dispatch(&self, input: &str, game: &Game<'a>, world: &'a mut World) -> Result<(), DispatchingError> {
        // Find which command to trigger based on the start of the input and strip the matching part.
        // For each command's selector, run it and remove the matching part of the input. Add the results to a list of arguments.
        // Run the command and pass it the list of arguments.
    }
}

Of course this is a simplified version. But I think that this defeats the purpose of an ECS...
Does anyone have an idea on how to put it differently? Many thanks in advance!

Bastczuak
@Bastczuak
Hello people! Can I access joined storages by index? For example like this:
    let position_storage = self.ecs.read_storage::<Position>();
    let rotation_storage = self.ecs.read_storage::<Rotation>();
    let something = (&position_storage, &rotation_storage).join().collect()[0];
Oscar Aguinagalde
@OAguinagalde

Hey people, I'm following the specs book, and I'm finding plenty of problems. Even if I just copy and paste the examples to follow, not even the first ones compile. Most of them seem to be related to how the modules are imported, but there are plenty of things I don't get.
For instance, while doing this

use specs::{Component, ReadStorage, System, VecStorage, World, RunNow, WorldExt, Join};

// a bunch of code...

let entity = world
        .create_entity()
        .with(Position { x: 4.0, y: 7.0 }) // ERROR: no method named `with` found for type `specs::world::EntityBuilder<'_>`
        .build();

I encounter that problem.
Which apparently gets fixed when "using" specs::world::* (not World)
I also needed to add WorldExt to the mix, because World::new() was not recognized. That is also not in the example of the book.

Also, even after fixing these 2 problems, I still cant compile:

let mut hello_world = HelloWorld;
hello_world.run_now(&world.res);
// ERROR: no field `res` on type `shred::world::World`

Here is the system pretty much the same as it is in the book:

struct HelloWorld;

impl<'a> System<'a> for HelloWorld {
    type SystemData = ReadStorage<'a, Position>;

    fn run(&mut self, position: Self::SystemData) {

        for position in position.join() {
            println!("Hello, {:?}", &position);
        }
    }
}

Not sure how I should make the "use" statements or what am I doing wrong. In any case, the book, while very easy to follow, it's examples straight up don't work, so that's something to keep in mind

Jean Manguy
@jeanmanguy
Hello there, is it possible to add and/or remove components to an entity in a system to later it's functioning in an ulterior system? What's the proper way to achieve something similar is it's not possible?
Christopher Sabater Cordero
@cs-cordero

Is there a way to specify a generic lifetime on a system?

struct Foo<'a> {
    something: Bar<'a>
}

impl<'a> System<'a> for Foo<'a> {
    ...
}

I get errors when I try to move the system into a DispatcherBuilder

cobalthex
@cobalthex
is there a way to remove a component from an entity while iterating? or remove a component from all entities matched in a join iter
Oscar
@tcmal

are there any examples of using a generic resource in a system? for example:

trait IsALevelThing: Send + Sync {}
struct LevelThing;

struct Renderer<T> {
    _a: PhantomData<T>,
}

impl<'a, T: 'a + IsALevelThing> System<'a> for Renderer<T> {
    type SystemData = Read<'a, T, PanicHandler>;

    fn run(&mut self, data: Self::SystemData) {
        // TODO
    }
}

seems like it should work but it gives the parameter typeTmay not live long enough, consider adding an explicit lifetime bound T: 'static. not sure why this is since from what i understand it just needs to live for 'a