Skip to main content
Version: 3.0

Creating Custom Markers

If the built-in Markers don't meet your specific requirements, you can create your own Marker from scratch.

In this guide, we'll create a Marker that displays a series of images with corresponding text underneath and a button that appears at the end. The Marker will also support localized text and images.

Adding C# Scripts

1. Create Settings Class

First, create a new C# script that derives from DerivedMarkerSettings. This class will store your Marker's custom settings. Any Unity-serializable property is supported.

using System;
using System.Collections.Generic;
using UnityEngine;
using WorldTools.TutorialMaster.Core.Data.Localization;
using WorldTools.TutorialMaster.Core.Data.Markers;

[Serializable]
public class Slide
{
[SerializeField]
public SpriteResource Image;

[SerializeField]
public StringResource Text;
}

public class CarouselBoxSettings : DerivedMarkerSettings
{
[SerializeField]
public List<Slide> Slides = new();

[SerializeField]
public StringResource HeaderText = new();
}
Localization-ready value types

Instead of using String and Sprite to store data, we're using StringResource and SpriteResource. These support localized text/assets and also work even if you don't have the Unity Localization package installed. It's recommended to use these types if you intend to add localization at some point in the future.

Additionally, they integrate well with Tutorial Master features (for example, you can inject Tutorial Master variables into Unity Localization's smart strings).

You can learn more about localization support here.

2. Create Marker Behavior Class

Next, create another C# script that derives from Marker<>. This class will define your Marker's behavior.

using System.Collections;
using TMPro;
using UnityEngine.UI;
using WorldTools.TutorialMaster.Core.Components;
using WorldTools.TutorialMaster.Core.Data;

public class CarouselBox : Marker<CarouselBoxSettings>
{
protected override IEnumerator OnApply(IStageContext context)
{
// set Image, Text etc. from `DerivedSettings` field
yield break;
}

protected override void OnReset(IStageContext context)
{
// reset any intermediate data that this Marker may have
}
}
Complete implementation of the CarouselBox component
using System;
using System.Collections;
using TMPro;
using UnityEngine;
using UnityEngine.Localization;
using UnityEngine.Localization.Settings;
using UnityEngine.UI;
using WorldTools.TutorialMaster.Core.Components;
using WorldTools.TutorialMaster.Core.Data;

public class CarouselBox : Marker<CarouselBoxSettings>
{
[NonSerialized]
private int m_CurrentSlide = 0;

[Header("Slide")]
public Image Image;
public TMP_Text Body;

[Header("Buttons")]
public Button ButtonConfirm;
public Button ButtonNextSlide;
public Button ButtonPrevSlide;

[Header("Text")]
public TMP_Text Header;
public TMP_Text SlideCountText;

private bool HasNextSlide => m_CurrentSlide + 1 < (DerivedSettings?.Slides.Count ?? 0);
private bool HasPrevSlide => m_CurrentSlide - 1 >= 0;

private void Start()
{
ButtonNextSlide.onClick.AddListener(NextSlide);
ButtonPrevSlide.onClick.AddListener(PrevSlide);
ButtonConfirm.onClick.AddListener(GoToNextStage);
}

private void OnDestroy()
{
ButtonNextSlide.onClick.RemoveListener(NextSlide);
ButtonPrevSlide.onClick.RemoveListener(PrevSlide);
ButtonConfirm.onClick.RemoveListener(GoToNextStage);
}

private void NextSlide()
{
if (!HasNextSlide)
{
return;
}

m_CurrentSlide += 1;
SetSlide(DerivedSettings.Slides[m_CurrentSlide]);
}

private void PrevSlide()
{
if (!HasPrevSlide)
{
return;
}

m_CurrentSlide -= 1;
SetSlide(DerivedSettings.Slides[m_CurrentSlide]);
}

private void SetSlide(Slide slide)
{
Image.sprite = slide.Image.GetValue();
Body.text = slide.Text.GetValue(Context);
SlideCountText.text = $"{m_CurrentSlide + 1}/{DerivedSettings.Slides.Count}";

ToggleButtonInteractivity();
}

private void ToggleButtonInteractivity()
{
ButtonNextSlide.interactable = HasNextSlide;
ButtonPrevSlide.interactable = HasPrevSlide;
ButtonConfirm.interactable = !HasNextSlide;
}

private void GoToNextStage()
{
Context?.NextStage();
}

private void OnLocaleChange(Locale _)
{
Header.text = DerivedSettings.HeaderText.GetValue(Context);
SetSlide(DerivedSettings.Slides[m_CurrentSlide]);
}

protected override IEnumerator OnApply(IStageContext context)
{
Header.text = DerivedSettings.HeaderText.GetValue(context);
SetSlide(DerivedSettings.Slides[m_CurrentSlide]);
LocalizationSettings.SelectedLocaleChanged += OnLocaleChange;
yield break;
}

protected override void OnReset(IStageContext context)
{
m_CurrentSlide = 0;
LocalizationSettings.SelectedLocaleChanged -= OnLocaleChange;
}
}

Creating the Prefab

warning

All Marker components must be defined at the root of your GameObject hierarchy.

  1. Create a UI and arrange it as intended. Below is the typical setup for a custom Marker:

info

Anchors must be centered. The dimensions of the UI do not matter.

  1. Add your newly created CarouselBox component at the root GameObject. Assign all necessary component references.

  1. Create a prefab from your GameObject. Once created, you'll need to register the Marker. Follow this guide to learn how to do that.

Testing Your Custom Marker

  1. Add a Spawn Marker action in your tutorial, and select your newly created Marker Pool. You should be able to modify the Marker settings as needed.

  2. Run your tutorial and test your newly created Marker!