I am trying to build a psychopy experiment on my Macbook, in which image types will be presented in a fixed order such as:
Intact, Intact, Scrambled…
and for example, within the stimuli type of Intact, there will be 100 images to randomly select from, and in the second Intact stimuli, there will now be 99 images to randomly select from (no image will be repeated), and then in the Scrambled image type, there will be 25 images to select from…
I had completed this on Eprime, in which basically there was a fixed sequential order for type of stimuli, but a random order within those stimuli.
Could anyone help me figure this out in Psychopy please?
Hi, to do this in psychoPy, you generally want to use two loops: an inner loop to control the trials (which will be randomised) and an outer loop to control the blocks (which will be sequential). The technique is described here:
In essence, the outer loop needs a conditions file that contains just three rows, each of which will point to the conditions file to be used for the inner loop for each block of trials, e.g.
i.e. this outer loop cycles through these three conditions sequentially, and the inner loop simply uses the variable $stimulus_file as the name of its conditions file.
That way the first block will have 100 randomised trials, the next 99 randomised trials, and the last, 25 randomised trials.
Thank you for the reply.
So would this outer loop just randomly pick an image out of the 100 images within Intact? How would I ensure that the image randomly selected the first time isn’t the same image randomly selected the second time?
The outer loop doesn’t know anything about your images at all. The only variables it knows about are the names of the three conditions files to be used by the inner loop.
Intact, Intact, Scrambled, Intact, Intact, Scrambled… and so on until it shows about 100 total images.
Everytime an Intact type is shown, it will randomly select out of 80 intact images and everytime a scrambled type is shown, it randomly selects out of 20 scrambled images. However, it can’t pick the same image twice.
I get that there needs to be two loops, but how would I structure the excel file for the sequential loop in that it selects images randomly but avoids images that have already been previously selected?
I read your original post as indicating that there were successive blocks of trials of the same type, rather than a single block of trials of interleaved trial types. A double-loop arrangement won’t help with this. What is probably easiest is a single loop, and with the conditions and image names determined in code rather than in a conditions file.
Insert a code component (from the “custom” component panel) and put something like this in the “begin experiment” tab, to create lists of image names (like intact_24.jpg etc):
# create image lists:
intact_images = [f'intact_image_{i}.jpg' for i in range(100)]
scrambled_images = [f'scrambled_image_{i}.jpg' for i in range(25)]
# randomise their order:
shuffle(intact_images)
shuffle(scrambled_images)
Then in the “Begin routine” tab, put something like this, to figure out what condition this trial is, and to sample an image name without replacement from the appropriate list:
# figure out the condition to use, based on every third trial
# being a "scrambled" one. calculate by checking that the
# remainder is 0 when diving by 3:
if (the_name_of_your_loop.thisN + 1) % 3 == 0:
# trial is a multiple of three, so:
image_type = 'scrambled'
image_file = scrambled_images.pop()
else:
image_type = 'intact'
image_file = intact_images.pop()
# record these in the data:
thisExp.addData('image_type', image_type)
thisExp.addData('image_file', image_file)
Now just put $image_file in the “image” field of your image stimulus component, set to update every repeat. Make sure that the code component is above the image stimulus component, so that this variable is updated in time for the image stimulus to refer to it.
Note that the loop will only run 75 times before an error occurs, as the scrambled_images list will be emptied at that point, but the numbers you give in your posts above jump around a bit.
It’s not that the ‘scrambled’ image will appear every third trial, as I was just using it as an example. The actual test will be more like, for example:
I, I, S, I, I, I, I, I, S, I, I, S, I, I, I, I, S
There is a fixed order to the stimuli type, but it’s not every third trial.
Does this help? Try replacing the code in the “begin routine” tab with this. You can adjust when the scrambled images appear in the sequence by changing the numbers in the first line of the “if” statement.
# Scrambled images are to appear at fixed positions in a
# sequence of stimuli.
If (the_name_of_your_loop.thisN + 1) == (3 or 9 or 12 or 17):
image_type = 'scrambled'
image_file = scrambled_images.pop()
else:
image_type = 'intact'
image_file = intact_images.pop()
# record these in the data:
thisExp.addData('image_type', image_type)
thisExp.addData('image_file', image_file)
Hi, has been a long weekend here in NZ (for Waitangi Day) and I have not been checking this board…
Andrew is on the right track but that syntax won’t quite work. To get multiple or comparisons to work would have to be something a bit clumsy like this:
if (the_name_of_your_loop.thisN + 1) == 3 or (the_name_of_your_loop.thisN + 1) == 9 or … # etc
The more concise approach would be to use in instead:
if (the_name_of_your_loop.thisN + 1) in [3, 9, 12, 17]:
But perhaps the best alternative would be to control the trial type from a conditions file, which you step through sequentially (to maintain your fixed order). Then you wouldn’t have to muck around with trial numbers at all. In that case, it would simply be:
Yes, those lists are still being created in code, separate from the conditions file (i.e. in the code in the “Begin experiment” tab).
What the pop() method of a list does is pull out the next entry from a list, shrinking that list in size by one entry. So that entry can’t be selected again, because the list no longer contains it.
In Python, the .pop() method only applies to a Python list object - you are trying to pop from a file. Python simply won’t know what to do with that.
In this post, we created the two lists you need (intact_images and scrambled_images):
Your files will likely have a different naming convention of course. Amend that code to match the way your files are actually stored, including if they are in a subfolder. The image file names don’t need to have a full absolute path starting at /Users/. If they are in a subfolder of the same folder as where your experiment is, they can be relative paths that start there. e.g.
intact_images = [f'my_image_subfolder_name/Wow_my_INTACT_pics_{i}.jpg' for i in range(100)]
You need to tweak that code to fit - e.g. reduce the number 100 to match however many images you actually have.
Ok I see. Thank you so much for everything.
I think I’m just about done, but one tiny thing is that my images are named so for example ‘avacado.jpg’ or ‘brush.jpg’ rather than numbered. How do I account for that in the code, because I think it is looking through numbers from 1-100 in the images file name. Or is it better to rename these object files from 1 to 100.