2018-04-13 09:19:50 +00:00
|
|
|
|
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
|
|
|
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
|
|
|
|
|
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Game.Beatmaps;
|
|
|
|
|
using osu.Game.Rulesets.Judgements;
|
|
|
|
|
using osu.Game.Rulesets.Mods;
|
|
|
|
|
using osu.Game.Rulesets.Objects;
|
|
|
|
|
using osu.Game.Rulesets.Objects.Drawables;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using osu.Framework.Configuration;
|
|
|
|
|
using osu.Framework.Graphics.Cursor;
|
|
|
|
|
using osu.Framework.Input;
|
|
|
|
|
using osu.Game.Configuration;
|
|
|
|
|
using osu.Game.Input.Handlers;
|
|
|
|
|
using osu.Game.Overlays;
|
2018-11-28 08:20:37 +00:00
|
|
|
|
using osu.Game.Replays;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Game.Rulesets.Configuration;
|
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.UI
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Base RulesetContainer. Doesn't hold objects.
|
|
|
|
|
/// <para>
|
|
|
|
|
/// Should not be derived - derive <see cref="RulesetContainer{TObject}"/> instead.
|
|
|
|
|
/// </para>
|
|
|
|
|
/// </summary>
|
|
|
|
|
public abstract class RulesetContainer : Container
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The selected variant.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public virtual int Variant => 0;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The input manager for this RulesetContainer.
|
|
|
|
|
/// </summary>
|
|
|
|
|
internal IHasReplayHandler ReplayInputManager => KeyBindingInputManager as IHasReplayHandler;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The key conversion input manager for this RulesetContainer.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public PassThroughInputManager KeyBindingInputManager;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether a replay is currently loaded.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly BindableBool HasReplayLoaded = new BindableBool();
|
|
|
|
|
|
|
|
|
|
public abstract IEnumerable<HitObject> Objects { get; }
|
|
|
|
|
|
2018-07-17 05:29:22 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The point in time at which gameplay starts, including any required lead-in for display purposes.
|
|
|
|
|
/// Defaults to two seconds before the first <see cref="HitObject"/>. Override as necessary.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public virtual double GameplayStartTime => Objects.First().StartTime - 2000;
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
private readonly Lazy<Playfield> playfield;
|
2018-06-06 05:20:51 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The playfield.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Playfield Playfield => playfield.Value;
|
|
|
|
|
|
2018-08-05 11:12:31 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Place to put drawables above hit objects but below UI.
|
|
|
|
|
/// </summary>
|
2018-12-20 10:35:32 +00:00
|
|
|
|
public Container Overlays { get; protected set; }
|
2018-08-05 11:12:31 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The cursor provided by this <see cref="RulesetContainer"/>. May be null if no cursor is provided.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly CursorContainer Cursor;
|
|
|
|
|
|
2018-10-17 09:01:38 +00:00
|
|
|
|
public readonly Ruleset Ruleset;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2018-07-11 08:25:57 +00:00
|
|
|
|
protected IRulesetConfigManager Config { get; private set; }
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
private OnScreenDisplay onScreenDisplay;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// A visual representation of a <see cref="Rulesets.Ruleset"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ruleset">The ruleset being repesented.</param>
|
|
|
|
|
protected RulesetContainer(Ruleset ruleset)
|
|
|
|
|
{
|
|
|
|
|
Ruleset = ruleset;
|
|
|
|
|
playfield = new Lazy<Playfield>(CreatePlayfield);
|
|
|
|
|
|
2018-07-11 08:01:27 +00:00
|
|
|
|
IsPaused.ValueChanged += paused =>
|
|
|
|
|
{
|
|
|
|
|
if (HasReplayLoaded)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
KeyBindingInputManager.UseParentInput = !paused;
|
|
|
|
|
};
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
Cursor = CreateCursor();
|
|
|
|
|
}
|
2018-05-06 10:57:52 +00:00
|
|
|
|
|
2018-07-11 08:07:14 +00:00
|
|
|
|
protected override IReadOnlyDependencyContainer CreateChildDependencies(IReadOnlyDependencyContainer parent)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-07-11 08:07:14 +00:00
|
|
|
|
var dependencies = new DependencyContainer(base.CreateChildDependencies(parent));
|
2018-06-11 05:36:56 +00:00
|
|
|
|
|
|
|
|
|
onScreenDisplay = dependencies.Get<OnScreenDisplay>();
|
2018-05-21 06:56:02 +00:00
|
|
|
|
|
2018-07-11 08:25:57 +00:00
|
|
|
|
Config = dependencies.Get<RulesetConfigCache>().GetConfigFor(Ruleset);
|
|
|
|
|
if (Config != null)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-07-11 08:25:57 +00:00
|
|
|
|
dependencies.Cache(Config);
|
|
|
|
|
onScreenDisplay?.BeginTracking(this, Config);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-06-11 05:36:56 +00:00
|
|
|
|
return dependencies;
|
|
|
|
|
}
|
2018-06-11 03:57:26 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
public abstract ScoreProcessor CreateScoreProcessor();
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a key conversion input manager. An exception will be thrown if a valid <see cref="RulesetInputManager{T}"/> is not returned.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>The input manager.</returns>
|
|
|
|
|
public abstract PassThroughInputManager CreateInputManager();
|
|
|
|
|
|
|
|
|
|
protected virtual ReplayInputHandler CreateReplayInputHandler(Replay replay) => null;
|
|
|
|
|
|
|
|
|
|
public Replay Replay { get; private set; }
|
|
|
|
|
|
2018-07-11 08:01:27 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the game is paused. Used to block user input.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly BindableBool IsPaused = new BindableBool();
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Sets a replay to be used, overriding local input.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="replay">The replay, null for local input.</param>
|
|
|
|
|
public virtual void SetReplay(Replay replay)
|
|
|
|
|
{
|
|
|
|
|
if (ReplayInputManager == null)
|
|
|
|
|
throw new InvalidOperationException($"A {nameof(KeyBindingInputManager)} which supports replay loading is not available");
|
|
|
|
|
|
|
|
|
|
Replay = replay;
|
|
|
|
|
ReplayInputManager.ReplayInputHandler = replay != null ? CreateReplayInputHandler(replay) : null;
|
|
|
|
|
|
|
|
|
|
HasReplayLoaded.Value = ReplayInputManager.ReplayInputHandler != null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates the cursor. May be null if the <see cref="RulesetContainer"/> doesn't provide a custom cursor.
|
|
|
|
|
/// </summary>
|
|
|
|
|
protected virtual CursorContainer CreateCursor() => null;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a Playfield.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>The Playfield.</returns>
|
|
|
|
|
protected abstract Playfield CreatePlayfield();
|
|
|
|
|
|
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
|
{
|
|
|
|
|
base.Dispose(isDisposing);
|
|
|
|
|
|
2018-07-11 08:25:57 +00:00
|
|
|
|
if (Config != null)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-07-11 08:25:57 +00:00
|
|
|
|
onScreenDisplay?.StopTracking(this, Config);
|
|
|
|
|
Config = null;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// RulesetContainer that applies conversion to Beatmaps. Does not contain a Playfield
|
|
|
|
|
/// and does not load drawable hit objects.
|
|
|
|
|
/// <para>
|
2018-06-19 12:31:24 +00:00
|
|
|
|
/// Should not be derived - derive <see cref="RulesetContainer{TPlayfield, TObject}"/> instead.
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// </para>
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <typeparam name="TObject">The type of HitObject contained by this RulesetContainer.</typeparam>
|
|
|
|
|
public abstract class RulesetContainer<TObject> : RulesetContainer
|
|
|
|
|
where TObject : HitObject
|
|
|
|
|
{
|
2018-08-06 01:54:16 +00:00
|
|
|
|
/// <summary>
|
2018-08-06 03:29:22 +00:00
|
|
|
|
/// Invoked when a <see cref="JudgementResult"/> has been applied by a <see cref="DrawableHitObject"/>.
|
2018-08-06 01:54:16 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
public event Action<JudgementResult> OnNewResult;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2018-08-06 03:29:22 +00:00
|
|
|
|
/// Invoked when a <see cref="JudgementResult"/> is being reverted by a <see cref="DrawableHitObject"/>.
|
2018-08-06 01:54:16 +00:00
|
|
|
|
/// </summary>
|
2018-08-06 03:29:22 +00:00
|
|
|
|
public event Action<JudgementResult> OnRevertResult;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The Beatmap
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Beatmap<TObject> Beatmap;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// All the converted hit objects contained by this hit renderer.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public override IEnumerable<HitObject> Objects => Beatmap.HitObjects;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The mods which are to be applied.
|
|
|
|
|
/// </summary>
|
2018-12-20 10:35:32 +00:00
|
|
|
|
public IEnumerable<Mod> Mods { get; protected set; }
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The <see cref="WorkingBeatmap"/> this <see cref="RulesetContainer{TObject}"/> was created with.
|
|
|
|
|
/// </summary>
|
|
|
|
|
protected readonly WorkingBeatmap WorkingBeatmap;
|
|
|
|
|
|
|
|
|
|
public override ScoreProcessor CreateScoreProcessor() => new ScoreProcessor<TObject>(this);
|
|
|
|
|
|
|
|
|
|
protected override Container<Drawable> Content => content;
|
|
|
|
|
private Container content;
|
2018-04-25 08:15:53 +00:00
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether to assume the beatmap passed into this <see cref="RulesetContainer{TObject}"/> is for the current ruleset.
|
|
|
|
|
/// Creates a hit renderer for a beatmap.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ruleset">The ruleset being repesented.</param>
|
|
|
|
|
/// <param name="workingBeatmap">The beatmap to create the hit renderer for.</param>
|
2018-05-07 01:17:54 +00:00
|
|
|
|
protected RulesetContainer(Ruleset ruleset, WorkingBeatmap workingBeatmap)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
: base(ruleset)
|
|
|
|
|
{
|
|
|
|
|
Debug.Assert(workingBeatmap != null, "RulesetContainer initialized with a null beatmap.");
|
|
|
|
|
|
|
|
|
|
WorkingBeatmap = workingBeatmap;
|
|
|
|
|
// ReSharper disable once PossibleNullReferenceException
|
|
|
|
|
Mods = workingBeatmap.Mods.Value;
|
|
|
|
|
|
|
|
|
|
RelativeSizeAxes = Axes.Both;
|
|
|
|
|
|
2018-05-07 01:40:30 +00:00
|
|
|
|
Beatmap = (Beatmap<TObject>)workingBeatmap.GetPlayableBeatmap(ruleset.RulesetInfo);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
KeyBindingInputManager = CreateInputManager();
|
|
|
|
|
KeyBindingInputManager.RelativeSizeAxes = Axes.Both;
|
2018-10-28 05:40:19 +00:00
|
|
|
|
|
|
|
|
|
applyBeatmapMods(Mods);
|
2018-05-21 06:59:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load(OsuConfigManager config)
|
|
|
|
|
{
|
2018-12-20 10:35:32 +00:00
|
|
|
|
KeyBindingInputManager.Children = new Drawable[]
|
2018-05-21 06:59:33 +00:00
|
|
|
|
{
|
2018-12-20 10:35:32 +00:00
|
|
|
|
content = new Container
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
|
},
|
|
|
|
|
Playfield
|
|
|
|
|
};
|
2018-05-21 06:59:33 +00:00
|
|
|
|
|
|
|
|
|
if (Cursor != null)
|
|
|
|
|
KeyBindingInputManager.Add(Cursor);
|
|
|
|
|
|
2018-12-20 10:35:32 +00:00
|
|
|
|
InternalChildren = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
KeyBindingInputManager,
|
|
|
|
|
Overlays = new Container { RelativeSizeAxes = Axes.Both }
|
|
|
|
|
};
|
|
|
|
|
|
2018-05-21 06:59:33 +00:00
|
|
|
|
// Apply mods
|
2018-10-28 05:40:19 +00:00
|
|
|
|
applyRulesetMods(Mods, config);
|
2018-05-21 06:59:33 +00:00
|
|
|
|
|
|
|
|
|
loadObjects();
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-28 05:40:19 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Applies the active mods to the Beatmap.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="mods"></param>
|
|
|
|
|
private void applyBeatmapMods(IEnumerable<Mod> mods)
|
|
|
|
|
{
|
|
|
|
|
if (mods == null)
|
|
|
|
|
return;
|
|
|
|
|
|
2018-10-30 04:13:33 +00:00
|
|
|
|
foreach (var mod in mods.OfType<IApplicableToBeatmap<TObject>>())
|
2018-10-28 05:40:19 +00:00
|
|
|
|
mod.ApplyToBeatmap(Beatmap);
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Applies the active mods to this RulesetContainer.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="mods"></param>
|
2018-10-28 05:40:19 +00:00
|
|
|
|
private void applyRulesetMods(IEnumerable<Mod> mods, OsuConfigManager config)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
|
|
|
|
if (mods == null)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
foreach (var mod in mods.OfType<IApplicableToRulesetContainer<TObject>>())
|
|
|
|
|
mod.ApplyToRulesetContainer(this);
|
2018-05-06 11:09:46 +00:00
|
|
|
|
|
2018-06-06 05:20:51 +00:00
|
|
|
|
foreach (var mod in mods.OfType<IReadFromConfig>())
|
|
|
|
|
mod.ReadFromConfig(config);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void SetReplay(Replay replay)
|
|
|
|
|
{
|
|
|
|
|
base.SetReplay(replay);
|
|
|
|
|
|
|
|
|
|
if (ReplayInputManager?.ReplayInputHandler != null)
|
2018-05-06 11:18:12 +00:00
|
|
|
|
ReplayInputManager.ReplayInputHandler.GamefieldToScreenSpace = Playfield.GamefieldToScreenSpace;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates and adds drawable representations of hit objects to the play field.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void loadObjects()
|
|
|
|
|
{
|
|
|
|
|
foreach (TObject h in Beatmap.HitObjects)
|
2018-10-03 06:36:14 +00:00
|
|
|
|
AddRepresentation(h);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
Playfield.PostProcess();
|
|
|
|
|
|
|
|
|
|
foreach (var mod in Mods.OfType<IApplicableToDrawableHitObjects>())
|
2018-08-30 04:30:23 +00:00
|
|
|
|
mod.ApplyToDrawableHitObjects(Playfield.HitObjectContainer.Objects);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-03 06:36:14 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates and adds the visual representation of a <see cref="TObject"/> to this <see cref="RulesetContainer{TObject}"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="hitObject">The <see cref="TObject"/> to add the visual representation for.</param>
|
|
|
|
|
internal void AddRepresentation(TObject hitObject)
|
|
|
|
|
{
|
|
|
|
|
var drawableObject = GetVisualRepresentation(hitObject);
|
|
|
|
|
|
|
|
|
|
if (drawableObject == null)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
drawableObject.OnNewResult += (_, r) => OnNewResult?.Invoke(r);
|
|
|
|
|
drawableObject.OnRevertResult += (_, r) => OnRevertResult?.Invoke(r);
|
|
|
|
|
|
|
|
|
|
Playfield.Add(drawableObject);
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-13 09:19:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a DrawableHitObject from a HitObject.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="h">The HitObject to make drawable.</param>
|
|
|
|
|
/// <returns>The DrawableHitObject.</returns>
|
2018-10-03 06:36:14 +00:00
|
|
|
|
public abstract DrawableHitObject<TObject> GetVisualRepresentation(TObject h);
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// A derivable RulesetContainer that manages the Playfield and HitObjects.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <typeparam name="TPlayfield">The type of Playfield contained by this RulesetContainer.</typeparam>
|
|
|
|
|
/// <typeparam name="TObject">The type of HitObject contained by this RulesetContainer.</typeparam>
|
|
|
|
|
public abstract class RulesetContainer<TPlayfield, TObject> : RulesetContainer<TObject>
|
|
|
|
|
where TObject : HitObject
|
|
|
|
|
where TPlayfield : Playfield
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// The playfield.
|
|
|
|
|
/// </summary>
|
|
|
|
|
protected new TPlayfield Playfield => (TPlayfield)base.Playfield;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a hit renderer for a beatmap.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="ruleset">The ruleset being repesented.</param>
|
|
|
|
|
/// <param name="beatmap">The beatmap to create the hit renderer for.</param>
|
2018-05-07 01:17:54 +00:00
|
|
|
|
protected RulesetContainer(Ruleset ruleset, WorkingBeatmap beatmap)
|
|
|
|
|
: base(ruleset, beatmap)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class BeatmapInvalidForRulesetException : ArgumentException
|
|
|
|
|
{
|
|
|
|
|
public BeatmapInvalidForRulesetException(string text)
|
|
|
|
|
: base(text)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|