Trouble changing default loading screen in Studio

Hi,
I’m not positive I can phrase this question the right way. You know how right after scanning the QR code, the screen will show your 3D content against a black screen for a while until the camera activates? I don’t want that; instead, I’d like for the screen to show a custom image background behind the loading progress.

I found an example project by George Butler with a splash screen and a tap to activate. I don’t need tap to activate, but I want to substitute my own image in place of the purple splash screen in the example. Because I’m already using a world tracking template, I can’t just build my project from the splash screen example project without losing the world tracking function. So, I figured I could just substitute my own image file, put it into an asset folder labeled Splash-Screen, and then tweak a few things in the code. However, I can’t find exactly what to tweak in order to make sure it’s my content that shows up. For starters, I don’t have any custom logos–I just want my image background to appear, so I would probably need to cut out some chunks of the code in splash-screen.ts. I’m a beginner to code, so the ts file is honestly difficult for me to understand. At first, I wanted to see if I could replicate the splash screen effect in my own project file using the example logos/svg. But I keep getting errors while building. The console outputs something like ‘Couldn’t create clone. Did you forget to set EntityToSpawn?’ I tried commenting out some lines, but got so turned around I don’t know what I’m doing anymore. Basically I want to adapt my own project to have the custom splash screen while loading my content without having to switch templates.

Thank you so much for any guidance.
Here is the code of splash-screen.ts in that example file if it helps:

import * as ecs from '@8thwall/ecs'

const componentsForClone = [
  ecs.Position, ecs.Quaternion, ecs.Scale, ecs.Shadow, ecs.BoxGeometry, ecs.Material, ecs.ScaleAnimation, ecs.PositionAnimation,
  ecs.RotateAnimation, ecs.CustomPropertyAnimation, ecs.CustomVec3Animation, ecs.FollowAnimation,
  ecs.LookAtAnimation, ecs.GltfModel, ecs.Collider, ecs.ParticleEmitter, ecs.Ui, ecs.Audio,
]

const cloneComponents = (sourceEid, targetEid, world) => {
  componentsForClone.forEach((component) => {
    if (component.has(world, sourceEid)) {
      const properties = component.get(world, sourceEid)
      component.set(world, targetEid, {...properties})
    }
  })
}

ecs.registerComponent({
  name: 'Tap Place',
  schema: {
    //entityToSpawn: ecs.eid,  // Entity ID for the entity to spawn
    //minScale: ecs.f32,  // Minimum scale for the spawned entity
    //maxScale: ecs.f32,  // Maximum scale for the spawned entity
 // },
  schemaDefaults: {
    minScale: 1.0,  // Default minimum scale is 1.0
    maxScale: 3.0,  // Default maximum scale is 3.0
  },
  data: {
    lastInteractionTime: ecs.f64,
  },
  stateMachine: ({world, eid, schemaAttribute, dataAttribute}) => {
    ecs.defineState('default')
      .initial()
      .onEnter(() => {
        const {entityToSpawn} = schemaAttribute.get(eid)

        if (entityToSpawn) {
        // Disable the entityToSpawn
          ecs.Disabled.set(world, entityToSpawn)
        }
      })
      .listen(eid, ecs.input.SCREEN_TOUCH_START, (e) => {
        const {entityToSpawn, minScale, maxScale} = schemaAttribute.get(eid)
        const currentTime = Date.now()

        if (currentTime - dataAttribute.get(eid).lastInteractionTime <= 500) {
          return
        }

        dataAttribute.set(eid, {
          lastInteractionTime: currentTime,
        })

        if (entityToSpawn) {
          const newEntity = world.createEntity()
          const randomScale = Math.random() * (maxScale - minScale) + minScale

          cloneComponents(entityToSpawn, newEntity, world)

          ecs.Position.set(world, newEntity, e.data.worldPosition)
          ecs.ScaleAnimation.set(world, newEntity, {
            fromX: 0,
            fromY: 0,
            fromZ: 0,
            toX: randomScale,
            toY: randomScale,
            toZ: randomScale,
            duration: 400,
            loop: false,
            easeOut: true,
            easingFunction: 'Quadratic',
          })
        } else {
          console.error('Couldn\'t create a clone. Did you forget to set entityToSpawn in the properties?')
        }
      })
  },
})

Hi! I’d be happy to take a look. Can you land your changes and share your project with the support workspace?

Awesome, thank you @GeorgeButler! Unfortunately it seems I don’t have the authority to share my project as a non-admin, so I’m not sure what I can do. But the loading screen effect I need comes from this template: Splash Screen | 8th Wall Playground | 8th Wall

I realize this may be very difficult without having the project to work from. But I’m hoping to find out how to modify the code in splashscreen.ts to do the following:

  • instead of ‘tap to begin’, the experience starts immediately after loading progress
  • Instead of the Background.svg in the Splash-Screen asset folder, my own cover image is displayed. (with none of the other logos in the asset folder)
  • My own 3D content is displayed, of course, instead of the example content. The readme says make sure your assets are in the ‘default’ space, but I could’t find anything labeled ‘default.’
  • The words ‘aim camera towards ground’ appear on screen during loading progress (This is not as important as the above)

Lastly, I wondered if it matters that my cover image is a png rather than an svg? I used the same dimensions as your background image, but when I tried converting it to svg it turned it black and white–so I left it as png. Would that not work?

Thank you so much, and I totally understand if it’s too hard to help without the file.

Are you using that Splash Screen template? (AFrame) or the Studio one?

Oh, sorry! The studio one version

No worries!

If you take a look at the StateMachine in the Splash Screen component we see that it has 3 states.

  • loading
  • onboarding
  • experience-ready

onboarding is the state that listens for the screen touch event and goes to experience-ready when it happens. If you wanted to bypass that you simply need to remove the onboarding state and edit the onTrigger in the loading state. I’ve done this for you below:

This also goes for changing the text that shows up during loading, you simply need to edit the cursor.text string.

import * as ecs from '@8thwall/ecs'

ecs.registerComponent({
  name: 'splash-screen',
  schema: {
    progressText: ecs.eid,  // Entity ID for the UI element showing progress
    messageText: ecs.eid,  // Entity ID for the UI element displaying messages
    beginMessage: ecs.string,  // Message displayed when onboarding is ready to begin
    startingSpaceName: ecs.string,  // The name of the Space that has your content
  },
  schemaDefaults: {
    beginMessage: 'Tap to begin',  // Default message to display at the start of onboarding
    startingSpaceName: 'Default',
  },
  data: {
  },
  stateMachine: ({world, eid, schemaAttribute}) => {
    // Trigger to transition from 'loading' to 'onboarding'
    const toOnboarding = ecs.defineTrigger()

    // Define the 'loading' state
    ecs.defineState('loading')
      .initial()  // Set as the initial state
      .onTick(() => {
        const {pending, complete} = ecs.assets.getStatistics()  // Retrieve asset loading stats
        const progress = complete / (pending + complete)  // Calculate progress as a ratio

        if (progress >= 1) {
          world.time.setTimeout(() => {
            toOnboarding.trigger()  // Transition to 'onboarding' when all assets are loaded
          }, 100)
        }

        const {progressText} = schemaAttribute.get(eid)  // Retrieve the progress text element

        if (progressText) {
          // Update the progress text with the calculated progress percentage
          ecs.Ui.mutate(world, progressText, (cursor) => {
            cursor.text = `${Number.isNaN(progress) ? 0 : Math.floor(progress * 100)}%`
          })
        }
      })
      .onTrigger(toOnboarding, 'experience-ready')

    // Define the 'experience-ready' state
    ecs.defineState('experience-ready')
      .onEnter(() => {
        const {startingSpaceName} = schemaAttribute.get(eid)

        world.spaces.loadSpace(`${startingSpaceName}`)
      })
  },
})

As for customizing the splash screen. The splash screen is in its own “Space”. If you want to customize anything about it you can go to this space and edit the objects in there. See the video below:

1 Like

Amazing, thank you so much @GeorgeButler !! I’ve started implementing your solution now, and will finish in the morning. Also, just one more question if it’s ok. What if I want to copy some of the UI components in your Splash Screen space from the template project to my own project? I notice there is no download option, so I figure it’s not something you can just download. Is there a way to copy those over to a different file?

Thank you!!!

There isn’t a way to copy objects between projects however I’ll make a note of this feature for the product team.

Edit: I didn’t set splash screen to entry space–that was the issue!

Thank you making note of that! After replicating the UI elements in your project, I’m finding that my device preview screen gets stuck on a white screen rather than displaying the cover image. I’m looking for any differences in how I set it up, but everything seems to match. What type of issue does this sound like it could be related to?

If you’re getting a white screen I have a sneaking suspicion that you might have forgotten to hook up one of the schema properties.

If you take a look at your Splash Screen component does everything appear to be correct like it does in the screenshot? Do you see any errors in the browser console? If you can provide a screenshot that would help as well.

1 Like

This topic was automatically closed 4 days after the last reply. New replies are no longer allowed.