Replace AdvertiseComponent with DatasetVocalizerComponent (#38887)

* Replace AdvertiseComponent with DatasetVocalizerComponent

* No vocalizing while broken or without power

* Kill AdvertiseComponent/System

* This really shouldn't be here

* xmldoc for VocalizerRequiresPowerComponent

* TryIndex -> Index
This commit is contained in:
Tayrtahn
2025-07-10 14:12:24 -04:00
committed by GitHub
parent a97223bc70
commit ecbff409b6
14 changed files with 231 additions and 271 deletions

View File

@@ -1,47 +0,0 @@
using Content.Server.Advertise.EntitySystems;
using Content.Shared.Dataset;
using Robust.Shared.Prototypes;
namespace Content.Server.Advertise.Components;
/// <summary>
/// Makes this entity periodically advertise by speaking a randomly selected
/// message from a specified dataset into local chat.
/// </summary>
[RegisterComponent, Access(typeof(AdvertiseSystem))]
public sealed partial class AdvertiseComponent : Component
{
/// <summary>
/// Minimum time in seconds to wait before saying a new ad, in seconds. Has to be larger than or equal to 1.
/// </summary>
[DataField]
public int MinimumWait { get; private set; } = 8 * 60;
/// <summary>
/// Maximum time in seconds to wait before saying a new ad, in seconds. Has to be larger than or equal
/// to <see cref="MinimumWait"/>
/// </summary>
[DataField]
public int MaximumWait { get; private set; } = 10 * 60;
/// <summary>
/// If true, the delay before the first advertisement (at MapInit) will ignore <see cref="MinimumWait"/>
/// and instead be rolled between 0 and <see cref="MaximumWait"/>. This only applies to the initial delay;
/// <see cref="MinimumWait"/> will be respected after that.
/// </summary>
[DataField]
public bool Prewarm = true;
/// <summary>
/// The identifier for the advertisements dataset prototype.
/// </summary>
[DataField(required: true)]
public ProtoId<LocalizedDatasetPrototype> Pack { get; private set; }
/// <summary>
/// The next time an advertisement will be said.
/// </summary>
[DataField]
public TimeSpan NextAdvertisementTime { get; set; } = TimeSpan.Zero;
}

View File

@@ -1,106 +0,0 @@
using Content.Server.Advertise.Components;
using Content.Server.Chat.Systems;
using Content.Server.Power.Components;
using Content.Shared.VendingMachines;
using Robust.Shared.Prototypes;
using Robust.Shared.Random;
using Robust.Shared.Timing;
namespace Content.Server.Advertise.EntitySystems;
public sealed class AdvertiseSystem : EntitySystem
{
[Dependency] private readonly IPrototypeManager _prototypeManager = default!;
[Dependency] private readonly IRobustRandom _random = default!;
[Dependency] private readonly IGameTiming _gameTiming = default!;
[Dependency] private readonly ChatSystem _chat = default!;
/// <summary>
/// The maximum amount of time between checking if advertisements should be displayed
/// </summary>
private readonly TimeSpan _maximumNextCheckDuration = TimeSpan.FromSeconds(15);
/// <summary>
/// The next time the game will check if advertisements should be displayed
/// </summary>
private TimeSpan _nextCheckTime = TimeSpan.MinValue;
public override void Initialize()
{
SubscribeLocalEvent<AdvertiseComponent, MapInitEvent>(OnMapInit);
SubscribeLocalEvent<ApcPowerReceiverComponent, AttemptAdvertiseEvent>(OnPowerReceiverAttemptAdvertiseEvent);
SubscribeLocalEvent<VendingMachineComponent, AttemptAdvertiseEvent>(OnVendingAttemptAdvertiseEvent);
_nextCheckTime = TimeSpan.MinValue;
}
private void OnMapInit(EntityUid uid, AdvertiseComponent advert, MapInitEvent args)
{
var prewarm = advert.Prewarm;
RandomizeNextAdvertTime(advert, prewarm);
_nextCheckTime = MathHelper.Min(advert.NextAdvertisementTime, _nextCheckTime);
}
private void RandomizeNextAdvertTime(AdvertiseComponent advert, bool prewarm = false)
{
var minDuration = prewarm ? 0 : Math.Max(1, advert.MinimumWait);
var maxDuration = Math.Max(minDuration, advert.MaximumWait);
var waitDuration = TimeSpan.FromSeconds(_random.Next(minDuration, maxDuration));
advert.NextAdvertisementTime = _gameTiming.CurTime + waitDuration;
}
public void SayAdvertisement(EntityUid uid, AdvertiseComponent? advert = null)
{
if (!Resolve(uid, ref advert))
return;
var attemptEvent = new AttemptAdvertiseEvent(uid);
RaiseLocalEvent(uid, ref attemptEvent);
if (attemptEvent.Cancelled)
return;
if (_prototypeManager.TryIndex(advert.Pack, out var advertisements))
_chat.TrySendInGameICMessage(uid, Loc.GetString(_random.Pick(advertisements.Values)), InGameICChatType.Speak, hideChat: true);
}
public override void Update(float frameTime)
{
var currentGameTime = _gameTiming.CurTime;
if (_nextCheckTime > currentGameTime)
return;
// _nextCheckTime starts at TimeSpan.MinValue, so this has to SET the value, not just increment it.
_nextCheckTime = currentGameTime + _maximumNextCheckDuration;
var query = EntityQueryEnumerator<AdvertiseComponent>();
while (query.MoveNext(out var uid, out var advert))
{
if (currentGameTime > advert.NextAdvertisementTime)
{
SayAdvertisement(uid, advert);
// The timer is always refreshed when it expires, to prevent mass advertising (ex: all the vending machines have no power, and get it back at the same time).
RandomizeNextAdvertTime(advert);
}
_nextCheckTime = MathHelper.Min(advert.NextAdvertisementTime, _nextCheckTime);
}
}
private static void OnPowerReceiverAttemptAdvertiseEvent(EntityUid uid, ApcPowerReceiverComponent powerReceiver, ref AttemptAdvertiseEvent args)
{
args.Cancelled |= !powerReceiver.Powered;
}
private static void OnVendingAttemptAdvertiseEvent(EntityUid uid, VendingMachineComponent machine, ref AttemptAdvertiseEvent args)
{
args.Cancelled |= machine.Broken;
}
}
[ByRefEvent]
public record struct AttemptAdvertiseEvent(EntityUid? Advertiser)
{
public bool Cancelled = false;
}

View File

@@ -4,6 +4,7 @@ using Content.Server.Cargo.Systems;
using Content.Server.Emp;
using Content.Server.Power.Components;
using Content.Server.Power.EntitySystems;
using Content.Server.Vocalization.Systems;
using Content.Shared.Cargo;
using Content.Shared.Damage;
using Content.Shared.Destructible;
@@ -42,6 +43,7 @@ namespace Content.Server.VendingMachines
SubscribeLocalEvent<VendingMachineComponent, DamageChangedEvent>(OnDamageChanged);
SubscribeLocalEvent<VendingMachineComponent, PriceCalculationEvent>(OnVendingPrice);
SubscribeLocalEvent<VendingMachineComponent, EmpPulseEvent>(OnEmpPulse);
SubscribeLocalEvent<VendingMachineComponent, TryVocalizeEvent>(OnTryVocalize);
SubscribeLocalEvent<VendingMachineComponent, ActivatableUIOpenAttemptEvent>(OnActivatableUIOpenAttempt);
@@ -309,5 +311,10 @@ namespace Content.Server.VendingMachines
component.NextEmpEject = _timing.CurTime;
}
}
private void OnTryVocalize(Entity<VendingMachineComponent> ent, ref TryVocalizeEvent args)
{
args.Cancelled |= ent.Comp.Broken;
}
}
}

View File

@@ -0,0 +1,18 @@
using Content.Shared.Dataset;
using Robust.Shared.Prototypes;
namespace Content.Server.Vocalization.Components;
/// <summary>
/// A simple message provider for <see cref="VocalizationSystem"/> that randomly selects
/// messages from a <see cref="LocalizedDatasetPrototype"/>.
/// </summary>
[RegisterComponent]
public sealed partial class DatasetVocalizerComponent : Component
{
/// <summary>
/// ID of the <see cref="LocalizedDatasetPrototype"/> that will provide messages.
/// </summary>
[DataField]
public ProtoId<LocalizedDatasetPrototype> Dataset;
}

View File

@@ -27,4 +27,11 @@ public sealed partial class VocalizerComponent : Component
[DataField(customTypeSerializer: typeof(TimeOffsetSerializer))]
[AutoPausedField]
public TimeSpan NextVocalizeInterval = TimeSpan.Zero;
/// <summary>
/// If true, messages spoken by this vocalizer will not be logged in the chat window
/// and will only be shown as speech bubbles.
/// </summary>
[DataField]
public bool HideChat;
}

View File

@@ -0,0 +1,9 @@
namespace Content.Server.Vocalization.Components;
/// <summary>
/// Used in combination with <see cref="VocalizerComponent"/>.
/// Blocks any attempts to vocalize if the entity has an <see cref="ApcPowerReceiverComponent"/>
/// and is currently unpowered.
/// </summary>
[RegisterComponent]
public sealed partial class VocalizerRequiresPowerComponent : Component;

View File

@@ -0,0 +1,31 @@
using Content.Server.Vocalization.Components;
using Content.Shared.Random.Helpers;
using Robust.Shared.Prototypes;
using Robust.Shared.Random;
namespace Content.Server.Vocalization.Systems;
/// <inheritdoc cref="DatasetVocalizerComponent"/>
public sealed class DatasetVocalizationSystem : EntitySystem
{
[Dependency] private readonly IPrototypeManager _protoMan = default!;
[Dependency] private readonly IRobustRandom _random = default!;
public override void Initialize()
{
base.Initialize();
SubscribeLocalEvent<DatasetVocalizerComponent, TryVocalizeEvent>(OnTryVocalize);
}
private void OnTryVocalize(Entity<DatasetVocalizerComponent> ent, ref TryVocalizeEvent args)
{
if (args.Handled)
return;
var dataset = _protoMan.Index(ent.Comp.Dataset);
args.Message = _random.Pick(dataset);
args.Handled = true;
}
}

View File

@@ -1,4 +1,5 @@
using Content.Server.Chat.Systems;
using Content.Server.Power.Components;
using Content.Server.Vocalization.Components;
using Content.Shared.ActionBlocker;
using Robust.Shared.Random;
@@ -18,6 +19,27 @@ public sealed partial class VocalizationSystem : EntitySystem
[Dependency] private readonly IGameTiming _gameTiming = default!;
[Dependency] private readonly IRobustRandom _random = default!;
public override void Initialize()
{
base.Initialize();
SubscribeLocalEvent<VocalizerComponent, MapInitEvent>(OnMapInit);
SubscribeLocalEvent<VocalizerRequiresPowerComponent, TryVocalizeEvent>(OnRequiresPowerTryVocalize);
}
private void OnMapInit(Entity<VocalizerComponent> ent, ref MapInitEvent args)
{
ent.Comp.NextVocalizeInterval = _random.Next(ent.Comp.MinVocalizeInterval, ent.Comp.MaxVocalizeInterval);
}
private void OnRequiresPowerTryVocalize(Entity<VocalizerRequiresPowerComponent> ent, ref TryVocalizeEvent args)
{
if (!TryComp<ApcPowerReceiverComponent>(ent, out var receiver))
return;
args.Cancelled |= !receiver.Powered;
}
/// <summary>
/// Try speaking by raising a TryVocalizeEvent
/// This event is passed to systems adding a message to it and setting it to handled
@@ -27,6 +49,10 @@ public sealed partial class VocalizationSystem : EntitySystem
var tryVocalizeEvent = new TryVocalizeEvent();
RaiseLocalEvent(entity.Owner, ref tryVocalizeEvent);
// If the event was cancelled, don't speak
if (tryVocalizeEvent.Cancelled)
return;
// if the event was never handled, return
// this happens if there are no components that trigger systems to add a message to this event
if (!tryVocalizeEvent.Handled)
@@ -60,7 +86,7 @@ public sealed partial class VocalizationSystem : EntitySystem
return;
// send the message
_chat.TrySendInGameICMessage(entity, message, InGameICChatType.Speak, ChatTransmitRange.Normal);
_chat.TrySendInGameICMessage(entity, message, InGameICChatType.Speak, entity.Comp.HideChat ? ChatTransmitRange.HideChat : ChatTransmitRange.Normal);
}
public override void Update(float frameTime)
@@ -99,7 +125,7 @@ public sealed partial class VocalizationSystem : EntitySystem
/// <param name="Message">Message to send, this is null when the event is just fired and should be set by a system</param>
/// <param name="Handled">Whether the message was handled by a system</param>
[ByRefEvent]
public record struct TryVocalizeEvent(string? Message = null, bool Handled = false);
public record struct TryVocalizeEvent(string? Message = null, bool Handled = false, bool Cancelled = false);
/// <summary>
/// Fired when the entity wants to vocalize and has a message. Allows for interception by other systems if the

View File

@@ -3958,8 +3958,6 @@ entities:
- type: Transform
pos: 8.5,27.5
parent: 2
- type: Advertise
nextAdvertisementTime: 6277.3942716
- proto: WallPlastitaniumDiagonalIndestructible
entities:
- uid: 171

View File

@@ -159,8 +159,9 @@
interactFailureString: petting-failure-firebot
interactSuccessSound:
path: /Audio/Ambience/Objects/periodic_beep.ogg
- type: Advertise
pack: FirebotAd
- type: Vocalizer
- type: DatasetVocalizer
dataset: FirebotAd
- type: entity
parent: MobSiliconBase
@@ -355,8 +356,9 @@
interactFailureString: petting-failure-medibot
interactSuccessSound:
path: /Audio/Ambience/Objects/periodic_beep.ogg
- type: Advertise
pack: MedibotAds
- type: Vocalizer
- type: DatasetVocalizer
dataset: MedibotAds
- type: Inventory
templateId: medibot
- type: DoAfter

View File

@@ -170,10 +170,12 @@
type: WiresBoundUserInterface
- type: Computer
board: SpaceVillainArcadeComputerCircuitboard
- type: Advertise
pack: SpaceVillainAds
minimumWait: 60 # Arcades are noisy
maximumWait: 240
- type: Vocalizer
minVocalizeInterval: 1m # Arcades are noisy
maxVocalizeInterval: 4m
hideChat: true
- type: DatasetVocalizer
dataset: SpaceVillainAds
- type: SpeakOnUIClosed
pack: SpaceVillainGoodbyes
@@ -215,9 +217,11 @@
type: WiresBoundUserInterface
- type: Computer
board: BlockGameArcadeComputerCircuitboard
- type: Advertise
pack: BlockGameAds
minimumWait: 60 # Arcades are noisy
maximumWait: 240
- type: Vocalizer
minVocalizeInterval: 1m # Arcades are noisy
maxVocalizeInterval: 4m
hideChat: true
- type: DatasetVocalizer
dataset: BlockGameAds
- type: SpeakOnUIClosed
pack: BlockGameGoodbyes

View File

@@ -101,8 +101,11 @@
- type: Appearance
- type: Speech
speechVerb: Robotic
- type: Advertise
pack: FatExtractorFacts
- type: Vocalizer
hideChat: true
- type: VocalizerRequiresPower
- type: DatasetVocalizer
dataset: FatExtractorFacts
- type: StaticPrice
price: 1000
- type: ResistLocker

View File

@@ -5,8 +5,11 @@
description: A refrigerated storage unit for keeping items cold and fresh.
components:
- type: StationAiWhitelist
- type: Advertise
pack: SmartFridgeAds
- type: Vocalizer
hideChat: true
- type: VocalizerRequiresPower
- type: DatasetVocalizer
dataset: SmartFridgeAds
- type: Speech
- type: Appearance
- type: Sprite

View File

@@ -73,6 +73,11 @@
- type: Speech
speechVerb: Robotic
speechSounds: Vending
- type: Vocalizer
minVocalizeInterval: 8m
maxVocalizeInterval: 10m
hideChat: true
- type: VocalizerRequiresPower
- type: SpookySpeaker
messageSet: SpookySpeakerMessagesGeneric
speakChance: 0.2
@@ -126,8 +131,8 @@
layer:
- MachineLayer
density: 190
- type: Advertise
pack: CondimentVendAds
- type: DatasetVocalizer
dataset: CondimentVendAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Transform
@@ -144,8 +149,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: AmmoVendAds
- type: DatasetVocalizer
dataset: CondimentVendAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -172,8 +177,8 @@
normalState: normal-unshaded
denyState: deny-unshaded
loopDeny: false
- type: Advertise
pack: BoozeOMatAds
- type: DatasetVocalizer
dataset: BoozeOMatAds
- type: SpeakOnUIClosed
pack: BoozeOMatGoodbyes
- type: Sprite
@@ -206,8 +211,8 @@
normalState: normal-unshaded
denyState: deny-unshaded
loopDeny: false
- type: Advertise
pack: BruiseOMatAds
- type: DatasetVocalizer
dataset: BruiseOMatAds
- type: SpeakOnUIClosed
pack: BruiseOMatGoodbyes
- type: Sprite
@@ -269,8 +274,8 @@
brokenState: broken
normalState: normal-unshaded
ejectState: eject-unshaded
- type: Advertise
pack: ChefvendAds
- type: DatasetVocalizer
dataset: ChefvendAds
- type: SpeakOnUIClosed
pack: ChefvendGoodbyes
- type: Sprite
@@ -305,8 +310,8 @@
normalState: normal-unshaded
ejectState: eject-unshaded
denyState: deny-unshaded
- type: Advertise
pack: CigaretteMachineAds
- type: DatasetVocalizer
dataset: CigaretteMachineAds
- type: SpeakOnUIClosed
pack: CigaretteMachineGoodbyes
- type: Sprite
@@ -332,8 +337,8 @@
brokenState: broken
normalState: normal-unshaded
denyState: deny-unshaded
- type: Advertise
pack: ClothesMateAds
- type: DatasetVocalizer
dataset: ClothesMateAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -363,8 +368,8 @@
brokenState: broken
normalState: normal-unshaded
denyState: deny-unshaded
- type: Advertise
pack: ClothesMateAds
- type: DatasetVocalizer
dataset: ClothesMateAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -401,8 +406,8 @@
ejectDelay: 5
soundVend: /Audio/Machines/machine_vend_hot_drink.ogg
initialStockQuality: 0.33
- type: Advertise
pack: HotDrinksMachineAds
- type: DatasetVocalizer
dataset: HotDrinksMachineAds
- type: SpeakOnUIClosed
pack: HotDrinksMachineGoodbyes
- type: Sprite
@@ -440,8 +445,8 @@
denyState: deny-unshaded
ejectDelay: 1.9
initialStockQuality: 0.33
- type: Advertise
pack: RobustSoftdrinksAds
- type: DatasetVocalizer
dataset: RobustSoftdrinksAds
- type: SpeakOnUIClosed
pack: RobustSoftdrinksGoodbyes
- type: Sprite
@@ -585,8 +590,8 @@
denyState: deny-unshaded
ejectDelay: 1.9
initialStockQuality: 0.33
- type: Advertise
pack: RobustSoftdrinksAds
- type: DatasetVocalizer
dataset: RobustSoftdrinksAds
- type: Sprite
sprite: Structures/Machines/VendingMachines/shamblersjuice.rsi
layers:
@@ -619,8 +624,8 @@
denyState: deny-unshaded
ejectDelay: 1.9
initialStockQuality: 0.33
- type: Advertise
pack: RobustSoftdrinksAds
- type: DatasetVocalizer
dataset: RobustSoftdrinksAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -655,8 +660,8 @@
denyState: deny-unshaded
ejectDelay: 1.9
initialStockQuality: 0.33
- type: Advertise
pack: DrGibbAds
- type: DatasetVocalizer
dataset: DrGibbAds
- type: SpeakOnUIClosed
pack: DrGibbGoodbyes
- type: Sprite
@@ -701,8 +706,8 @@
denyState: deny-unshaded
ejectDelay: 1.9
initialStockQuality: 0.33
- type: Advertise
pack: SmiteAds
- type: DatasetVocalizer
dataset: SmiteAds
- type: SpeakOnUIClosed
pack: SmiteGoodbyes
- type: PointLight
@@ -722,8 +727,8 @@
brokenState: broken
normalState: normal-unshaded
ejectState: eject-unshaded
- type: Advertise
pack: DinnerwareAds
- type: DatasetVocalizer
dataset: DinnerwareAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -754,8 +759,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: MagiVendAds
- type: DatasetVocalizer
dataset: MagiVendAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -787,8 +792,8 @@
brokenState: broken
normalState: normal-unshaded
initialStockQuality: 0.33
- type: Advertise
pack: DiscountDansAds
- type: DatasetVocalizer
dataset: DiscountDansAds
- type: SpeakOnUIClosed
pack: DiscountDansGoodbyes
- type: Sprite
@@ -850,8 +855,8 @@
ejectState: eject-unshaded
denyState: deny-unshaded
ejectDelay: 0.6
- type: Advertise
pack: NanoMedAds
- type: DatasetVocalizer
dataset: NanoMedAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -887,8 +892,8 @@
normalState: normal-unshaded
ejectState: eject-unshaded
denyState: deny-unshaded
- type: Advertise
pack: NutriMaxAds
- type: DatasetVocalizer
dataset: NutriMaxAds
- type: SpeakOnUIClosed
pack: NutriMaxGoodbyes
- type: Sprite
@@ -921,8 +926,8 @@
normalState: normal-unshaded
ejectState: eject-unshaded
denyState: deny-unshaded
- type: Advertise
pack: SecTechAds
- type: DatasetVocalizer
dataset: SecTechAds
- type: SpeakOnUIClosed
pack: SecTechGoodbyes
- type: Sprite
@@ -960,8 +965,8 @@
normalState: normal-unshaded
ejectState: eject-unshaded
denyState: deny-unshaded
- type: Advertise
pack: MegaSeedAds
- type: DatasetVocalizer
dataset: MegaSeedAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1003,8 +1008,8 @@
ejectState: eject-unshaded
denyState: deny-unshaded
initialStockQuality: 0.33
- type: Advertise
pack: GetmoreChocolateCorpAds
- type: DatasetVocalizer
dataset: GetmoreChocolateCorpAds
- type: SpeakOnUIClosed
pack: GetmoreChocolateCorpGoodbyes
- type: Sprite
@@ -1148,8 +1153,8 @@
ejectState: eject-unshaded
denyState: deny-unshaded
initialStockQuality: 0.33
- type: Advertise
pack: BodaAds
- type: DatasetVocalizer
dataset: BodaAds
- type: SpeakOnUIClosed
pack: BodaGoodbyes
- type: Sprite
@@ -1182,8 +1187,8 @@
ejectState: eject-unshaded
denyState: deny-unshaded
screenState: screen
- type: Advertise
pack: AutoDrobeAds
- type: DatasetVocalizer
dataset: AutoDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1217,8 +1222,8 @@
normalState: normal-unshaded
ejectState: eject-unshaded
denyState: deny-unshaded
- type: Advertise
pack: VendomatAds
- type: DatasetVocalizer
dataset: VendomatAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1249,8 +1254,8 @@
normalState: normal-unshaded
ejectState: eject-unshaded
denyState: deny-unshaded
- type: Advertise
pack: VendomatAds
- type: DatasetVocalizer
dataset: VendomatAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1314,8 +1319,8 @@
normalState: normal-unshaded
ejectState: eject-unshaded
ejectDelay: 1.8
- type: Advertise
pack: GoodCleanFunAds
- type: DatasetVocalizer
dataset: GoodCleanFunAds
- type: SpeakOnUIClosed
pack: GoodCleanFunGoodbyes
- type: Sprite
@@ -1347,8 +1352,8 @@
brokenState: broken
normalState: normal-unshaded
initialStockQuality: 0.33
- type: Advertise
pack: ChangAds
- type: DatasetVocalizer
dataset: ChangAds
- type: SpeakOnUIClosed
pack: ChangGoodbyes
- type: Sprite
@@ -1412,8 +1417,8 @@
brokenState: broken
normalState: normal-unshaded
initialStockQuality: 0.33
- type: Advertise
pack: DonutAds
- type: DatasetVocalizer
dataset: DonutAds
- type: SpeakOnUIClosed
pack: DonutGoodbyes
- type: Sprite
@@ -1497,8 +1502,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: HyDrobeAds
- type: DatasetVocalizer
dataset: HyDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1529,8 +1534,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: LawDrobeAds
- type: DatasetVocalizer
dataset: LawDrobeAds
- type: SpeakOnUIClosed
pack: LawDrobeGoodbyes
- type: Sprite
@@ -1557,8 +1562,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: SecDrobeAds
- type: DatasetVocalizer
dataset: SecDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1589,8 +1594,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: BarDrobeAds
- type: DatasetVocalizer
dataset: BarDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1645,8 +1650,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: CargoDrobeAds
- type: DatasetVocalizer
dataset: CargoDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1677,8 +1682,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: MediDrobeAds
- type: DatasetVocalizer
dataset: MediDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1705,8 +1710,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: ChemDrobeAds
- type: DatasetVocalizer
dataset: ChemDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1733,8 +1738,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: CuraDrobeAds
- type: DatasetVocalizer
dataset: CuraDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1761,8 +1766,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: AtmosDrobeAds
- type: DatasetVocalizer
dataset: AtmosDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1793,8 +1798,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: EngiDrobeAds
- type: DatasetVocalizer
dataset: EngiDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1825,8 +1830,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: ChefDrobeAds
- type: DatasetVocalizer
dataset: ChefDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1853,8 +1858,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: DetDrobeAds
- type: DatasetVocalizer
dataset: DetDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1881,8 +1886,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: JaniDrobeAds
- type: DatasetVocalizer
dataset: JaniDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1913,8 +1918,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: SciDrobeAds
- type: DatasetVocalizer
dataset: SciDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -1941,8 +1946,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: SyndieDrobeAds
- type: DatasetVocalizer
dataset: SyndieDrobeAds
- type: SpeakOnUIClosed
pack: SyndieDrobeGoodbyes
- type: Sprite
@@ -1973,8 +1978,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: RoboDrobeAds
- type: DatasetVocalizer
dataset: RoboDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -2001,8 +2006,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: GeneDrobeAds
- type: DatasetVocalizer
dataset: GeneDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -2029,8 +2034,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: ViroDrobeAds
- type: DatasetVocalizer
dataset: ViroDrobeAds
- type: SpeakOnUIClosed
pack: GenericVendGoodbyes
- type: Sprite
@@ -2105,8 +2110,8 @@
radius: 1.5
energy: 1.6
color: "#3c5eb5"
- type: Advertise
pack: HappyHonkAds
- type: DatasetVocalizer
dataset: HappyHonkAds
- type: SpeakOnUIClosed
pack: HappyHonkGoodbyes
- type: AccessReader
@@ -2125,8 +2130,8 @@
offState: off
brokenState: broken
normalState: normal-unshaded
- type: Advertise
pack: PrideDrobeAds
- type: DatasetVocalizer
dataset: PrideDrobeAds
- type: SpeakOnUIClosed
pack: PrideDrobeGoodbyes
- type: Speech