diff --git a/src/Artemis.Core/Models/Profile/Profile.cs b/src/Artemis.Core/Models/Profile/Profile.cs index c511f9ec9..2a0d96a23 100644 --- a/src/Artemis.Core/Models/Profile/Profile.cs +++ b/src/Artemis.Core/Models/Profile/Profile.cs @@ -17,13 +17,12 @@ public sealed class Profile : ProfileElement private readonly ObservableCollection _scriptConfigurations; private readonly ObservableCollection _scripts; private bool _isFreshImport; - private ProfileElement? _lastSelectedProfileElement; internal Profile(ProfileConfiguration configuration, ProfileEntity profileEntity) : base(null!) { _scripts = new ObservableCollection(); _scriptConfigurations = new ObservableCollection(); - + Opacity = 0d; ShouldDisplay = true; Configuration = configuration; @@ -67,15 +66,6 @@ public sealed class Profile : ProfileElement set => SetAndNotify(ref _isFreshImport, value); } - /// - /// Gets or sets the last selected profile element of this profile - /// - public ProfileElement? LastSelectedProfileElement - { - get => _lastSelectedProfileElement; - set => SetAndNotify(ref _lastSelectedProfileElement, value); - } - /// /// Gets the profile entity this profile uses for persistent storage /// @@ -105,7 +95,7 @@ public sealed class Profile : ProfileElement profileScript.OnProfileUpdated(deltaTime); const double OPACITY_PER_SECOND = 1; - + if (ShouldDisplay && Opacity < 1) Opacity = Math.Clamp(Opacity + OPACITY_PER_SECOND * deltaTime, 0d, 1d); if (!ShouldDisplay && Opacity > 0) @@ -123,14 +113,14 @@ public sealed class Profile : ProfileElement foreach (ProfileScript profileScript in Scripts) profileScript.OnProfileRendering(canvas, canvas.LocalClipBounds); - + SKPaint? opacityPaint = null; bool applyOpacityLayer = Configuration.FadeInAndOut && Opacity < 1; - + if (applyOpacityLayer) { opacityPaint = new SKPaint(); - opacityPaint.Color = new SKColor(0, 0, 0, (byte)(255d * Easings.CubicEaseInOut(Opacity))); + opacityPaint.Color = new SKColor(0, 0, 0, (byte) (255d * Easings.CubicEaseInOut(Opacity))); canvas.SaveLayer(opacityPaint); } @@ -242,20 +232,13 @@ public sealed class Profile : ProfileElement AddChild(new Folder(this, this, rootFolder)); } - List renderElements = GetAllRenderElements(); - - if (ProfileEntity.LastSelectedProfileElement != Guid.Empty) - LastSelectedProfileElement = renderElements.FirstOrDefault(f => f.EntityId == ProfileEntity.LastSelectedProfileElement); - else - LastSelectedProfileElement = null; - while (_scriptConfigurations.Any()) RemoveScriptConfiguration(_scriptConfigurations[0]); foreach (ScriptConfiguration scriptConfiguration in ProfileEntity.ScriptConfigurations.Select(e => new ScriptConfiguration(e))) AddScriptConfiguration(scriptConfiguration); // Load node scripts last since they may rely on the profile structure being in place - foreach (RenderProfileElement renderProfileElement in renderElements) + foreach (RenderProfileElement renderProfileElement in GetAllRenderElements()) renderProfileElement.LoadNodeScript(); } @@ -312,7 +295,6 @@ public sealed class Profile : ProfileElement ProfileEntity.Id = EntityId; ProfileEntity.Name = Configuration.Name; ProfileEntity.IsFreshImport = IsFreshImport; - ProfileEntity.LastSelectedProfileElement = LastSelectedProfileElement?.EntityId ?? Guid.Empty; foreach (ProfileElement profileElement in Children) profileElement.Save(); diff --git a/src/Artemis.Core/Models/Profile/ProfileCategory.cs b/src/Artemis.Core/Models/Profile/ProfileCategory.cs index 13a3e2f4b..4dc3f73e6 100644 --- a/src/Artemis.Core/Models/Profile/ProfileCategory.cs +++ b/src/Artemis.Core/Models/Profile/ProfileCategory.cs @@ -98,6 +98,7 @@ public class ProfileCategory : CorePropertyChanged, IStorageModel /// public void AddProfileConfiguration(ProfileConfiguration configuration, int? targetIndex) { + // TODO: Look into this, it doesn't seem to make sense // Removing the original will shift every item in the list forwards, keep that in mind with the target index if (configuration.Category == this && targetIndex != null && targetIndex.Value > _profileConfigurations.IndexOf(configuration)) targetIndex -= 1; diff --git a/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfiguration.cs b/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfiguration.cs index 06a2090d3..b0dec1d29 100644 --- a/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfiguration.cs +++ b/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfiguration.cs @@ -147,16 +147,7 @@ public class ProfileConfiguration : BreakableModel, IStorageModel, IDisposable get => _activationConditionMet; private set => SetAndNotify(ref _activationConditionMet, value); } - - /// - /// Gets or sets a boolean indicating whether this profile configuration is being edited - /// - public bool IsBeingEdited - { - get => _isBeingEdited; - set => SetAndNotify(ref _isBeingEdited, value); - } - + /// /// Gets the profile of this profile configuration /// @@ -243,8 +234,6 @@ public class ProfileConfiguration : BreakableModel, IStorageModel, IDisposable { if (_disposed) throw new ObjectDisposedException("ProfileConfiguration"); - if (IsBeingEdited) - return true; if (Category.IsSuspended || IsSuspended || IsMissingModule) return false; diff --git a/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfigurationIcon.cs b/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfigurationIcon.cs index ed9646e7a..5a477b5fd 100644 --- a/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfigurationIcon.cs +++ b/src/Artemis.Core/Models/ProfileConfiguration/ProfileConfigurationIcon.cs @@ -15,7 +15,6 @@ public class ProfileConfigurationIcon : CorePropertyChanged, IStorageModel private string? _iconName; private Stream? _iconStream; private ProfileConfigurationIconType _iconType; - private string? _originalFileName; internal ProfileConfigurationIcon(ProfileConfigurationEntity entity) { diff --git a/src/Artemis.Core/Services/Storage/Interfaces/IProfileService.cs b/src/Artemis.Core/Services/Storage/Interfaces/IProfileService.cs index 099138b92..59fae7a9d 100644 --- a/src/Artemis.Core/Services/Storage/Interfaces/IProfileService.cs +++ b/src/Artemis.Core/Services/Storage/Interfaces/IProfileService.cs @@ -28,19 +28,26 @@ public interface IProfileService : IArtemisService ReadOnlyCollection ProfileConfigurations { get; } /// - /// Gets or sets a boolean indicating whether hotkeys are enabled. + /// Gets or sets the focused profile configuration which is rendered exclusively. /// - bool HotkeysEnabled { get; set; } + ProfileConfiguration? FocusProfile { get; set; } /// - /// Gets or sets a boolean indicating whether rendering should only be done for profiles being edited. + /// Gets or sets the profile element which is rendered exclusively. /// - bool RenderForEditor { get; set; } + ProfileElement? FocusProfileElement { get; set; } /// - /// Gets or sets the profile element to focus on while rendering for the editor. + /// Gets or sets a value indicating whether the currently focused profile should receive updates. /// - ProfileElement? EditorFocus { get; set; } + bool UpdateFocusProfile { get; set; } + + /// + /// Creates a copy of the provided profile configuration. + /// + /// The profile configuration to clone. + /// The resulting clone. + ProfileConfiguration CloneProfileConfiguration(ProfileConfiguration profileConfiguration); /// /// Activates the profile of the given with the currently active surface. @@ -71,8 +78,9 @@ public interface IProfileService : IArtemisService /// Creates a new profile category and saves it to persistent storage. /// /// The name of the new profile category, must be unique. + /// A boolean indicating whether or not to add the category to the top. /// The newly created profile category. - ProfileCategory CreateProfileCategory(string name); + ProfileCategory CreateProfileCategory(string name, bool addToTop = false); /// /// Permanently deletes the provided profile category. @@ -119,7 +127,7 @@ public interface IProfileService : IArtemisService Task ExportProfile(ProfileConfiguration profileConfiguration); /// - /// Imports the provided base64 encoded GZIPed JSON as a profile configuration. + /// Imports the provided ZIP archive stream as a profile configuration. /// /// The zip archive containing the profile to import. /// The in which to import the profile. @@ -129,8 +137,17 @@ public interface IProfileService : IArtemisService /// any changes are made to it. /// /// Text to add after the name of the profile (separated by a dash). + /// The index at which to import the profile into the category. /// The resulting profile configuration. - Task ImportProfile(Stream archiveStream, ProfileCategory category, bool makeUnique, bool markAsFreshImport, string? nameAffix = "imported"); + Task ImportProfile(Stream archiveStream, ProfileCategory category, bool makeUnique, bool markAsFreshImport, string? nameAffix = "imported", int targetIndex = 0); + + /// + /// Imports the provided ZIP archive stream into the provided profile configuration + /// + /// The zip archive containing the profile to import. + /// The profile configuration to overwrite. + /// The resulting profile configuration. + Task OverwriteProfile(MemoryStream archiveStream, ProfileConfiguration profileConfiguration); /// /// Adapts a given profile to the currently active devices. @@ -168,4 +185,5 @@ public interface IProfileService : IArtemisService /// Occurs whenever a profile category is removed. /// public event EventHandler? ProfileCategoryRemoved; + } \ No newline at end of file diff --git a/src/Artemis.Core/Services/Storage/ProfileService.cs b/src/Artemis.Core/Services/Storage/ProfileService.cs index b01e6bc56..7321fc6b2 100644 --- a/src/Artemis.Core/Services/Storage/ProfileService.cs +++ b/src/Artemis.Core/Services/Storage/ProfileService.cs @@ -19,14 +19,14 @@ namespace Artemis.Core.Services; internal class ProfileService : IProfileService { private readonly ILogger _logger; + private readonly IRgbService _rgbService; + private readonly IProfileCategoryRepository _profileCategoryRepository; + private readonly IPluginManagementService _pluginManagementService; private readonly List _pendingKeyboardEvents = new(); - private readonly IPluginManagementService _pluginManagementService; private readonly List _profileCategories; - private readonly IProfileCategoryRepository _profileCategoryRepository; private readonly IProfileRepository _profileRepository; private readonly List _renderExceptions = new(); - private readonly IRgbService _rgbService; private readonly List _updateExceptions = new(); private DateTime _lastRenderExceptionLog; private DateTime _lastUpdateExceptionLog; @@ -49,7 +49,6 @@ internal class ProfileService : IProfileService _pluginManagementService.PluginFeatureEnabled += PluginManagementServiceOnPluginFeatureToggled; _pluginManagementService.PluginFeatureDisabled += PluginManagementServiceOnPluginFeatureToggled; - HotkeysEnabled = true; inputService.KeyboardKeyUp += InputServiceOnKeyboardKeyUp; if (!_profileCategories.Any()) @@ -57,11 +56,488 @@ internal class ProfileService : IProfileService UpdateModules(); } - private void InputServiceOnKeyboardKeyUp(object? sender, ArtemisKeyboardKeyEventArgs e) + public ProfileConfiguration? FocusProfile { get; set; } + public ProfileElement? FocusProfileElement { get; set; } + public bool UpdateFocusProfile { get; set; } + + /// + public void UpdateProfiles(double deltaTime) { - if (!HotkeysEnabled) + // If there is a focus profile update only that, and only if UpdateFocusProfile is true + if (FocusProfile != null) + { + if (UpdateFocusProfile) + FocusProfile.Profile?.Update(deltaTime); + return; + } + + lock (_profileCategories) + { + // Iterate the children in reverse because the first category must be rendered last to end up on top + for (int i = _profileCategories.Count - 1; i > -1; i--) + { + ProfileCategory profileCategory = _profileCategories[i]; + for (int j = profileCategory.ProfileConfigurations.Count - 1; j > -1; j--) + { + ProfileConfiguration profileConfiguration = profileCategory.ProfileConfigurations[j]; + + // Process hotkeys that where pressed since this profile last updated + ProcessPendingKeyEvents(profileConfiguration); + + bool shouldBeActive = profileConfiguration.ShouldBeActive(false); + if (shouldBeActive) + { + profileConfiguration.Update(); + shouldBeActive = profileConfiguration.ActivationConditionMet; + } + + try + { + // Make sure the profile is active or inactive according to the parameters above + if (shouldBeActive && profileConfiguration.Profile == null && profileConfiguration.BrokenState != "Failed to activate profile") + profileConfiguration.TryOrBreak(() => ActivateProfile(profileConfiguration), "Failed to activate profile"); + if (shouldBeActive && profileConfiguration.Profile != null && !profileConfiguration.Profile.ShouldDisplay) + profileConfiguration.Profile.ShouldDisplay = true; + else if (!shouldBeActive && profileConfiguration.Profile != null) + { + if (!profileConfiguration.FadeInAndOut) + DeactivateProfile(profileConfiguration); + else if (!profileConfiguration.Profile.ShouldDisplay && profileConfiguration.Profile.Opacity <= 0) + DeactivateProfile(profileConfiguration); + else if (profileConfiguration.Profile.Opacity > 0) + RequestDeactivation(profileConfiguration); + } + + profileConfiguration.Profile?.Update(deltaTime); + } + catch (Exception e) + { + _updateExceptions.Add(e); + } + } + } + + LogProfileUpdateExceptions(); + _pendingKeyboardEvents.Clear(); + } + } + + /// + public void RenderProfiles(SKCanvas canvas) + { + // If there is a focus profile, render only that + if (FocusProfile != null) + { + FocusProfile.Profile?.Render(canvas, SKPointI.Empty, FocusProfileElement); + return; + } + + lock (_profileCategories) + { + // Iterate the children in reverse because the first category must be rendered last to end up on top + for (int i = _profileCategories.Count - 1; i > -1; i--) + { + ProfileCategory profileCategory = _profileCategories[i]; + for (int j = profileCategory.ProfileConfigurations.Count - 1; j > -1; j--) + { + try + { + ProfileConfiguration profileConfiguration = profileCategory.ProfileConfigurations[j]; + // Ensure all criteria are met before rendering + bool fadingOut = profileConfiguration.Profile?.ShouldDisplay == false && profileConfiguration.Profile?.Opacity > 0; + if (!profileConfiguration.IsSuspended && !profileConfiguration.IsMissingModule && (profileConfiguration.ActivationConditionMet || fadingOut)) + profileConfiguration.Profile?.Render(canvas, SKPointI.Empty, null); + } + catch (Exception e) + { + _renderExceptions.Add(e); + } + } + } + + LogProfileRenderExceptions(); + } + } + + /// + public ReadOnlyCollection ProfileCategories + { + get + { + lock (_profileRepository) + { + return _profileCategories.AsReadOnly(); + } + } + } + + /// + public ReadOnlyCollection ProfileConfigurations + { + get + { + lock (_profileRepository) + { + return _profileCategories.SelectMany(c => c.ProfileConfigurations).ToList().AsReadOnly(); + } + } + } + + /// + public void LoadProfileConfigurationIcon(ProfileConfiguration profileConfiguration) + { + if (profileConfiguration.Icon.IconType == ProfileConfigurationIconType.MaterialIcon) return; + using Stream? stream = _profileCategoryRepository.GetProfileIconStream(profileConfiguration.Entity.FileIconId); + if (stream != null) + profileConfiguration.Icon.SetIconByStream(stream); + } + + /// + public void SaveProfileConfigurationIcon(ProfileConfiguration profileConfiguration) + { + if (profileConfiguration.Icon.IconType == ProfileConfigurationIconType.MaterialIcon) + return; + + using Stream? stream = profileConfiguration.Icon.GetIconStream(); + if (stream != null) + _profileCategoryRepository.SaveProfileIconStream(profileConfiguration.Entity, stream); + } + + /// + public ProfileConfiguration CloneProfileConfiguration(ProfileConfiguration profileConfiguration) + { + return new ProfileConfiguration(profileConfiguration.Category, profileConfiguration.Entity); + } + + /// + public Profile ActivateProfile(ProfileConfiguration profileConfiguration) + { + if (profileConfiguration.Profile != null) + { + profileConfiguration.Profile.ShouldDisplay = true; + return profileConfiguration.Profile; + } + + ProfileEntity profileEntity; + try + { + profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); + } + catch (Exception e) + { + profileConfiguration.SetBrokenState("Failed to activate profile", e); + throw; + } + + if (profileEntity == null) + throw new ArtemisCoreException($"Cannot find profile named: {profileConfiguration.Name} ID: {profileConfiguration.Entity.ProfileId}"); + + Profile profile = new(profileConfiguration, profileEntity); + profile.PopulateLeds(_rgbService.EnabledDevices); + + if (profile.IsFreshImport) + { + _logger.Debug("Profile is a fresh import, adapting to surface - {profile}", profile); + AdaptProfile(profile); + } + + profileConfiguration.Profile = profile; + + OnProfileActivated(new ProfileConfigurationEventArgs(profileConfiguration)); + return profile; + } + + /// + public void DeactivateProfile(ProfileConfiguration profileConfiguration) + { + if (FocusProfile == profileConfiguration) + throw new ArtemisCoreException("Cannot disable a profile that is being edited, that's rude"); + if (profileConfiguration.Profile == null) + return; + + Profile profile = profileConfiguration.Profile; + profileConfiguration.Profile = null; + profile.Dispose(); + + OnProfileDeactivated(new ProfileConfigurationEventArgs(profileConfiguration)); + } + + private void RequestDeactivation(ProfileConfiguration profileConfiguration) + { + if (FocusProfile == profileConfiguration) + throw new ArtemisCoreException("Cannot disable a profile that is being edited, that's rude"); + if (profileConfiguration.Profile == null) + return; + + profileConfiguration.Profile.ShouldDisplay = false; + } + + /// + public void DeleteProfile(ProfileConfiguration profileConfiguration) + { + DeactivateProfile(profileConfiguration); + + ProfileEntity profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); + if (profileEntity == null) + return; + + profileConfiguration.Category.RemoveProfileConfiguration(profileConfiguration); + _profileRepository.Remove(profileEntity); + SaveProfileCategory(profileConfiguration.Category); + } + + /// + public ProfileCategory CreateProfileCategory(string name, bool addToTop = false) + { + ProfileCategory profileCategory; + lock (_profileRepository) + { + if (addToTop) + { + profileCategory = new ProfileCategory(name, 1); + foreach (ProfileCategory category in _profileCategories) + { + category.Order++; + category.Save(); + _profileCategoryRepository.Save(category.Entity); + } + } + else + { + profileCategory = new ProfileCategory(name, _profileCategories.Count + 1); + } + + _profileCategories.Add(profileCategory); + SaveProfileCategory(profileCategory); + } + + OnProfileCategoryAdded(new ProfileCategoryEventArgs(profileCategory)); + return profileCategory; + } + + /// + public void DeleteProfileCategory(ProfileCategory profileCategory) + { + List profileConfigurations = profileCategory.ProfileConfigurations.ToList(); + foreach (ProfileConfiguration profileConfiguration in profileConfigurations) + RemoveProfileConfiguration(profileConfiguration); + + lock (_profileRepository) + { + _profileCategories.Remove(profileCategory); + _profileCategoryRepository.Remove(profileCategory.Entity); + } + + OnProfileCategoryRemoved(new ProfileCategoryEventArgs(profileCategory)); + } + + /// + public ProfileConfiguration CreateProfileConfiguration(ProfileCategory category, string name, string icon) + { + ProfileConfiguration configuration = new(category, name, icon); + ProfileEntity entity = new(); + _profileRepository.Add(entity); + + configuration.Entity.ProfileId = entity.Id; + category.AddProfileConfiguration(configuration, 0); + return configuration; + } + + /// + public void RemoveProfileConfiguration(ProfileConfiguration profileConfiguration) + { + profileConfiguration.Category.RemoveProfileConfiguration(profileConfiguration); + + DeactivateProfile(profileConfiguration); + SaveProfileCategory(profileConfiguration.Category); + ProfileEntity profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); + if (profileEntity != null) + _profileRepository.Remove(profileEntity); + + profileConfiguration.Dispose(); + } + + /// + public void SaveProfileCategory(ProfileCategory profileCategory) + { + profileCategory.Save(); + _profileCategoryRepository.Save(profileCategory.Entity); + + lock (_profileCategories) + { + _profileCategories.Sort((a, b) => a.Order - b.Order); + } + } + + /// + public void SaveProfile(Profile profile, bool includeChildren) + { + _logger.Debug("Updating profile - Saving {Profile}", profile); + profile.Save(); + if (includeChildren) + { + foreach (RenderProfileElement child in profile.GetAllRenderElements()) + child.Save(); + } + + // At this point the user made actual changes, save that + profile.IsFreshImport = false; + profile.ProfileEntity.IsFreshImport = false; + + _profileRepository.Save(profile.ProfileEntity); + + // If the provided profile is external (cloned or from the workshop?) but it is loaded locally too, reload the local instance + // A bit dodge but it ensures local instances always represent the latest stored version + ProfileConfiguration? localInstance = ProfileConfigurations.FirstOrDefault(p => p.Profile != null && p.Profile != profile && p.ProfileId == profile.ProfileEntity.Id); + if (localInstance == null) + return; + DeactivateProfile(localInstance); + ActivateProfile(localInstance); + } + + /// + public async Task ExportProfile(ProfileConfiguration profileConfiguration) + { + ProfileEntity? profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); + if (profileEntity == null) + throw new ArtemisCoreException("Could not locate profile entity"); + + string configurationJson = JsonConvert.SerializeObject(profileConfiguration.Entity, IProfileService.ExportSettings); + string profileJson = JsonConvert.SerializeObject(profileEntity, IProfileService.ExportSettings); + + MemoryStream archiveStream = new(); + + // Create a ZIP archive + using (ZipArchive archive = new(archiveStream, ZipArchiveMode.Create, true)) + { + ZipArchiveEntry configurationEntry = archive.CreateEntry("configuration.json"); + await using (Stream entryStream = configurationEntry.Open()) + { + await entryStream.WriteAsync(Encoding.Default.GetBytes(configurationJson)); + } + + ZipArchiveEntry profileEntry = archive.CreateEntry("profile.json"); + await using (Stream entryStream = profileEntry.Open()) + { + await entryStream.WriteAsync(Encoding.Default.GetBytes(profileJson)); + } + + await using Stream? iconStream = profileConfiguration.Icon.GetIconStream(); + if (iconStream != null) + { + ZipArchiveEntry iconEntry = archive.CreateEntry("icon.png"); + await using Stream entryStream = iconEntry.Open(); + await iconStream.CopyToAsync(entryStream); + } + } + + archiveStream.Seek(0, SeekOrigin.Begin); + return archiveStream; + } + + /// + public async Task ImportProfile(Stream archiveStream, ProfileCategory category, bool makeUnique, bool markAsFreshImport, string? nameAffix, int targetIndex = 0) + { + using ZipArchive archive = new(archiveStream, ZipArchiveMode.Read, true); + + // There should be a configuration.json and profile.json + ZipArchiveEntry? configurationEntry = archive.Entries.FirstOrDefault(e => e.Name.EndsWith("configuration.json")); + ZipArchiveEntry? profileEntry = archive.Entries.FirstOrDefault(e => e.Name.EndsWith("profile.json")); + ZipArchiveEntry? iconEntry = archive.Entries.FirstOrDefault(e => e.Name.EndsWith("icon.png")); + + if (configurationEntry == null) + throw new ArtemisCoreException("Could not import profile, configuration.json missing"); + if (profileEntry == null) + throw new ArtemisCoreException("Could not import profile, profile.json missing"); + + await using Stream configurationStream = configurationEntry.Open(); + using StreamReader configurationReader = new(configurationStream); + ProfileConfigurationEntity? configurationEntity = JsonConvert.DeserializeObject(await configurationReader.ReadToEndAsync(), IProfileService.ExportSettings); + if (configurationEntity == null) + throw new ArtemisCoreException("Could not import profile, failed to deserialize configuration.json"); + + await using Stream profileStream = profileEntry.Open(); + using StreamReader profileReader = new(profileStream); + ProfileEntity? profileEntity = JsonConvert.DeserializeObject(await profileReader.ReadToEndAsync(), IProfileService.ExportSettings); + if (profileEntity == null) + throw new ArtemisCoreException("Could not import profile, failed to deserialize profile.json"); + + // Assign a new GUID to make sure it is unique in case of a previous import of the same content + if (makeUnique) + profileEntity.UpdateGuid(Guid.NewGuid()); + else + { + // If the profile already exists and this one is not to be made unique, return the existing profile + ProfileConfiguration? existing = ProfileCategories.SelectMany(c => c.ProfileConfigurations).FirstOrDefault(p => p.ProfileId == profileEntity.Id); + if (existing != null) + return existing; + } + + if (nameAffix != null) + profileEntity.Name = $"{profileEntity.Name} - {nameAffix}"; + if (markAsFreshImport) + profileEntity.IsFreshImport = true; + + if (_profileRepository.Get(profileEntity.Id) == null) + _profileRepository.Add(profileEntity); + else + throw new ArtemisCoreException($"Cannot import this profile without {nameof(makeUnique)} being true"); + + // A new GUID will be given on save + configurationEntity.FileIconId = Guid.Empty; + ProfileConfiguration profileConfiguration = new(category, configurationEntity); + if (nameAffix != null) + profileConfiguration.Name = $"{profileConfiguration.Name} - {nameAffix}"; + + // If an icon was provided, import that as well + if (iconEntry != null) + { + await using Stream iconStream = iconEntry.Open(); + profileConfiguration.Icon.SetIconByStream(iconStream); + SaveProfileConfigurationIcon(profileConfiguration); + } + + profileConfiguration.Entity.ProfileId = profileEntity.Id; + category.AddProfileConfiguration(profileConfiguration, targetIndex); + + List modules = _pluginManagementService.GetFeaturesOfType(); + profileConfiguration.LoadModules(modules); + SaveProfileCategory(category); + + return profileConfiguration; + } + + /// + public async Task OverwriteProfile(MemoryStream archiveStream, ProfileConfiguration profileConfiguration) + { + ProfileConfiguration imported = await ImportProfile(archiveStream, profileConfiguration.Category, true, true, null, profileConfiguration.Order + 1); + + DeleteProfile(profileConfiguration); + SaveProfileCategory(imported.Category); + + return imported; + } + + /// + public void AdaptProfile(Profile profile) + { + List devices = _rgbService.EnabledDevices.ToList(); + foreach (Layer layer in profile.GetAllLayers()) + layer.Adapter.Adapt(devices); + + profile.Save(); + + foreach (RenderProfileElement renderProfileElement in profile.GetAllRenderElements()) + renderProfileElement.Save(); + + _logger.Debug("Adapt profile - Saving " + profile); + _profileRepository.Save(profile.ProfileEntity); + } + + private void InputServiceOnKeyboardKeyUp(object? sender, ArtemisKeyboardKeyEventArgs e) + { lock (_profileCategories) { _pendingKeyboardEvents.Add(e); @@ -129,7 +605,7 @@ internal class ProfileService : IProfileService profileConfiguration.IsSuspended = true; } } - + // If suspension was changed, save the category if (before != profileConfiguration.IsSuspended) SaveProfileCategory(profileConfiguration.Category); @@ -185,432 +661,6 @@ internal class ProfileService : IProfileService _renderExceptions.Clear(); } - public bool HotkeysEnabled { get; set; } - public bool RenderForEditor { get; set; } - public ProfileElement? EditorFocus { get; set; } - - public void UpdateProfiles(double deltaTime) - { - lock (_profileCategories) - { - // Iterate the children in reverse because the first category must be rendered last to end up on top - for (int i = _profileCategories.Count - 1; i > -1; i--) - { - ProfileCategory profileCategory = _profileCategories[i]; - for (int j = profileCategory.ProfileConfigurations.Count - 1; j > -1; j--) - { - ProfileConfiguration profileConfiguration = profileCategory.ProfileConfigurations[j]; - - // Process hotkeys that where pressed since this profile last updated - ProcessPendingKeyEvents(profileConfiguration); - - // Profiles being edited are updated at their own leisure - if (profileConfiguration.IsBeingEdited && RenderForEditor) - continue; - - bool shouldBeActive = profileConfiguration.ShouldBeActive(false); - if (shouldBeActive) - { - profileConfiguration.Update(); - if (!profileConfiguration.IsBeingEdited) - shouldBeActive = profileConfiguration.ActivationConditionMet; - } - - try - { - // Make sure the profile is active or inactive according to the parameters above - if (shouldBeActive && profileConfiguration.Profile == null && profileConfiguration.BrokenState != "Failed to activate profile") - profileConfiguration.TryOrBreak(() => ActivateProfile(profileConfiguration), "Failed to activate profile"); - if (shouldBeActive && profileConfiguration.Profile != null && !profileConfiguration.Profile.ShouldDisplay) - profileConfiguration.Profile.ShouldDisplay = true; - else if (!shouldBeActive && profileConfiguration.Profile != null) - { - if (!profileConfiguration.FadeInAndOut) - DeactivateProfile(profileConfiguration); - else if (!profileConfiguration.Profile.ShouldDisplay && profileConfiguration.Profile.Opacity <= 0) - DeactivateProfile(profileConfiguration); - else if (profileConfiguration.Profile.Opacity > 0) - RequestDeactivation(profileConfiguration); - } - - profileConfiguration.Profile?.Update(deltaTime); - } - catch (Exception e) - { - _updateExceptions.Add(e); - } - } - } - - LogProfileUpdateExceptions(); - _pendingKeyboardEvents.Clear(); - } - } - - public void RenderProfiles(SKCanvas canvas) - { - lock (_profileCategories) - { - ProfileConfiguration? editedProfileConfiguration = _profileCategories.SelectMany(c => c.ProfileConfigurations).FirstOrDefault(p => p.IsBeingEdited); - if (editedProfileConfiguration != null) - { - editedProfileConfiguration.Profile?.Render(canvas, SKPointI.Empty, RenderForEditor ? EditorFocus : null); - return; - } - - if (RenderForEditor) - return; - - // Iterate the children in reverse because the first category must be rendered last to end up on top - for (int i = _profileCategories.Count - 1; i > -1; i--) - { - ProfileCategory profileCategory = _profileCategories[i]; - for (int j = profileCategory.ProfileConfigurations.Count - 1; j > -1; j--) - { - try - { - ProfileConfiguration profileConfiguration = profileCategory.ProfileConfigurations[j]; - // Ensure all criteria are met before rendering - bool fadingOut = profileConfiguration.Profile?.ShouldDisplay == false && profileConfiguration.Profile?.Opacity > 0; - if (!profileConfiguration.IsSuspended && !profileConfiguration.IsMissingModule && (profileConfiguration.ActivationConditionMet || fadingOut)) - profileConfiguration.Profile?.Render(canvas, SKPointI.Empty, null); - } - catch (Exception e) - { - _renderExceptions.Add(e); - } - } - } - - LogProfileRenderExceptions(); - } - } - - public ReadOnlyCollection ProfileCategories - { - get - { - lock (_profileRepository) - { - return _profileCategories.AsReadOnly(); - } - } - } - - public ReadOnlyCollection ProfileConfigurations - { - get - { - lock (_profileRepository) - { - return _profileCategories.SelectMany(c => c.ProfileConfigurations).ToList().AsReadOnly(); - } - } - } - - public void LoadProfileConfigurationIcon(ProfileConfiguration profileConfiguration) - { - if (profileConfiguration.Icon.IconType == ProfileConfigurationIconType.MaterialIcon) - return; - - using Stream? stream = _profileCategoryRepository.GetProfileIconStream(profileConfiguration.Entity.FileIconId); - if (stream != null) - profileConfiguration.Icon.SetIconByStream(stream); - } - - public void SaveProfileConfigurationIcon(ProfileConfiguration profileConfiguration) - { - if (profileConfiguration.Icon.IconType == ProfileConfigurationIconType.MaterialIcon) - return; - - using Stream? stream = profileConfiguration.Icon.GetIconStream(); - if (stream != null) - { - _profileCategoryRepository.SaveProfileIconStream(profileConfiguration.Entity, stream); - } - } - - public Profile ActivateProfile(ProfileConfiguration profileConfiguration) - { - if (profileConfiguration.Profile != null) - { - profileConfiguration.Profile.ShouldDisplay = true; - return profileConfiguration.Profile; - } - - ProfileEntity profileEntity; - try - { - profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); - } - catch (Exception e) - { - profileConfiguration.SetBrokenState("Failed to activate profile", e); - throw; - } - - if (profileEntity == null) - throw new ArtemisCoreException($"Cannot find profile named: {profileConfiguration.Name} ID: {profileConfiguration.Entity.ProfileId}"); - - Profile profile = new(profileConfiguration, profileEntity); - profile.PopulateLeds(_rgbService.EnabledDevices); - - if (profile.IsFreshImport) - { - _logger.Debug("Profile is a fresh import, adapting to surface - {profile}", profile); - AdaptProfile(profile); - } - - profileConfiguration.Profile = profile; - - OnProfileActivated(new ProfileConfigurationEventArgs(profileConfiguration)); - return profile; - } - - public void DeactivateProfile(ProfileConfiguration profileConfiguration) - { - if (profileConfiguration.IsBeingEdited) - throw new ArtemisCoreException("Cannot disable a profile that is being edited, that's rude"); - if (profileConfiguration.Profile == null) - return; - - Profile profile = profileConfiguration.Profile; - profileConfiguration.Profile = null; - profile.Dispose(); - - OnProfileDeactivated(new ProfileConfigurationEventArgs(profileConfiguration)); - } - - private void RequestDeactivation(ProfileConfiguration profileConfiguration) - { - if (profileConfiguration.IsBeingEdited) - throw new ArtemisCoreException("Cannot disable a profile that is being edited, that's rude"); - if (profileConfiguration.Profile == null) - return; - - profileConfiguration.Profile.ShouldDisplay = false; - } - - public void DeleteProfile(ProfileConfiguration profileConfiguration) - { - DeactivateProfile(profileConfiguration); - - ProfileEntity profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); - if (profileEntity == null) - return; - - profileConfiguration.Category.RemoveProfileConfiguration(profileConfiguration); - _profileRepository.Remove(profileEntity); - SaveProfileCategory(profileConfiguration.Category); - } - - public ProfileCategory CreateProfileCategory(string name) - { - ProfileCategory profileCategory; - lock (_profileRepository) - { - profileCategory = new ProfileCategory(name, _profileCategories.Count + 1); - _profileCategories.Add(profileCategory); - SaveProfileCategory(profileCategory); - } - - OnProfileCategoryAdded(new ProfileCategoryEventArgs(profileCategory)); - return profileCategory; - } - - public void DeleteProfileCategory(ProfileCategory profileCategory) - { - List profileConfigurations = profileCategory.ProfileConfigurations.ToList(); - foreach (ProfileConfiguration profileConfiguration in profileConfigurations) - RemoveProfileConfiguration(profileConfiguration); - - lock (_profileRepository) - { - _profileCategories.Remove(profileCategory); - _profileCategoryRepository.Remove(profileCategory.Entity); - } - - OnProfileCategoryRemoved(new ProfileCategoryEventArgs(profileCategory)); - } - - public ProfileConfiguration CreateProfileConfiguration(ProfileCategory category, string name, string icon) - { - ProfileConfiguration configuration = new(category, name, icon); - ProfileEntity entity = new(); - _profileRepository.Add(entity); - - configuration.Entity.ProfileId = entity.Id; - category.AddProfileConfiguration(configuration, 0); - return configuration; - } - - public void RemoveProfileConfiguration(ProfileConfiguration profileConfiguration) - { - profileConfiguration.Category.RemoveProfileConfiguration(profileConfiguration); - - DeactivateProfile(profileConfiguration); - SaveProfileCategory(profileConfiguration.Category); - ProfileEntity profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); - if (profileEntity != null) - _profileRepository.Remove(profileEntity); - - profileConfiguration.Dispose(); - } - - public void SaveProfileCategory(ProfileCategory profileCategory) - { - profileCategory.Save(); - _profileCategoryRepository.Save(profileCategory.Entity); - - lock (_profileCategories) - { - _profileCategories.Sort((a, b) => a.Order - b.Order); - } - } - - public void SaveProfile(Profile profile, bool includeChildren) - { - _logger.Debug("Updating profile - Saving {Profile}", profile); - profile.Save(); - if (includeChildren) - foreach (RenderProfileElement child in profile.GetAllRenderElements()) - child.Save(); - - // At this point the user made actual changes, save that - profile.IsFreshImport = false; - profile.ProfileEntity.IsFreshImport = false; - - _profileRepository.Save(profile.ProfileEntity); - } - - /// - public async Task ExportProfile(ProfileConfiguration profileConfiguration) - { - ProfileEntity? profileEntity = _profileRepository.Get(profileConfiguration.Entity.ProfileId); - if (profileEntity == null) - throw new ArtemisCoreException("Could not locate profile entity"); - - string configurationJson = JsonConvert.SerializeObject(profileConfiguration.Entity, IProfileService.ExportSettings); - string profileJson = JsonConvert.SerializeObject(profileEntity, IProfileService.ExportSettings); - - MemoryStream archiveStream = new(); - - // Create a ZIP archive - using (ZipArchive archive = new(archiveStream, ZipArchiveMode.Create, true)) - { - ZipArchiveEntry configurationEntry = archive.CreateEntry("configuration.json"); - await using (Stream entryStream = configurationEntry.Open()) - { - await entryStream.WriteAsync(Encoding.Default.GetBytes(configurationJson)); - } - - ZipArchiveEntry profileEntry = archive.CreateEntry("profile.json"); - await using (Stream entryStream = profileEntry.Open()) - { - await entryStream.WriteAsync(Encoding.Default.GetBytes(profileJson)); - } - - await using Stream? iconStream = profileConfiguration.Icon.GetIconStream(); - if (iconStream != null) - { - ZipArchiveEntry iconEntry = archive.CreateEntry("icon.png"); - await using Stream entryStream = iconEntry.Open(); - await iconStream.CopyToAsync(entryStream); - } - } - - archiveStream.Seek(0, SeekOrigin.Begin); - return archiveStream; - } - - /// - public async Task ImportProfile(Stream archiveStream, ProfileCategory category, bool makeUnique, bool markAsFreshImport, string? nameAffix) - { - using ZipArchive archive = new(archiveStream, ZipArchiveMode.Read, true); - - // There should be a configuration.json and profile.json - ZipArchiveEntry? configurationEntry = archive.Entries.FirstOrDefault(e => e.Name.EndsWith("configuration.json")); - ZipArchiveEntry? profileEntry = archive.Entries.FirstOrDefault(e => e.Name.EndsWith("profile.json")); - ZipArchiveEntry? iconEntry = archive.Entries.FirstOrDefault(e => e.Name.EndsWith("icon.png")); - - if (configurationEntry == null) - throw new ArtemisCoreException("Could not import profile, configuration.json missing"); - if (profileEntry == null) - throw new ArtemisCoreException("Could not import profile, profile.json missing"); - - await using Stream configurationStream = configurationEntry.Open(); - using StreamReader configurationReader = new(configurationStream); - ProfileConfigurationEntity? configurationEntity = JsonConvert.DeserializeObject(await configurationReader.ReadToEndAsync(), IProfileService.ExportSettings); - if (configurationEntity == null) - throw new ArtemisCoreException("Could not import profile, failed to deserialize configuration.json"); - - await using Stream profileStream = profileEntry.Open(); - using StreamReader profileReader = new(profileStream); - ProfileEntity? profileEntity = JsonConvert.DeserializeObject(await profileReader.ReadToEndAsync(), IProfileService.ExportSettings); - if (profileEntity == null) - throw new ArtemisCoreException("Could not import profile, failed to deserialize profile.json"); - - // Assign a new GUID to make sure it is unique in case of a previous import of the same content - if (makeUnique) - profileEntity.UpdateGuid(Guid.NewGuid()); - else - { - // If the profile already exists and this one is not to be made unique, return the existing profile - ProfileConfiguration? existing = ProfileCategories.SelectMany(c => c.ProfileConfigurations).FirstOrDefault(p => p.ProfileId == profileEntity.Id); - if (existing != null) - return existing; - } - - if (nameAffix != null) - profileEntity.Name = $"{profileEntity.Name} - {nameAffix}"; - if (markAsFreshImport) - profileEntity.IsFreshImport = true; - - if (_profileRepository.Get(profileEntity.Id) == null) - _profileRepository.Add(profileEntity); - else - throw new ArtemisCoreException($"Cannot import this profile without {nameof(makeUnique)} being true"); - - // A new GUID will be given on save - configurationEntity.FileIconId = Guid.Empty; - ProfileConfiguration profileConfiguration = new(category, configurationEntity); - if (nameAffix != null) - profileConfiguration.Name = $"{profileConfiguration.Name} - {nameAffix}"; - - // If an icon was provided, import that as well - if (iconEntry != null) - { - await using Stream iconStream = iconEntry.Open(); - profileConfiguration.Icon.SetIconByStream(iconStream); - SaveProfileConfigurationIcon(profileConfiguration); - } - - profileConfiguration.Entity.ProfileId = profileEntity.Id; - category.AddProfileConfiguration(profileConfiguration, 0); - - List modules = _pluginManagementService.GetFeaturesOfType(); - profileConfiguration.LoadModules(modules); - SaveProfileCategory(category); - - return profileConfiguration; - } - - /// - public void AdaptProfile(Profile profile) - { - List devices = _rgbService.EnabledDevices.ToList(); - foreach (Layer layer in profile.GetAllLayers()) - layer.Adapter.Adapt(devices); - - profile.Save(); - - foreach (RenderProfileElement renderProfileElement in profile.GetAllRenderElements()) - renderProfileElement.Save(); - - _logger.Debug("Adapt profile - Saving " + profile); - _profileRepository.Save(profile.ProfileEntity); - } - #region Events public event EventHandler? ProfileActivated; diff --git a/src/Artemis.Storage/Entities/Profile/ProfileEntity.cs b/src/Artemis.Storage/Entities/Profile/ProfileEntity.cs index 8a8616ce7..da5bbb903 100644 --- a/src/Artemis.Storage/Entities/Profile/ProfileEntity.cs +++ b/src/Artemis.Storage/Entities/Profile/ProfileEntity.cs @@ -18,7 +18,6 @@ public class ProfileEntity public string Name { get; set; } public bool IsFreshImport { get; set; } - public Guid LastSelectedProfileElement { get; set; } public List Folders { get; set; } public List Layers { get; set; } diff --git a/src/Artemis.Storage/Entities/Workshop/EntryEntity.cs b/src/Artemis.Storage/Entities/Workshop/EntryEntity.cs new file mode 100644 index 000000000..86bff6dcd --- /dev/null +++ b/src/Artemis.Storage/Entities/Workshop/EntryEntity.cs @@ -0,0 +1,21 @@ +using System; + +namespace Artemis.Storage.Entities.Workshop; + +public class EntryEntity +{ + public Guid Id { get; set; } + + public long EntryId { get; set; } + public int EntryType { get; set; } + + public string Author { get; set; } + public string Name { get; set; } = string.Empty; + public string Summary { get; set; } = string.Empty; + + public long ReleaseId { get; set; } + public string ReleaseVersion { get; set; } + public DateTimeOffset InstalledAt { get; set; } + + public string LocalReference { get; set; } +} \ No newline at end of file diff --git a/src/Artemis.Storage/Repositories/EntryRepository.cs b/src/Artemis.Storage/Repositories/EntryRepository.cs new file mode 100644 index 000000000..61d3da672 --- /dev/null +++ b/src/Artemis.Storage/Repositories/EntryRepository.cs @@ -0,0 +1,54 @@ +using System; +using System.Collections.Generic; +using Artemis.Storage.Entities.Workshop; +using Artemis.Storage.Repositories.Interfaces; +using LiteDB; + +namespace Artemis.Storage.Repositories; + +internal class EntryRepository : IEntryRepository +{ + private readonly LiteRepository _repository; + + public EntryRepository(LiteRepository repository) + { + _repository = repository; + _repository.Database.GetCollection().EnsureIndex(s => s.Id); + _repository.Database.GetCollection().EnsureIndex(s => s.EntryId); + } + + public void Add(EntryEntity entryEntity) + { + _repository.Insert(entryEntity); + } + + public void Remove(EntryEntity entryEntity) + { + _repository.Delete(entryEntity.Id); + } + + public EntryEntity Get(Guid id) + { + return _repository.FirstOrDefault(s => s.Id == id); + } + + public EntryEntity GetByEntryId(long entryId) + { + return _repository.FirstOrDefault(s => s.EntryId == entryId); + } + + public List GetAll() + { + return _repository.Query().ToList(); + } + + public void Save(EntryEntity entryEntity) + { + _repository.Upsert(entryEntity); + } + + public void Save(IEnumerable entryEntities) + { + _repository.Upsert(entryEntities); + } +} \ No newline at end of file diff --git a/src/Artemis.Storage/Repositories/Interfaces/IEntryRepository.cs b/src/Artemis.Storage/Repositories/Interfaces/IEntryRepository.cs new file mode 100644 index 000000000..1e9ecd1c5 --- /dev/null +++ b/src/Artemis.Storage/Repositories/Interfaces/IEntryRepository.cs @@ -0,0 +1,16 @@ +using System; +using System.Collections.Generic; +using Artemis.Storage.Entities.Workshop; + +namespace Artemis.Storage.Repositories.Interfaces; + +public interface IEntryRepository : IRepository +{ + void Add(EntryEntity entryEntity); + void Remove(EntryEntity entryEntity); + EntryEntity Get(Guid id); + EntryEntity GetByEntryId(long entryId); + List GetAll(); + void Save(EntryEntity entryEntity); + void Save(IEnumerable entryEntities); +} \ No newline at end of file diff --git a/src/Artemis.UI.Linux/App.axaml b/src/Artemis.UI.Linux/App.axaml index a63d617b9..d2109667a 100644 --- a/src/Artemis.UI.Linux/App.axaml +++ b/src/Artemis.UI.Linux/App.axaml @@ -18,7 +18,7 @@ - + diff --git a/src/Artemis.UI.MacOS/App.axaml b/src/Artemis.UI.MacOS/App.axaml index 6ef3efd97..5a000915e 100644 --- a/src/Artemis.UI.MacOS/App.axaml +++ b/src/Artemis.UI.MacOS/App.axaml @@ -18,7 +18,7 @@ - + diff --git a/src/Artemis.UI.Shared/Artemis.UI.Shared.csproj b/src/Artemis.UI.Shared/Artemis.UI.Shared.csproj index d6c8ca8c9..5e84fa443 100644 --- a/src/Artemis.UI.Shared/Artemis.UI.Shared.csproj +++ b/src/Artemis.UI.Shared/Artemis.UI.Shared.csproj @@ -15,6 +15,7 @@ + diff --git a/src/Artemis.UI.Shared/Controls/NotificationHost.cs b/src/Artemis.UI.Shared/Controls/NotificationHost.cs new file mode 100644 index 000000000..4dce59804 --- /dev/null +++ b/src/Artemis.UI.Shared/Controls/NotificationHost.cs @@ -0,0 +1,54 @@ +using System; +using Avalonia; +using Avalonia.Controls; +using Avalonia.Controls.Primitives; +using Avalonia.Layout; + +namespace Artemis.UI.Shared; + +internal class NotificationHost : ContentControl +{ + private IDisposable? _rootBoundsWatcher; + + public NotificationHost() + { + Background = null; + HorizontalAlignment = HorizontalAlignment.Center; + VerticalAlignment = VerticalAlignment.Center; + } + + protected override Type StyleKeyOverride => typeof(OverlayPopupHost); + + protected override Size MeasureOverride(Size availableSize) + { + _ = base.MeasureOverride(availableSize); + + if (VisualRoot is TopLevel tl) + return tl.ClientSize; + if (VisualRoot is Control c) + return c.Bounds.Size; + + return default; + } + + protected override void OnAttachedToVisualTree(VisualTreeAttachmentEventArgs e) + { + base.OnAttachedToVisualTree(e); + if (e.Root is Control wb) + // OverlayLayer is a Canvas, so we won't get a signal to resize if the window + // bounds change. Subscribe to force update + _rootBoundsWatcher = wb.GetObservable(BoundsProperty).Subscribe(_ => OnRootBoundsChanged()); + } + + protected override void OnDetachedFromVisualTree(VisualTreeAttachmentEventArgs e) + { + base.OnDetachedFromVisualTree(e); + _rootBoundsWatcher?.Dispose(); + _rootBoundsWatcher = null; + } + + private void OnRootBoundsChanged() + { + InvalidateMeasure(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Controls/TagsInput/TagsInput.cs b/src/Artemis.UI.Shared/Controls/TagsInput/TagsInput.cs new file mode 100644 index 000000000..3b94d3766 --- /dev/null +++ b/src/Artemis.UI.Shared/Controls/TagsInput/TagsInput.cs @@ -0,0 +1,78 @@ +using System.Text.RegularExpressions; +using System.Windows.Input; +using Avalonia.Controls; +using Avalonia.Controls.Metadata; +using Avalonia.Controls.Primitives; +using Avalonia.Input; +using ReactiveUI; + +namespace Artemis.UI.Shared.TagsInput; + +[TemplatePart("PART_TagInputBox", typeof(TextBox))] +public partial class TagsInput : TemplatedControl +{ + public TextBox? TagInputBox { get; set; } + public ICommand RemoveTag { get; } + + /// + public TagsInput() + { + RemoveTag = ReactiveCommand.Create(ExecuteRemoveTag); + } + + /// + protected override void OnApplyTemplate(TemplateAppliedEventArgs e) + { + if (TagInputBox != null) + { + TagInputBox.KeyDown -= TagInputBoxOnKeyDown; + TagInputBox.TextChanging -= TagInputBoxOnTextChanging; + } + + TagInputBox = e.NameScope.Find("PART_TagInputBox"); + + if (TagInputBox != null) + { + TagInputBox.KeyDown += TagInputBoxOnKeyDown; + TagInputBox.TextChanging += TagInputBoxOnTextChanging; + } + } + + private void ExecuteRemoveTag(string t) + { + Tags.Remove(t); + + if (TagInputBox != null) + TagInputBox.IsEnabled = Tags.Count < MaxLength; + } + + private void TagInputBoxOnTextChanging(object? sender, TextChangingEventArgs e) + { + if (TagInputBox?.Text == null) + return; + + TagInputBox.Text = CleanTagRegex().Replace(TagInputBox.Text.ToLower(), ""); + } + + private void TagInputBoxOnKeyDown(object? sender, KeyEventArgs e) + { + if (TagInputBox == null) + return; + + if (e.Key == Key.Space) + e.Handled = true; + if (e.Key != Key.Enter) + return; + + if (string.IsNullOrWhiteSpace(TagInputBox.Text) || Tags.Contains(TagInputBox.Text) || Tags.Count >= MaxLength) + return; + + Tags.Add(CleanTagRegex().Replace(TagInputBox.Text.ToLower(), "")); + + TagInputBox.Text = ""; + TagInputBox.IsEnabled = Tags.Count < MaxLength; + } + + [GeneratedRegex("[\\s\\-]+")] + private static partial Regex CleanTagRegex(); +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Controls/TagsInput/TagsInput.properties.cs b/src/Artemis.UI.Shared/Controls/TagsInput/TagsInput.properties.cs new file mode 100644 index 000000000..2660e7bb8 --- /dev/null +++ b/src/Artemis.UI.Shared/Controls/TagsInput/TagsInput.properties.cs @@ -0,0 +1,54 @@ +using System.Collections.ObjectModel; +using Avalonia; +using Avalonia.Controls.Primitives; +using Avalonia.Data; + +namespace Artemis.UI.Shared.TagsInput; + +public partial class TagsInput : TemplatedControl +{ + /// + /// Defines the property + /// + public static readonly StyledProperty> TagsProperty = + AvaloniaProperty.Register>(nameof(Tags), new ObservableCollection()); + + /// + /// Gets or sets the selected tags. + /// + public ObservableCollection Tags + { + get => GetValue(TagsProperty); + set => SetValue(TagsProperty, value); + } + + /// + /// Defines the property + /// + public static readonly StyledProperty MaxLengthProperty = + AvaloniaProperty.Register(nameof(MaxLength), 20); + + /// + /// Gets or sets the max length of each tag + /// + public int MaxLength + { + get => GetValue(MaxLengthProperty); + set => SetValue(MaxLengthProperty, value); + } + + /// + /// Defines the property + /// + public static readonly StyledProperty MaxTagsProperty = + AvaloniaProperty.Register(nameof(MaxTags), 20); + + /// + /// Gets or sets the max amount of tags to be added + /// + public int MaxTags + { + get => GetValue(MaxTagsProperty); + set => SetValue(MaxTagsProperty, value); + } +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Controls/TagsInput/TagsInputStyles.axaml b/src/Artemis.UI.Shared/Controls/TagsInput/TagsInputStyles.axaml new file mode 100644 index 000000000..e2a0ff8e5 --- /dev/null +++ b/src/Artemis.UI.Shared/Controls/TagsInput/TagsInputStyles.axaml @@ -0,0 +1,49 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Extensions/HttpClientExtensions.cs b/src/Artemis.UI.Shared/Extensions/HttpClientExtensions.cs similarity index 78% rename from src/Artemis.UI/Extensions/HttpClientExtensions.cs rename to src/Artemis.UI.Shared/Extensions/HttpClientExtensions.cs index 50af33443..d9adda42f 100644 --- a/src/Artemis.UI/Extensions/HttpClientExtensions.cs +++ b/src/Artemis.UI.Shared/Extensions/HttpClientExtensions.cs @@ -3,12 +3,13 @@ using System.IO; using System.Net.Http; using System.Threading; using System.Threading.Tasks; +using Artemis.UI.Shared.Utilities; -namespace Artemis.UI.Extensions +namespace Artemis.UI.Shared.Extensions { public static class HttpClientProgressExtensions { - public static async Task DownloadDataAsync(this HttpClient client, string requestUrl, Stream destination, IProgress? progress, CancellationToken cancellationToken) + public static async Task DownloadDataAsync(this HttpClient client, string requestUrl, Stream destination, IProgress? progress, CancellationToken cancellationToken) { using HttpResponseMessage response = await client.GetAsync(requestUrl, HttpCompletionOption.ResponseHeadersRead, cancellationToken); response.EnsureSuccessStatusCode(); @@ -23,13 +24,10 @@ namespace Artemis.UI.Extensions } // Such progress and contentLength much reporting Wow! - Progress progressWrapper = new(totalBytes => progress.Report(GetProgressPercentage(totalBytes, contentLength.Value))); - await download.CopyToAsync(destination, 81920, progressWrapper, cancellationToken); - - float GetProgressPercentage(float totalBytes, float currentBytes) => (totalBytes / currentBytes) * 100f; + await download.CopyToAsync(destination, 81920, progress, contentLength, cancellationToken); } - static async Task CopyToAsync(this Stream source, Stream destination, int bufferSize, IProgress progress, CancellationToken cancellationToken) + static async Task CopyToAsync(this Stream source, Stream destination, int bufferSize, IProgress progress, long? contentLength, CancellationToken cancellationToken) { if (bufferSize < 0) throw new ArgumentOutOfRangeException(nameof(bufferSize)); @@ -49,7 +47,7 @@ namespace Artemis.UI.Extensions { await destination.WriteAsync(buffer, 0, bytesRead, cancellationToken).ConfigureAwait(false); totalBytesRead += bytesRead; - progress?.Report(totalBytesRead); + progress?.Report(new StreamProgress(totalBytesRead, contentLength ?? totalBytesRead)); } } } diff --git a/src/Artemis.UI.Shared/Routing/Routable/IRoutableHostScreen.cs b/src/Artemis.UI.Shared/Routing/Routable/IRoutableHostScreen.cs new file mode 100644 index 000000000..b09afb676 --- /dev/null +++ b/src/Artemis.UI.Shared/Routing/Routable/IRoutableHostScreen.cs @@ -0,0 +1,13 @@ +namespace Artemis.UI.Shared.Routing; + +/// +/// For internal use. +/// +/// +/// +internal interface IRoutableHostScreen : IRoutableScreen +{ + bool RecycleScreen { get; } + IRoutableScreen? InternalScreen { get; } + void InternalChangeScreen(IRoutableScreen? screen); +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Routing/Routable/IRoutableScreen.cs b/src/Artemis.UI.Shared/Routing/Routable/IRoutableScreen.cs new file mode 100644 index 000000000..3b28384cd --- /dev/null +++ b/src/Artemis.UI.Shared/Routing/Routable/IRoutableScreen.cs @@ -0,0 +1,14 @@ +using System.Threading; +using System.Threading.Tasks; +using ReactiveUI; + +namespace Artemis.UI.Shared.Routing; + +/// +/// For internal use. +/// +internal interface IRoutableScreen : IActivatableViewModel +{ + Task InternalOnNavigating(NavigationArguments args, CancellationToken cancellationToken); + Task InternalOnClosing(NavigationArguments args); +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Routing/Routable/RoutableHostScreenOfTScreen.cs b/src/Artemis.UI.Shared/Routing/Routable/RoutableHostScreenOfTScreen.cs new file mode 100644 index 000000000..c61c27820 --- /dev/null +++ b/src/Artemis.UI.Shared/Routing/Routable/RoutableHostScreenOfTScreen.cs @@ -0,0 +1,42 @@ +namespace Artemis.UI.Shared.Routing; + +/// +/// Represents a view model to which routing can take place and which in turn can host another view model. +/// +/// The type of view model the screen can host. +public abstract class RoutableHostScreen : RoutableScreen, IRoutableHostScreen where TScreen : RoutableScreen +{ + private bool _recycleScreen = true; + private TScreen? _screen; + + /// + /// Gets the currently active child screen. + /// + public TScreen? Screen + { + get => _screen; + private set => RaiseAndSetIfChanged(ref _screen, value); + } + + /// + public bool RecycleScreen + { + get => _recycleScreen; + protected set => RaiseAndSetIfChanged(ref _recycleScreen, value); + } + + IRoutableScreen? IRoutableHostScreen.InternalScreen => Screen; + + void IRoutableHostScreen.InternalChangeScreen(IRoutableScreen? screen) + { + if (screen == null) + { + Screen = null; + return; + } + + if (screen is not TScreen typedScreen) + throw new ArtemisRoutingException($"Screen cannot be hosted, {screen.GetType().Name} is not assignable to {typeof(TScreen).Name}."); + Screen = typedScreen; + } +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Routing/Routable/RoutableHostScreenOfTScreenTParam.cs b/src/Artemis.UI.Shared/Routing/Routable/RoutableHostScreenOfTScreenTParam.cs new file mode 100644 index 000000000..89773e8d4 --- /dev/null +++ b/src/Artemis.UI.Shared/Routing/Routable/RoutableHostScreenOfTScreenTParam.cs @@ -0,0 +1,44 @@ +namespace Artemis.UI.Shared.Routing; + +/// +/// Represents a view model to which routing with parameters can take place and which in turn can host another view +/// model. +/// +/// The type of view model the screen can host. +/// The type of parameters the screen expects. It must have a parameterless constructor. +public abstract class RoutableHostScreen : RoutableScreen, IRoutableHostScreen where TScreen : RoutableScreen where TParam : new() +{ + private bool _recycleScreen = true; + private TScreen? _screen; + + /// + /// Gets the currently active child screen. + /// + public TScreen? Screen + { + get => _screen; + private set => RaiseAndSetIfChanged(ref _screen, value); + } + + /// + public bool RecycleScreen + { + get => _recycleScreen; + protected set => RaiseAndSetIfChanged(ref _recycleScreen, value); + } + + IRoutableScreen? IRoutableHostScreen.InternalScreen => Screen; + + void IRoutableHostScreen.InternalChangeScreen(IRoutableScreen? screen) + { + if (screen == null) + { + Screen = null; + return; + } + + if (screen is not TScreen typedScreen) + throw new ArtemisRoutingException($"Screen cannot be hosted, {screen.GetType().Name} is not assignable to {typeof(TScreen).Name}."); + Screen = typedScreen; + } +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Routing/Routable/RoutableScreen.cs b/src/Artemis.UI.Shared/Routing/Routable/RoutableScreen.cs index 53973de46..7db5dbe2e 100644 --- a/src/Artemis.UI.Shared/Routing/Routable/RoutableScreen.cs +++ b/src/Artemis.UI.Shared/Routing/Routable/RoutableScreen.cs @@ -1,24 +1,55 @@ using System.Threading; using System.Threading.Tasks; -using ReactiveUI; namespace Artemis.UI.Shared.Routing; /// -/// For internal use. +/// Represents a view model to which routing can take place. /// -/// -/// -internal interface IRoutableScreen : IActivatableViewModel +public abstract class RoutableScreen : ActivatableViewModelBase, IRoutableScreen { /// - /// Gets or sets a value indicating whether or not to reuse the child screen instance if the type has not changed. + /// Called before navigating to this screen. /// - /// Defaults to . - bool RecycleScreen { get; } + /// Navigation arguments containing information about the navigation action. + public virtual Task BeforeNavigating(NavigationArguments args) + { + return Task.CompletedTask; + } - object? InternalScreen { get; } - void InternalChangeScreen(object? screen); - Task InternalOnNavigating(NavigationArguments args, CancellationToken cancellationToken); - Task InternalOnClosing(NavigationArguments args); + /// + /// Called while navigating to this screen. + /// + /// Navigation arguments containing information about the navigation action. + /// + /// A cancellation token that can be used by other objects or threads to receive notice of + /// cancellation. + /// + public virtual Task OnNavigating(NavigationArguments args, CancellationToken cancellationToken) + { + return Task.CompletedTask; + } + + /// + /// Called before navigating away from this screen. + /// + /// Navigation arguments containing information about the navigation action. + public virtual Task OnClosing(NavigationArguments args) + { + return Task.CompletedTask; + } + + #region Overrides of RoutableScreen + + async Task IRoutableScreen.InternalOnNavigating(NavigationArguments args, CancellationToken cancellationToken) + { + await OnNavigating(args, cancellationToken); + } + + async Task IRoutableScreen.InternalOnClosing(NavigationArguments args) + { + await OnClosing(args); + } + + #endregion } \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTScreenTParam.cs b/src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTParam.cs similarity index 68% rename from src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTScreenTParam.cs rename to src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTParam.cs index ac62129dc..bbe71fe2a 100644 --- a/src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTScreenTParam.cs +++ b/src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTParam.cs @@ -4,39 +4,15 @@ using System.Linq; using System.Linq.Expressions; using System.Threading; using System.Threading.Tasks; -using Avalonia.Platform; namespace Artemis.UI.Shared.Routing; /// -/// Represents a view model to which routing with parameters can take place and which in turn can host another view -/// model. +/// Represents a view model to which routing with parameters can take place. /// -/// The type of view model the screen can host. /// The type of parameters the screen expects. It must have a parameterless constructor. -public abstract class RoutableScreen : ActivatableViewModelBase, IRoutableScreen where TScreen : class where TParam : new() +public abstract class RoutableScreen : RoutableScreen, IRoutableScreen where TParam : new() { - private bool _recycleScreen = true; - private TScreen? _screen; - - /// - /// Gets the currently active child screen. - /// - public TScreen? Screen - { - get => _screen; - private set => RaiseAndSetIfChanged(ref _screen, value); - } - - /// - /// Called before navigating to this screen. - /// - /// Navigation arguments containing information about the navigation action. - public virtual Task BeforeNavigating(NavigationArguments args) - { - return Task.CompletedTask; - } - /// /// Called while navigating to this screen. /// @@ -50,40 +26,7 @@ public abstract class RoutableScreen : ActivatableViewModelBase { return Task.CompletedTask; } - - /// - /// Called before navigating away from this screen. - /// - /// Navigation arguments containing information about the navigation action. - public virtual Task OnClosing(NavigationArguments args) - { - return Task.CompletedTask; - } - - /// - public bool RecycleScreen - { - get => _recycleScreen; - protected set => RaiseAndSetIfChanged(ref _recycleScreen, value); - } - - #region Overrides of RoutableScreen - - object? IRoutableScreen.InternalScreen => Screen; - - void IRoutableScreen.InternalChangeScreen(object? screen) - { - if (screen == null) - { - Screen = null; - return; - } - - if (screen is not TScreen typedScreen) - throw new ArtemisRoutingException($"Provided screen is not assignable to {typeof(TScreen).FullName}"); - Screen = typedScreen; - } - + async Task IRoutableScreen.InternalOnNavigating(NavigationArguments args, CancellationToken cancellationToken) { Func activator = GetParameterActivator(); @@ -92,6 +35,7 @@ public abstract class RoutableScreen : ActivatableViewModelBase throw new ArtemisRoutingException($"Did not retrieve the required amount of parameters, expects {_parameterPropertyCount}, got {args.SegmentParameters.Length}."); TParam parameters = activator(args.SegmentParameters); + await OnNavigating(args, cancellationToken); await OnNavigating(parameters, args, cancellationToken); } @@ -100,8 +44,6 @@ public abstract class RoutableScreen : ActivatableViewModelBase await OnClosing(args); } - #endregion - #region Parameter generation // ReSharper disable once StaticMemberInGenericType - That's intentional, each kind of TParam should have its own property count diff --git a/src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTScreen.cs b/src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTScreen.cs deleted file mode 100644 index b4f60ca47..000000000 --- a/src/Artemis.UI.Shared/Routing/Routable/RoutableScreenOfTScreen.cs +++ /dev/null @@ -1,87 +0,0 @@ -using System.Threading; -using System.Threading.Tasks; - -namespace Artemis.UI.Shared.Routing; - -/// -/// Represents a view model to which routing can take place and which in turn can host another view model. -/// -/// The type of view model the screen can host. -public abstract class RoutableScreen : ActivatableViewModelBase, IRoutableScreen where TScreen : class -{ - private TScreen? _screen; - private bool _recycleScreen = true; - - /// - /// Gets the currently active child screen. - /// - public TScreen? Screen - { - get => _screen; - private set => RaiseAndSetIfChanged(ref _screen, value); - } - - /// - public bool RecycleScreen - { - get => _recycleScreen; - protected set => RaiseAndSetIfChanged(ref _recycleScreen, value); - } - - /// - /// Called before navigating to this screen. - /// - /// Navigation arguments containing information about the navigation action. - public virtual Task BeforeNavigating(NavigationArguments args) - { - return Task.CompletedTask; - } - - /// - /// Called while navigating to this screen. - /// - /// Navigation arguments containing information about the navigation action. - /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. - public virtual Task OnNavigating(NavigationArguments args, CancellationToken cancellationToken) - { - return Task.CompletedTask; - } - - /// - /// Called before navigating away from this screen. - /// - /// Navigation arguments containing information about the navigation action. - public virtual Task OnClosing(NavigationArguments args) - { - return Task.CompletedTask; - } - - #region Overrides of RoutableScreen - - object? IRoutableScreen.InternalScreen => Screen; - - void IRoutableScreen.InternalChangeScreen(object? screen) - { - if (screen == null) - { - Screen = null; - return; - } - - if (screen is not TScreen typedScreen) - throw new ArtemisRoutingException($"Screen cannot be hosted, {screen.GetType().Name} is not assignable to {typeof(TScreen).Name}."); - Screen = typedScreen; - } - - async Task IRoutableScreen.InternalOnNavigating(NavigationArguments args, CancellationToken cancellationToken) - { - await OnNavigating(args, cancellationToken); - } - - async Task IRoutableScreen.InternalOnClosing(NavigationArguments args) - { - await OnClosing(args); - } - - #endregion -} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Routing/Route/ParameterParsers/LongParameterParser.cs b/src/Artemis.UI.Shared/Routing/Route/ParameterParsers/LongParameterParser.cs new file mode 100644 index 000000000..a8e098579 --- /dev/null +++ b/src/Artemis.UI.Shared/Routing/Route/ParameterParsers/LongParameterParser.cs @@ -0,0 +1,17 @@ + +namespace Artemis.UI.Shared.Routing.ParameterParsers; + +internal class LongParameterParser : IRouteParameterParser +{ + /// + public bool IsMatch(RouteSegment segment, string source) + { + return long.TryParse(source, out _); + } + + /// + public object GetValue(RouteSegment segment, string source) + { + return long.Parse(source); + } +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Routing/Route/RouteRegistration.cs b/src/Artemis.UI.Shared/Routing/Route/RouteRegistration.cs index 2d9f8508f..c62437d03 100644 --- a/src/Artemis.UI.Shared/Routing/Route/RouteRegistration.cs +++ b/src/Artemis.UI.Shared/Routing/Route/RouteRegistration.cs @@ -7,7 +7,7 @@ namespace Artemis.UI.Shared.Routing; /// Represents a registration for a route and its associated view model. /// /// The type of the view model associated with the route. -public class RouteRegistration : IRouterRegistration where TViewModel : ViewModelBase +public class RouteRegistration : IRouterRegistration where TViewModel : RoutableScreen { /// /// Initializes a new instance of the class. diff --git a/src/Artemis.UI.Shared/Routing/Route/RouteResolution.cs b/src/Artemis.UI.Shared/Routing/Route/RouteResolution.cs index 32d285ccd..b80bad548 100644 --- a/src/Artemis.UI.Shared/Routing/Route/RouteResolution.cs +++ b/src/Artemis.UI.Shared/Routing/Route/RouteResolution.cs @@ -74,19 +74,12 @@ internal class RouteResolution }; } - public object GetViewModel(IContainer container) + public RoutableScreen GetViewModel(IContainer container) { - if (ViewModel == null) - throw new ArtemisRoutingException("Cannot get a view model of a non-success route resolution"); - - object? viewModel = container.Resolve(ViewModel); - if (viewModel == null) - throw new ArtemisRoutingException($"Could not resolve view model of type {ViewModel}"); - - return viewModel; + return GetViewModel(container); } - public T GetViewModel(IContainer container) + public T GetViewModel(IContainer container) where T : RoutableScreen { if (ViewModel == null) throw new ArtemisRoutingException("Cannot get a view model of a non-success route resolution"); diff --git a/src/Artemis.UI.Shared/Routing/Route/RouteSegment.cs b/src/Artemis.UI.Shared/Routing/Route/RouteSegment.cs index 1f25b67da..b92e36c3a 100644 --- a/src/Artemis.UI.Shared/Routing/Route/RouteSegment.cs +++ b/src/Artemis.UI.Shared/Routing/Route/RouteSegment.cs @@ -79,6 +79,7 @@ public partial class RouteSegment return parameterType switch { "guid" => new GuidParameterParser(), + "long" => new LongParameterParser(), "int" => new IntParameterParser(), _ => new StringParameterParser() }; diff --git a/src/Artemis.UI.Shared/Routing/Router/IRouter.cs b/src/Artemis.UI.Shared/Routing/Router/IRouter.cs index 44d77bb5c..cfbd0eb89 100644 --- a/src/Artemis.UI.Shared/Routing/Router/IRouter.cs +++ b/src/Artemis.UI.Shared/Routing/Router/IRouter.cs @@ -50,7 +50,7 @@ public interface IRouter /// /// The root screen to set. /// The type of the root screen. It must be a class. - void SetRoot(RoutableScreen root) where TScreen : class; + void SetRoot(RoutableHostScreen root) where TScreen : RoutableScreen; /// /// Sets the root screen from which navigation takes place. @@ -58,7 +58,7 @@ public interface IRouter /// The root screen to set. /// The type of the root screen. It must be a class. /// The type of the parameters for the root screen. It must have a parameterless constructor. - void SetRoot(RoutableScreen root) where TScreen : class where TParam : new(); + void SetRoot(RoutableHostScreen root) where TScreen : RoutableScreen where TParam : new(); /// /// Clears the route used by the previous window, so that it is not restored when the main window opens. diff --git a/src/Artemis.UI.Shared/Routing/Router/Navigation.cs b/src/Artemis.UI.Shared/Routing/Router/Navigation.cs index d04fded6f..0d0f6083c 100644 --- a/src/Artemis.UI.Shared/Routing/Router/Navigation.cs +++ b/src/Artemis.UI.Shared/Routing/Router/Navigation.cs @@ -14,12 +14,12 @@ internal class Navigation private readonly IContainer _container; private readonly ILogger _logger; - private readonly IRoutableScreen _root; + private readonly IRoutableHostScreen _root; private readonly RouteResolution _resolution; private readonly RouterNavigationOptions _options; private CancellationTokenSource _cts; - public Navigation(IContainer container, ILogger logger, IRoutableScreen root, RouteResolution resolution, RouterNavigationOptions options) + public Navigation(IContainer container, ILogger logger, IRoutableHostScreen root, RouteResolution resolution, RouterNavigationOptions options) { _container = container; _logger = logger; @@ -54,21 +54,21 @@ internal class Navigation _cts.Cancel(); } - private async Task NavigateResolution(RouteResolution resolution, NavigationArguments args, IRoutableScreen host) + private async Task NavigateResolution(RouteResolution resolution, NavigationArguments args, IRoutableHostScreen host) { if (Cancelled) return; // Reuse the screen if its type has not changed, if a new one must be created, don't do so on the UI thread - object screen; + IRoutableScreen screen; if (_options.RecycleScreens && host.RecycleScreen && host.InternalScreen != null && host.InternalScreen.GetType() == resolution.ViewModel) screen = host.InternalScreen; else screen = await Task.Run(() => resolution.GetViewModel(_container)); // If resolution has a child, ensure the screen can host it - if (resolution.Child != null && screen is not IRoutableScreen) - throw new ArtemisRoutingException($"Route resolved with a child but view model of type {resolution.ViewModel} is does mot implement {nameof(IRoutableScreen)}."); + if (resolution.Child != null && screen is not IRoutableHostScreen) + throw new ArtemisRoutingException($"Route resolved with a child but view model of type {resolution.ViewModel} is does mot implement {nameof(IRoutableHostScreen)}."); // Only change the screen if it wasn't reused if (!ReferenceEquals(host.InternalScreen, screen)) @@ -87,27 +87,24 @@ internal class Navigation if (CancelIfRequested(args, "ChangeScreen", screen)) return; - - // If the screen implements some form of Navigable, activate it + + // Navigate on the screen args.SegmentParameters = resolution.Parameters ?? Array.Empty(); - if (screen is IRoutableScreen routableScreen) + try { - try - { - await routableScreen.InternalOnNavigating(args, _cts.Token); - } - catch (Exception e) - { - Cancel(); - if (e is not TaskCanceledException) - _logger.Error(e, "Failed to navigate to {Path}", resolution.Path); - } - - if (CancelIfRequested(args, "OnNavigating", screen)) - return; + await screen.InternalOnNavigating(args, _cts.Token); + } + catch (Exception e) + { + Cancel(); + if (e is not TaskCanceledException) + _logger.Error(e, "Failed to navigate to {Path}", resolution.Path); } - if (screen is IRoutableScreen childScreen) + if (CancelIfRequested(args, "OnNavigating", screen)) + return; + + if (screen is IRoutableHostScreen childScreen) { // Navigate the child too if (resolution.Child != null) @@ -121,11 +118,9 @@ internal class Navigation Completed = true; } - public bool PathEquals(string path, bool allowPartialMatch) + public bool PathEquals(string path, RouterNavigationOptions options) { - if (allowPartialMatch) - return _resolution.Path.StartsWith(path, StringComparison.InvariantCultureIgnoreCase); - return string.Equals(_resolution.Path, path, StringComparison.InvariantCultureIgnoreCase); + return options.PathEquals(_resolution.Path, path); } private bool CancelIfRequested(NavigationArguments args, string stage, object screen) diff --git a/src/Artemis.UI.Shared/Routing/Router/Router.cs b/src/Artemis.UI.Shared/Routing/Router/Router.cs index 028d7be02..282511c40 100644 --- a/src/Artemis.UI.Shared/Routing/Router/Router.cs +++ b/src/Artemis.UI.Shared/Routing/Router/Router.cs @@ -14,15 +14,15 @@ internal class Router : CorePropertyChanged, IRouter, IDisposable private readonly Stack _backStack = new(); private readonly BehaviorSubject _currentRouteSubject; private readonly Stack _forwardStack = new(); - private readonly Func _getNavigation; + private readonly Func _getNavigation; private readonly ILogger _logger; private readonly IMainWindowService _mainWindowService; private Navigation? _currentNavigation; - private IRoutableScreen? _root; + private IRoutableHostScreen? _root; private string? _previousWindowRoute; - public Router(ILogger logger, IMainWindowService mainWindowService, Func getNavigation) + public Router(ILogger logger, IMainWindowService mainWindowService, Func getNavigation) { _logger = logger; _mainWindowService = mainWindowService; @@ -45,28 +45,22 @@ internal class Router : CorePropertyChanged, IRouter, IDisposable return RouteResolution.AsFailure(path); } - private async Task RequestClose(object screen, NavigationArguments args) + private async Task RequestClose(IRoutableScreen screen, NavigationArguments args) { - if (screen is not IRoutableScreen routableScreen) - return true; - - await routableScreen.InternalOnClosing(args); - if (args.Cancelled) - { - _logger.Debug("Navigation to {Path} cancelled during RequestClose by {Screen}", args.Path, screen.GetType().Name); + // Drill down to child screens first + if (screen is IRoutableHostScreen hostScreen && hostScreen.InternalScreen != null && !await RequestClose(hostScreen.InternalScreen, args)) return false; - } - if (routableScreen.InternalScreen == null) + await screen.InternalOnClosing(args); + if (!args.Cancelled) return true; - return await RequestClose(routableScreen.InternalScreen, args); + _logger.Debug("Navigation to {Path} cancelled during RequestClose by {Screen}", args.Path, screen.GetType().Name); + return false; } - private bool PathEquals(string path, bool allowPartialMatch) + private bool PathEquals(string path, RouterNavigationOptions options) { - if (allowPartialMatch) - return _currentRouteSubject.Value != null && _currentRouteSubject.Value.StartsWith(path, StringComparison.InvariantCultureIgnoreCase); - return string.Equals(_currentRouteSubject.Value, path, StringComparison.InvariantCultureIgnoreCase); + return _currentRouteSubject.Value != null && options.PathEquals(_currentRouteSubject.Value, path); } /// @@ -88,7 +82,7 @@ internal class Router : CorePropertyChanged, IRouter, IDisposable { if (_root == null) throw new ArtemisRoutingException("Cannot navigate without a root having been set"); - if (PathEquals(path, options.IgnoreOnPartialMatch) || (_currentNavigation != null && _currentNavigation.PathEquals(path, options.IgnoreOnPartialMatch))) + if (PathEquals(path, options) || (_currentNavigation != null && _currentNavigation.PathEquals(path, options))) return; string? previousPath = _currentRouteSubject.Value; @@ -161,13 +155,13 @@ internal class Router : CorePropertyChanged, IRouter, IDisposable } /// - public void SetRoot(RoutableScreen root) where TScreen : class + public void SetRoot(RoutableHostScreen root) where TScreen : RoutableScreen { _root = root; } /// - public void SetRoot(RoutableScreen root) where TScreen : class where TParam : new() + public void SetRoot(RoutableHostScreen root) where TScreen : RoutableScreen where TParam : new() { _root = root; } diff --git a/src/Artemis.UI.Shared/Routing/Router/RouterNavigationOptions.cs b/src/Artemis.UI.Shared/Routing/Router/RouterNavigationOptions.cs index a5c63ab88..273f85c2c 100644 --- a/src/Artemis.UI.Shared/Routing/Router/RouterNavigationOptions.cs +++ b/src/Artemis.UI.Shared/Routing/Router/RouterNavigationOptions.cs @@ -1,3 +1,5 @@ +using System; + namespace Artemis.UI.Shared.Routing; /// @@ -21,9 +23,31 @@ public class RouterNavigationOptions /// If set to true, a route change from page/subpage1/subpage2 to page/subpage1 will be ignored. public bool IgnoreOnPartialMatch { get; set; } = false; + /// + /// Gets or sets the path to use when determining whether the path is a partial match, + /// only has any effect if is . + /// + public string? PartialMatchOverride { get; set; } + /// /// Gets or sets a boolean value indicating whether logging should be enabled. /// Errors and warnings are always logged. /// public bool EnableLogging { get; set; } = true; + + /// + /// Determines whether the given two paths are considered equal using these navigation options. + /// + /// The current path. + /// The target path. + /// if the paths are considered equal; otherwise . + internal bool PathEquals(string current, string target) + { + if (PartialMatchOverride != null && IgnoreOnPartialMatch) + target = PartialMatchOverride; + + if (IgnoreOnPartialMatch) + return current.StartsWith(target, StringComparison.InvariantCultureIgnoreCase); + return string.Equals(current, target, StringComparison.InvariantCultureIgnoreCase); + } } \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Services/Builders/NotificationBuilder.cs b/src/Artemis.UI.Shared/Services/Builders/NotificationBuilder.cs index 5dfe0ecb9..805c5cbd0 100644 --- a/src/Artemis.UI.Shared/Services/Builders/NotificationBuilder.cs +++ b/src/Artemis.UI.Shared/Services/Builders/NotificationBuilder.cs @@ -2,11 +2,11 @@ using System.Threading.Tasks; using System.Windows.Input; using Avalonia.Controls; +using Avalonia.Controls.Primitives; using Avalonia.Layout; using Avalonia.Threading; using FluentAvalonia.UI.Controls; using ReactiveUI; -using Button = Avalonia.Controls.Button; namespace Artemis.UI.Shared.Services.Builders; @@ -117,34 +117,34 @@ public class NotificationBuilder /// public Action Show() { - Panel? panel = _parent.Find("NotificationContainer"); - if (panel == null) - throw new ArtemisSharedUIException("Can't display a notification on a window without a NotificationContainer."); - Dispatcher.UIThread.Post(() => { - panel.Children.Add(_infoBar); + OverlayLayer? overlayLayer = OverlayLayer.GetOverlayLayer(_parent); + if (overlayLayer == null) + throw new ArtemisSharedUIException("Can't display a notification on a window an overlay layer."); + + NotificationHost container = new() {Content = _infoBar}; + overlayLayer.Children.Add(container); _infoBar.Closed += InfoBarOnClosed; _infoBar.IsOpen = true; - }); - Task.Run(async () => - { - await Task.Delay(_timeout); - Dispatcher.UIThread.Post(() => _infoBar.IsOpen = false); + Dispatcher.UIThread.InvokeAsync(async () => + { + await Task.Delay(_timeout); + _infoBar.IsOpen = false; + }); + + return; + + void InfoBarOnClosed(InfoBar sender, InfoBarClosedEventArgs args) + { + overlayLayer.Children.Remove(container); + _infoBar.Closed -= InfoBarOnClosed; + } }); return () => Dispatcher.UIThread.Post(() => _infoBar.IsOpen = false); } - - private void InfoBarOnClosed(InfoBar sender, InfoBarClosedEventArgs args) - { - _infoBar.Closed -= InfoBarOnClosed; - if (_parent.Content is not Panel panel) - return; - - panel.Children.Remove(_infoBar); - } } /// @@ -180,7 +180,7 @@ public class NotificationButtonBuilder _action = action; return this; } - + /// /// Changes action that is called when the button is clicked. /// @@ -222,9 +222,13 @@ public class NotificationButtonBuilder button.Classes.Add("AppBarButton"); if (_action != null) + { button.Command = ReactiveCommand.Create(() => _action()); + } else if (_asyncAction != null) + { button.Command = ReactiveCommand.CreateFromTask(() => _asyncAction()); + } else if (_command != null) { button.Command = _command; diff --git a/src/Artemis.UI.Shared/Services/Interfaces/IWindowService.cs b/src/Artemis.UI.Shared/Services/Interfaces/IWindowService.cs index ff4394fde..15756d2a0 100644 --- a/src/Artemis.UI.Shared/Services/Interfaces/IWindowService.cs +++ b/src/Artemis.UI.Shared/Services/Interfaces/IWindowService.cs @@ -16,7 +16,7 @@ public interface IWindowService : IArtemisSharedUIService /// /// The type of view model to create /// The created view model - TViewModel ShowWindow(params object[] parameters); + Window ShowWindow(out TViewModel viewModel, params object[] parameters); /// /// Given a ViewModel, show its corresponding View as a window diff --git a/src/Artemis.UI.Shared/Services/ProfileEditor/ProfileEditorService.cs b/src/Artemis.UI.Shared/Services/ProfileEditor/ProfileEditorService.cs index 42390dd3e..f0c4cf085 100644 --- a/src/Artemis.UI.Shared/Services/ProfileEditor/ProfileEditorService.cs +++ b/src/Artemis.UI.Shared/Services/ProfileEditor/ProfileEditorService.cs @@ -10,7 +10,6 @@ using Artemis.Core; using Artemis.Core.Services; using Artemis.UI.Shared.Services.MainWindow; using Artemis.UI.Shared.Services.ProfileEditor.Commands; -using Avalonia.Threading; using DynamicData; using Serilog; @@ -140,14 +139,14 @@ internal class ProfileEditorService : IProfileEditorService private void ApplyFocusMode() { if (_suspendedEditingSubject.Value) - _profileService.EditorFocus = null; + _profileService.FocusProfileElement = null; - _profileService.EditorFocus = _focusModeSubject.Value switch + _profileService.FocusProfileElement = _focusModeSubject.Value switch { ProfileEditorFocusMode.None => null, ProfileEditorFocusMode.Folder => _profileElementSubject.Value?.Parent, ProfileEditorFocusMode.Selection => _profileElementSubject.Value, - _ => _profileService.EditorFocus + _ => _profileService.FocusProfileElement }; } @@ -164,52 +163,38 @@ internal class ProfileEditorService : IProfileEditorService public async Task ChangeCurrentProfileConfiguration(ProfileConfiguration? profileConfiguration) { - if (ReferenceEquals(_profileConfigurationSubject.Value, profileConfiguration)) + ProfileConfiguration? previous = _profileConfigurationSubject.Value; + if (ReferenceEquals(previous, profileConfiguration)) return; _logger.Verbose("ChangeCurrentProfileConfiguration {profile}", profileConfiguration); // Stop playing and save the current profile Pause(); - if (_profileConfigurationSubject.Value?.Profile != null) - { - _profileConfigurationSubject.Value.Profile.Reset(); - _profileConfigurationSubject.Value.Profile.LastSelectedProfileElement = _profileElementSubject.Value; - } - await SaveProfileAsync(); - // No need to deactivate the profile, if needed it will be deactivated next update - if (_profileConfigurationSubject.Value != null) - _profileConfigurationSubject.Value.IsBeingEdited = false; - // Deselect whatever profile element was active ChangeCurrentProfileElement(null); + ChangeSuspendedEditing(false); // Close the command scope if one was open _profileEditorHistoryScope?.Dispose(); - // The new profile may need activation - if (profileConfiguration != null) + // Activate the profile and it's mode off of the UI thread + await Task.Run(() => { - await Task.Run(() => - { - profileConfiguration.IsBeingEdited = true; - _moduleService.SetActivationOverride(profileConfiguration.Module); + // Activate the profile if one was provided + if (profileConfiguration != null) _profileService.ActivateProfile(profileConfiguration); - _profileService.RenderForEditor = true; - }); - if (profileConfiguration.Profile?.LastSelectedProfileElement is RenderProfileElement renderProfileElement) - ChangeCurrentProfileElement(renderProfileElement); - } - else - { - _moduleService.SetActivationOverride(null); - _profileService.RenderForEditor = false; - } + // If there is no profile configuration or module, deliberately set the override to null + _moduleService.SetActivationOverride(profileConfiguration?.Module); + }); + _profileService.FocusProfile = profileConfiguration; _profileConfigurationSubject.OnNext(profileConfiguration); + ChangeTime(TimeSpan.Zero); + previous?.Profile?.Reset(); } public void ChangeCurrentProfileElement(RenderProfileElement? renderProfileElement) @@ -238,23 +223,23 @@ internal class ProfileEditorService : IProfileEditorService if (_suspendedEditingSubject.Value == suspend) return; - _suspendedEditingSubject.OnNext(suspend); if (suspend) { Pause(); - _profileService.RenderForEditor = false; + _profileService.UpdateFocusProfile = true; _profileConfigurationSubject.Value?.Profile?.Reset(); } else { if (_profileConfigurationSubject.Value != null) - _profileService.RenderForEditor = true; + _profileService.UpdateFocusProfile = false; Tick(_timeSubject.Value); } + _suspendedEditingSubject.OnNext(suspend); ApplyFocusMode(); } - + public void ChangeFocusMode(ProfileEditorFocusMode focusMode) { if (_focusModeSubject.Value == focusMode) @@ -411,10 +396,8 @@ internal class ProfileEditorService : IProfileEditorService public void SaveProfile() { Profile? profile = _profileConfigurationSubject.Value?.Profile; - if (profile == null) - return; - - _profileService.SaveProfile(profile, true); + if (profile != null) + _profileService.SaveProfile(profile, true); } /// diff --git a/src/Artemis.UI.Shared/Services/Window/ExceptionDialogView.axaml b/src/Artemis.UI.Shared/Services/Window/ExceptionDialogView.axaml index 3bb950a0f..b228ebe2f 100644 --- a/src/Artemis.UI.Shared/Services/Window/ExceptionDialogView.axaml +++ b/src/Artemis.UI.Shared/Services/Window/ExceptionDialogView.axaml @@ -32,7 +32,7 @@ diff --git a/src/Artemis.UI.Shared/Services/Window/WindowService.cs b/src/Artemis.UI.Shared/Services/Window/WindowService.cs index c30688eef..5db1716d2 100644 --- a/src/Artemis.UI.Shared/Services/Window/WindowService.cs +++ b/src/Artemis.UI.Shared/Services/Window/WindowService.cs @@ -21,14 +21,13 @@ internal class WindowService : IWindowService _container = container; } - public T ShowWindow(params object[] parameters) + public Window ShowWindow(out T viewModel, params object[] parameters) { - T viewModel = _container.Resolve(parameters); + viewModel = _container.Resolve(parameters); if (viewModel == null) throw new ArtemisSharedUIException($"Failed to show window for VM of type {typeof(T).Name}, could not create instance."); - ShowWindow(viewModel); - return viewModel; + return ShowWindow(viewModel); } public Window ShowWindow(object viewModel) diff --git a/src/Artemis.UI.Shared/Styles/Artemis.axaml b/src/Artemis.UI.Shared/Styles/Artemis.axaml index 870b5f242..d32872413 100644 --- a/src/Artemis.UI.Shared/Styles/Artemis.axaml +++ b/src/Artemis.UI.Shared/Styles/Artemis.axaml @@ -15,6 +15,7 @@ + @@ -31,6 +32,7 @@ + diff --git a/src/Artemis.UI.Shared/Styles/Border.axaml b/src/Artemis.UI.Shared/Styles/Border.axaml index d753809a0..e3997e5ec 100644 --- a/src/Artemis.UI.Shared/Styles/Border.axaml +++ b/src/Artemis.UI.Shared/Styles/Border.axaml @@ -23,15 +23,11 @@ - - - 8 - - + + + + + + - - + + - - - - - + + \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Styles/Control.axaml b/src/Artemis.UI.Shared/Styles/Control.axaml new file mode 100644 index 000000000..b20c87ff5 --- /dev/null +++ b/src/Artemis.UI.Shared/Styles/Control.axaml @@ -0,0 +1,25 @@ + + + + + + + + + + + + diff --git a/src/Artemis.UI.Shared/Styles/Controls/DataModelPicker.axaml b/src/Artemis.UI.Shared/Styles/Controls/DataModelPicker.axaml index 662a1109f..0175ae880 100644 --- a/src/Artemis.UI.Shared/Styles/Controls/DataModelPicker.axaml +++ b/src/Artemis.UI.Shared/Styles/Controls/DataModelPicker.axaml @@ -66,7 +66,8 @@ @@ -81,7 +82,7 @@ IsVisible="{CompiledBinding IsEventPicker, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type dataModelPicker:DataModelPicker}}}"/> - + @@ -90,7 +91,8 @@ diff --git a/src/Artemis.UI.Shared/Styles/InfoBar.axaml b/src/Artemis.UI.Shared/Styles/InfoBar.axaml index 3f9b4a9bc..7b6f7f62c 100644 --- a/src/Artemis.UI.Shared/Styles/InfoBar.axaml +++ b/src/Artemis.UI.Shared/Styles/InfoBar.axaml @@ -17,6 +17,8 @@ + + diff --git a/src/Artemis.UI.Shared/Styles/Notifications.axaml b/src/Artemis.UI.Shared/Styles/Notifications.axaml index 0e9553ca2..94bc9039c 100644 --- a/src/Artemis.UI.Shared/Styles/Notifications.axaml +++ b/src/Artemis.UI.Shared/Styles/Notifications.axaml @@ -1,17 +1,14 @@  + xmlns:controls="clr-namespace:FluentAvalonia.UI.Controls;assembly=FluentAvalonia" + xmlns:shared="clr-namespace:Artemis.UI.Shared"> - - - + + + + + + + + + + diff --git a/src/Artemis.UI.Shared/Utilities/ProgressableStreamContent.cs b/src/Artemis.UI.Shared/Utilities/ProgressableStreamContent.cs new file mode 100644 index 000000000..bac5bb9d1 --- /dev/null +++ b/src/Artemis.UI.Shared/Utilities/ProgressableStreamContent.cs @@ -0,0 +1,119 @@ +// Heavily based on: +// SkyClip +// - ProgressableStreamContent.cs +// -------------------------------------------------------------------- +// Author: Jeff Hansen +// Copyright (C) Jeff Hansen 2015. All rights reserved. + +using System; +using System.IO; +using System.Net; +using System.Net.Http; +using System.Threading; +using System.Threading.Tasks; + +namespace Artemis.UI.Shared.Utilities; + +/// +/// Provides HTTP content based on a stream with support for IProgress. +/// +public class ProgressableStreamContent : StreamContent +{ + private const int DEFAULT_BUFFER_SIZE = 4096; + + private readonly int _bufferSize; + private readonly IProgress _progress; + private readonly Stream _streamToWrite; + private bool _contentConsumed; + + /// + /// Initializes a new instance of the class. + /// + /// The stream to write. + /// The downloader. + public ProgressableStreamContent(Stream streamToWrite, IProgress progress) : this(streamToWrite, DEFAULT_BUFFER_SIZE, progress) + { + } + + /// + /// Initializes a new instance of the class. + /// + /// The stream to write. + /// The buffer size. + /// The downloader. + public ProgressableStreamContent(Stream streamToWrite, int bufferSize, IProgress progress) : base(streamToWrite, bufferSize) + { + if (bufferSize <= 0) + throw new ArgumentOutOfRangeException(nameof(bufferSize)); + + _streamToWrite = streamToWrite; + _bufferSize = bufferSize; + _progress = progress; + } + + /// + protected override void Dispose(bool disposing) + { + if (disposing) + _streamToWrite.Dispose(); + + base.Dispose(disposing); + } + + /// + protected override async Task SerializeToStreamAsync(Stream stream, TransportContext? context) + { + await SerializeToStreamAsync(stream, context, CancellationToken.None); + } + + /// + protected override async Task SerializeToStreamAsync(Stream stream, TransportContext? context, CancellationToken cancellationToken) + { + PrepareContent(); + + byte[] buffer = new byte[_bufferSize]; + long size = _streamToWrite.Length; + int uploaded = 0; + + await using (_streamToWrite) + { + while (!cancellationToken.IsCancellationRequested) + { + int length = await _streamToWrite.ReadAsync(buffer, cancellationToken); + if (length <= 0) + break; + + uploaded += length; + _progress.Report(new StreamProgress(uploaded, size)); + await stream.WriteAsync(buffer, 0, length, cancellationToken); + } + } + } + + /// + protected override bool TryComputeLength(out long length) + { + length = _streamToWrite.Length; + return true; + } + + /// + /// Prepares the content. + /// + /// The stream has already been read. + private void PrepareContent() + { + if (_contentConsumed) + { + // If the content needs to be written to a target stream a 2nd time, then the stream must support + // seeking (e.g. a FileStream), otherwise the stream can't be copied a second time to a target + // stream (e.g. a NetworkStream). + if (_streamToWrite.CanSeek) + _streamToWrite.Position = 0; + else + throw new InvalidOperationException("The stream has already been read."); + } + + _contentConsumed = true; + } +} \ No newline at end of file diff --git a/src/Artemis.UI.Shared/Utilities/StreamProgress.cs b/src/Artemis.UI.Shared/Utilities/StreamProgress.cs new file mode 100644 index 000000000..eb0b7a713 --- /dev/null +++ b/src/Artemis.UI.Shared/Utilities/StreamProgress.cs @@ -0,0 +1,57 @@ +// Heavily based on: +// SkyClip +// - UploadProgress.cs +// -------------------------------------------------------------------- +// Author: Jeff Hansen +// Copyright (C) Jeff Hansen 2015. All rights reserved. + +namespace Artemis.UI.Shared.Utilities; + +/// +/// The upload progress. +/// +public class StreamProgress +{ + /// + /// Initializes a new instance of the class. + /// + /// + /// The bytes transfered. + /// + /// + /// The total bytes. + /// + public StreamProgress(long bytesTransfered, long? totalBytes) + { + BytesTransfered = bytesTransfered; + TotalBytes = totalBytes; + if (totalBytes.HasValue) + ProgressPercentage = (int) ((float) bytesTransfered / totalBytes.Value * 100); + } + + /// + /// Returns a that represents this instance. + /// + /// + /// A that represents this instance. + /// + public override string ToString() + { + return string.Format("{0}% ({1} / {2})", ProgressPercentage, BytesTransfered, TotalBytes); + } + + /// + /// Gets the bytes transfered. + /// + public long BytesTransfered { get; } + + /// + /// Gets the progress percentage. + /// + public int ProgressPercentage { get; } + + /// + /// Gets the total bytes. + /// + public long? TotalBytes { get; } +} \ No newline at end of file diff --git a/src/Artemis.UI.Windows/App.axaml b/src/Artemis.UI.Windows/App.axaml index 836d6a11f..63ff0b8a1 100644 --- a/src/Artemis.UI.Windows/App.axaml +++ b/src/Artemis.UI.Windows/App.axaml @@ -18,7 +18,7 @@ - + diff --git a/src/Artemis.UI/Artemis.UI.csproj b/src/Artemis.UI/Artemis.UI.csproj index 914c45ccd..c1b71c6f2 100644 --- a/src/Artemis.UI/Artemis.UI.csproj +++ b/src/Artemis.UI/Artemis.UI.csproj @@ -18,18 +18,24 @@ + + + + + - + + @@ -38,9 +44,30 @@ + + + + + + ProfileListView.axaml + Code + + + LayoutListView.axaml + Code + + + SubmissionsDetailView.axaml + Code + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Animations/email.json b/src/Artemis.UI/Assets/Animations/email.json new file mode 100644 index 000000000..0d274481c --- /dev/null +++ b/src/Artemis.UI/Assets/Animations/email.json @@ -0,0 +1 @@ +{"v":"5.7.1","fr":25,"ip":0,"op":200,"w":1000,"h":1000,"nm":"Work frome Home Mailing F","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Null 1","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[500,652,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":200,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"mail 4","parent":1,"td":1,"sr":1,"ks":{"o":{"a":0,"k":99,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[183.055,-315.211,0],"ix":2},"a":{"a":0,"k":[2153.478,680.843,0],"ix":1},"s":{"a":0,"k":[56,56,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-0.758,-2.403],[0,0],[0,0],[0,0],[-21.953,-7.476],[-0.627,-0.393]],"o":[[0,0],[0,0],[0,0],[0,0],[9.299,3.167],[1.978,1.239]],"v":[[78.735,-34.708],[80.678,-28.718],[-58.964,72.319],[-68.916,40.154],[58.006,-46.114],[74.435,-40.273]],"c":true},"ix":2},"nm":"Path 4","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.174548238516,0.452203959227,0.890196084976,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 3","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1801.139,934.164],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[594,594],"ix":3},"r":{"a":0,"k":17.3,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"back 2","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":204,"st":-23,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"plane 2","parent":1,"tt":2,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-30.27,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.57,"y":1},"o":{"x":0.43,"y":0},"t":0,"s":[-417.664,54.901,0],"to":[29.664,67.099,0],"ti":[8.659,75.131,0]},{"i":{"x":0.57,"y":1},"o":{"x":0.444,"y":0},"t":50,"s":[384.659,-254.869,0],"to":[-6.219,-53.959,0],"ti":[-29.664,-67.099,0]},{"i":{"x":0.57,"y":1},"o":{"x":0.43,"y":0},"t":100,"s":[-417.664,54.901,0],"to":[29.664,67.099,0],"ti":[8.659,75.131,0]},{"i":{"x":0.595,"y":1},"o":{"x":0.444,"y":0},"t":150,"s":[384.659,-254.869,0],"to":[-6.219,-53.959,0],"ti":[-17.336,-57.901,0]},{"t":199,"s":[-417.664,54.901,0]}],"ix":2},"a":{"a":0,"k":[816.438,755.113,0],"ix":1},"s":{"a":0,"k":[65,65,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":-7,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[18.514,-6.941],[-33.276,21.927],[93.386,-0.562]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-6.915,-14.251],[-9.38,40.839],[47.51,-30.24]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[21.986,-29.805],[113.019,32.241],[42.547,-64.73]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":30,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-0.528,-14.328],[129.556,-5.673],[-49.243,-49.537]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":48,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[12.419,-8.39],[20.033,-35.22],[-88.885,-12.543]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":61,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[55.573,8.709],[-40.935,-18.354],[28.444,45.375]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[6.11,3.203],[-110.008,-19.184],[7.061,50.133]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":84,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[70.462,-13.025],[-68.13,-5.656],[131.7,43.826]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":93,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[18.514,-6.941],[-33.276,21.927],[93.387,-0.562]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":103,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-6.915,-14.251],[-9.38,40.839],[47.51,-30.24]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":112,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[21.986,-29.805],[113.019,32.241],[42.547,-64.73]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":130,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-0.528,-14.328],[129.556,-5.673],[-49.243,-49.537]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":148,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[12.419,-8.39],[20.033,-35.22],[-88.885,-12.543]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":161,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[55.573,8.709],[-40.935,-18.354],[28.444,45.375]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":170,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[6.11,3.203],[-110.008,-19.184],[7.061,50.133]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":184,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[70.462,-13.025],[-68.13,-5.656],[131.7,43.826]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":193,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[18.514,-6.941],[-33.276,21.927],[93.386,-0.562]],"c":true}]},{"t":203,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-6.915,-14.251],[-9.38,40.839],[47.51,-30.24]],"c":true}]}],"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":-7,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-33.276,21.927],[5.299,-10.555],[-29.451,-26.172]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-9.38,40.839],[-24.162,-11.06],[-91.62,-22.853]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[113.019,32.241],[-1.135,-26.615],[-113.019,-32.241]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":30,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[129.556,-5.673],[7.207,-4.426],[12.17,72.534]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":48,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[20.033,-35.22],[24.82,-6.359],[108.512,9]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":61,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-40.935,-18.354],[69.258,0.624],[114.118,-27.203]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-110.008,-19.184],[15.525,-1.402],[54.092,-19.43]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":84,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-68.13,-5.656],[63.129,-26.565],[63.443,-50.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":93,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-33.276,21.927],[5.299,-10.555],[-29.451,-26.172]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":103,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-9.38,40.839],[-24.162,-11.06],[-91.62,-22.853]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":112,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[113.019,32.241],[-1.135,-26.615],[-113.019,-32.241]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":130,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[129.556,-5.673],[7.207,-4.426],[12.17,72.534]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":148,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[20.033,-35.22],[24.82,-6.359],[108.512,9]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":161,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-40.935,-18.354],[69.258,0.624],[114.118,-27.203]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":170,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-110.008,-19.184],[15.525,-1.402],[54.092,-19.43]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":184,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-68.13,-5.656],[63.129,-26.565],[63.443,-50.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":193,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-33.276,21.927],[5.299,-10.555],[-29.451,-26.172]],"c":true}]},{"t":203,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-9.38,40.839],[-24.162,-11.06],[-91.62,-22.853]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.828505811504,0.876804486443,0.931862146714,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ind":3,"ty":"sh","ix":4,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":-7,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-14.241,3.304],[-30.454,20.957],[19.514,-1.564],[11.823,5.95],[3.437,-9.631]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-21.862,17.885],[-9.38,40.839],[-6.915,-14.251],[-10.368,5.64],[-25.224,-7.275]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-24.159,12.674],[113.019,32.241],[21.986,-29.805],[17.357,-9.915],[-16.004,-3.872]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":30,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.35,-4.914],[129.556,-5.673],[-3.271,-18.353],[-13.529,7.902],[42.148,1]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":48,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[34.167,-10.201],[20.033,-35.22],[10.099,-7.209],[13.95,35.083],[26.183,-5.636]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":61,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[75.86,-3.345],[-40.935,-18.354],[42.881,12.14],[74.348,24.549],[70.369,1.013]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[12.51,-7.277],[-110.008,-19.184],[-19.192,3.494],[20.236,23.254],[16.22,-0.587]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":84,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-3.126,-17.964],[-61.192,-8.042],[72.921,0.195],[59.046,-10.343],[60.098,-29.814]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":93,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-14.241,3.304],[-30.454,20.957],[19.514,-1.564],[11.823,5.95],[3.437,-9.631]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":103,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-21.861,17.885],[-9.38,40.839],[-6.915,-14.251],[-10.368,5.64],[-25.224,-7.275]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":112,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-24.159,12.674],[113.019,32.241],[21.986,-29.805],[17.357,-9.915],[-16.004,-3.872]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":130,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.35,-4.914],[129.556,-5.673],[-3.271,-18.353],[-13.529,7.902],[42.148,1]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":148,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[34.167,-10.201],[20.033,-35.22],[10.099,-7.209],[13.95,35.083],[26.183,-5.636]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":161,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[75.86,-3.345],[-40.935,-18.354],[42.881,12.14],[74.348,24.549],[70.369,1.013]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":170,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[12.51,-7.277],[-110.008,-19.184],[-19.192,3.494],[20.236,23.254],[16.22,-0.587]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":184,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-3.126,-17.964],[-61.192,-8.042],[72.921,0.195],[59.046,-10.343],[60.098,-29.814]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":193,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-14.241,3.304],[-30.454,20.957],[19.514,-1.564],[11.823,5.95],[3.437,-9.631]],"c":true}]},{"t":203,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-21.862,17.885],[-9.38,40.839],[-6.915,-14.251],[-10.368,5.64],[-25.224,-7.275]],"c":true}]}],"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.65451594334,0.742976110122,0.858894258387,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[795.902,768.678],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 9","np":5,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":-7,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[18.514,-6.941],[-33.276,21.927],[93.386,-0.562]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-6.915,-14.251],[-9.38,40.839],[47.51,-30.24]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[21.986,-29.805],[113.019,32.241],[42.547,-64.73]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":30,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-0.528,-14.328],[129.556,-5.673],[-49.243,-49.537]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":48,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[12.419,-8.39],[20.033,-35.22],[-88.885,-12.543]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":61,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[55.573,8.709],[-40.935,-18.354],[28.444,45.375]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[6.11,3.203],[-110.008,-19.184],[7.061,50.133]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":84,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[70.462,-13.025],[-68.13,-5.656],[131.7,43.826]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":93,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[18.514,-6.941],[-33.276,21.927],[93.387,-0.562]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":103,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-6.915,-14.251],[-9.38,40.839],[47.51,-30.24]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":112,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[21.986,-29.805],[113.019,32.241],[42.547,-64.73]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":130,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-0.528,-14.328],[129.556,-5.673],[-49.243,-49.537]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":148,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[12.419,-8.39],[20.033,-35.22],[-88.885,-12.543]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":161,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[55.573,8.709],[-40.935,-18.354],[28.444,45.375]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":170,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[6.11,3.203],[-110.008,-19.184],[7.061,50.133]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":184,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[70.462,-13.025],[-68.13,-5.656],[131.7,43.826]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":193,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[18.514,-6.941],[-33.276,21.927],[93.386,-0.562]],"c":true}]},{"t":203,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-6.915,-14.251],[-9.38,40.839],[47.51,-30.24]],"c":true}]}],"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":-7,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-33.276,21.927],[5.299,-10.555],[-29.451,-26.172]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-9.38,40.839],[-24.162,-11.06],[-91.62,-22.853]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[113.019,32.241],[-1.135,-26.615],[-113.019,-32.241]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":30,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[129.556,-5.673],[7.207,-4.426],[12.17,72.534]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":48,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[20.033,-35.22],[24.82,-6.359],[108.512,9]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":61,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-40.935,-18.354],[69.258,0.624],[114.118,-27.203]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-110.008,-19.184],[15.525,-1.402],[54.092,-19.43]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":84,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-68.13,-5.656],[63.129,-26.565],[63.443,-50.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":93,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-33.276,21.927],[5.299,-10.555],[-29.451,-26.172]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":103,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-9.38,40.839],[-24.162,-11.06],[-91.62,-22.853]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":112,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[113.019,32.241],[-1.135,-26.615],[-113.019,-32.241]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":130,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[129.556,-5.673],[7.207,-4.426],[12.17,72.534]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":148,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[20.033,-35.22],[24.82,-6.359],[108.512,9]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":161,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-40.935,-18.354],[69.258,0.624],[114.118,-27.203]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":170,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-110.008,-19.184],[15.525,-1.402],[54.092,-19.43]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":184,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-68.13,-5.656],[63.129,-26.565],[63.443,-50.053]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":193,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-33.276,21.927],[5.299,-10.555],[-29.451,-26.172]],"c":true}]},{"t":203,"s":[{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-9.38,40.839],[-24.162,-11.06],[-91.62,-22.853]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":-7,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-14.241,3.304],[-30.454,20.957],[19.514,-1.564],[11.823,5.95],[3.437,-9.631]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":3,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-21.862,17.885],[-9.38,40.839],[-6.915,-14.251],[-10.368,5.64],[-25.224,-7.275]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-24.159,12.674],[113.019,32.241],[21.986,-29.805],[17.357,-9.915],[-16.004,-3.872]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":30,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.35,-4.914],[129.556,-5.673],[-3.271,-18.353],[-13.529,7.902],[42.148,1]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":48,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[34.167,-10.201],[20.033,-35.22],[10.099,-7.209],[13.95,35.083],[26.183,-5.636]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":61,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[75.86,-3.345],[-40.935,-18.354],[42.881,12.14],[74.348,24.549],[70.369,1.013]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[12.51,-7.277],[-110.008,-19.184],[-19.192,3.494],[20.236,23.254],[16.22,-0.587]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":84,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-3.126,-17.964],[-61.192,-8.042],[72.921,0.195],[59.046,-10.343],[60.098,-29.814]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":93,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-14.241,3.304],[-30.454,20.957],[19.514,-1.564],[11.823,5.95],[3.437,-9.631]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":103,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-21.861,17.885],[-9.38,40.839],[-6.915,-14.251],[-10.368,5.64],[-25.224,-7.275]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":112,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-24.159,12.674],[113.019,32.241],[21.986,-29.805],[17.357,-9.915],[-16.004,-3.872]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":130,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[55.35,-4.914],[129.556,-5.673],[-3.271,-18.353],[-13.529,7.902],[42.148,1]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":148,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[34.167,-10.201],[20.033,-35.22],[10.099,-7.209],[13.95,35.083],[26.183,-5.636]],"c":true}]},{"i":{"x":0.32,"y":1},"o":{"x":0.167,"y":0.167},"t":161,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[75.86,-3.345],[-40.935,-18.354],[42.881,12.14],[74.348,24.549],[70.369,1.013]],"c":true}]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":170,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[12.51,-7.277],[-110.008,-19.184],[-19.192,3.494],[20.236,23.254],[16.22,-0.587]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":184,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-3.126,-17.964],[-61.192,-8.042],[72.921,0.195],[59.046,-10.343],[60.098,-29.814]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":193,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-14.241,3.304],[-30.454,20.957],[19.514,-1.564],[11.823,5.95],[3.437,-9.631]],"c":true}]},{"t":203,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-21.862,17.885],[-9.38,40.839],[-6.915,-14.251],[-10.368,5.64],[-25.224,-7.275]],"c":true}]}],"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0,0,0,1],"ix":4},"o":{"a":0,"k":20,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[790.902,773.678],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 8","np":4,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false}],"ip":-2,"op":201,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Shape Layer 9","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[141.25,25,0],"to":[-3.73,-1.383,0],"ti":[-8.37,-1.955,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":38,"s":[146.62,14.455,0],"to":[5.12,2.58,0],"ti":[5.195,1.594,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":75,"s":[150.18,5.281,0],"to":[-5.195,-1.594,0],"ti":[-0.953,-5.614,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":106,"s":[134.453,14.614,0],"to":[0.953,5.614,0],"ti":[-3.875,4.525,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":137,"s":[149.75,10.975,0],"to":[3.875,-4.525,0],"ti":[1.75,-5.25,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":170,"s":[140.625,8.625,0],"to":[-1.75,5.25,0],"ti":[11.125,-4.375,0]},{"t":199,"s":[141.25,25,0]}],"ix":2},"a":{"a":0,"k":[197.25,-284,0],"ix":1},"s":{"a":0,"k":[82,82,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[192.5,-319.5],[250.5,-319.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.235294133425,0.509803950787,0.945098102093,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":10,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[192.5,-298],[313,-298]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"rp","c":{"a":0,"k":3,"ix":1},"o":{"a":0,"k":0,"ix":2},"m":1,"ix":2,"tr":{"ty":"tr","p":{"a":0,"k":[0,21],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":4},"so":{"a":0,"k":100,"ix":5},"eo":{"a":0,"k":100,"ix":6},"nm":"Transform"},"nm":"Repeater 1","mn":"ADBE Vector Filter - Repeater","hd":false},{"ty":"st","c":{"a":0,"k":[0.874463677406,0.896885812283,0.952941179276,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":10,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-10.631,0],[0,-10.631],[8.016,-3.698],[-1.065,-13.634],[13.375,0],[4.627,5.177],[1.051,0.57],[0,8.437]],"o":[[10.631,0],[0,8.766],[-0.831,0.384],[-4.815,6.616],[-14.376,0],[0.627,-13.698],[-7.554,-4.095],[0,-10.631]],"v":[[-74,-32],[-54.75,-12.75],[-68.625,11.021],[-43.935,27.259],[-74,40.25],[-105.002,26.948],[-80.268,10.573],[-93.25,-12.75]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 3","mn":"ADBE Vector Graphic - Fill","hd":false},{"d":1,"ty":"el","s":{"a":0,"k":[93.5,93.5],"ix":2},"p":{"a":0,"k":[-74,0],"ix":3},"nm":"Ellipse Path 2","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"fl","c":{"a":0,"k":[0.874509871006,0.898039281368,0.95294123888,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"rc","d":1,"s":{"a":0,"k":[267.5,111],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":10,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"rc","d":1,"s":{"a":0,"k":[267.5,111],"ix":2},"p":{"a":0,"k":[-6,6],"ix":3},"r":{"a":0,"k":10,"ix":4},"nm":"Rectangle Path 2","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[0,0,0,1],"ix":4},"o":{"a":0,"k":17,"ix":5},"r":1,"bm":0,"nm":"Fill 4","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[200.25,-287],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":8,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":200,"st":0,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"mail 6","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":22,"s":[15.427]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":56,"s":[-12.946]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":90,"s":[28.456]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":126,"s":[-12.946]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":167,"s":[24]},{"t":199,"s":[0]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":0,"s":[-192.052,-110.008,0],"to":[2.552,4.383,0],"ti":[6.537,13.177,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":51,"s":[-171.912,-105.677,0],"to":[-6.537,-13.177,0],"ti":[6.357,-1.675,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":85,"s":[-191.107,-93.45,0],"to":[-6.357,1.675,0],"ti":[8.571,-10.521,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":124,"s":[-212.946,-117.604,0],"to":[-8.571,10.521,0],"ti":[9.927,9.883,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":163,"s":[-174.552,-122.008,0],"to":[-9.927,-9.883,0],"ti":[-4.323,-4.992,0]},{"t":199,"s":[-192.052,-110.008,0]}],"ix":2},"a":{"a":0,"k":[2153.478,680.843,0],"ix":1},"s":{"a":0,"k":[67,67,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[4.932,0],[0,5.066],[-8.798,0],[-1.066,-0.267]],"o":[[-1.2,7.732],[-4.398,0],[0,-10.264],[1.732,0],[0,0]],"v":[[6.604,-0.329],[-4.726,13.668],[-11.391,5.403],[4.339,-13.526],[8.47,-12.993]],"c":true},"ix":2},"nm":"@","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[6.798,0],[0,22.262],[-20.396,0],[0,-17.33],[5.73,0.205],[-1.333,7.865],[0,0],[5.865,0],[0,-16.263],[-8.665,0],[-4.133,6.799],[0,0],[-6.266,0],[0,19.462],[23.461,0],[0,-27.194],[-22.127,0],[-7.998,3.599]],"o":[[-5.333,2.799],[-19.862,0],[0,-25.461],[20.662,0],[0,15.33],[-3.732,-0.133],[0,0],[-3.333,-1.866],[-19.196,0],[0,10.931],[7.065,0],[0,0],[0.667,7.065],[14.396,0],[0,-22.395],[-30.127,0],[0,26.128],[9.465,0],[0,0]],"v":[[17.936,34.33],[-1.393,38.596],[-35.519,3.67],[2.74,-37.787],[34.598,-6.994],[21.668,15.534],[18.336,3.937],[22.335,-20.724],[6.472,-24.057],[-24.722,7.003],[-9.791,24.332],[7.405,14.068],[7.671,14.068],[19.136,24.332],[44.463,-7.394],[4.206,-45.919],[-45.65,4.87],[-4.06,46.861],[20.335,42.062]],"c":true},"ix":2},"nm":"@ 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.227450996637,0.509803950787,0.95294123888,0.988235354424],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 3","mn":"ADBE Vector Graphic - Fill","hd":false},{"d":1,"ty":"el","s":{"a":0,"k":[126.905,126.905],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"fl","c":{"a":0,"k":[0.885428667068,0.920325279236,0.960784316063,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[2140.425,652.021],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":5,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[126.905,126.905],"ix":2},"p":{"a":0,"k":[-4,11],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"fl","c":{"a":0,"k":[0,0,0,1],"ix":4},"o":{"a":0,"k":18,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[2140.425,652.021],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":200,"st":0,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"mail 3","parent":1,"sr":1,"ks":{"o":{"a":0,"k":99,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[183.055,-315.211,0],"ix":2},"a":{"a":0,"k":[2153.478,680.843,0],"ix":1},"s":{"a":0,"k":[56,56,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":1,"y":0},"t":5,"s":[{"i":[[1.212,0.293],[-0.053,1.063],[-0.465,1.092],[-2.964,0.923],[0,0],[-2.408,-1.462],[-0.75,-0.938],[9.093,-10.048]],"o":[[-13.028,-3.146],[0.06,-1.209],[1.127,-2.65],[0,0],[2.9,-0.902],[1.008,0.612],[0.699,0.874],[-0.845,0.934]],"v":[[11.967,33.017],[-77.516,10.287],[-76.716,6.809],[-70.385,1.081],[66.005,-41.368],[74.313,-40.349],[76.976,-38.011],[15.15,32.025]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":10,"s":[{"i":[[1.113,-0.027],[12.953,-2.709],[-0.38,0.852],[-2.964,0.923],[0,0],[-2.227,-0.88],[0.429,-0.708],[10.164,-6.61]],"o":[[-11.959,0.286],[-0.893,-0.313],[0.756,-1.716],[0,0],[2.9,-0.903],[0.932,0.368],[-12.428,5.23],[-0.945,0.614]],"v":[[4.34,8.53],[-73.957,7.989],[-75.054,6.191],[-70.014,2.274],[66.374,-40.184],[72.789,-39.988],[73.406,-38.058],[7.523,7.538]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":11,"s":[{"i":[[1.022,-0.318],[22.981,-7.154],[-0.472,0.148],[-2.964,0.923],[0,0],[-1.781,0.553],[1.404,-0.438],[11.142,-3.471]],"o":[[-10.983,3.42],[-1.629,0.507],[0.69,-0.217],[0,0],[2.899,-0.903],[0.745,-0.231],[-23.438,7.303],[-1.036,0.323]],"v":[[-4.379,-19.464],[-72.465,1.734],[-74.732,2.437],[-70.385,1.081],[65.996,-41.397],[70.589,-42.826],[69.196,-42.389],[-1.196,-20.456]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":12,"s":[{"i":[[0,0],[0.003,-0.005],[-0.001,-0.001],[-0.001,0],[0,0],[0.001,0.003],[0,0],[0.002,0]],"o":[[-0.002,0.002],[0,0],[0.001,0.002],[0,0],[0.001,0],[0,-0.001],[-0.005,-0.003],[0,0]],"v":[[-73.926,5.759],[-73.934,5.774],[-73.933,5.778],[-73.932,5.781],[-73.908,5.773],[-73.908,5.77],[-73.909,5.767],[-73.925,5.759]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":87,"s":[{"i":[[0,0],[0.003,-0.005],[-0.001,-0.001],[0,0],[0,0],[0.001,0.003],[0,0],[0.002,0]],"o":[[-0.001,0.002],[0,0],[0.001,0.002],[0,0],[0,0],[0,-0.001],[-0.005,-0.003],[0,0]],"v":[[-73.926,5.759],[-73.934,5.774],[-73.933,5.778],[-73.932,5.781],[-73.908,5.773],[-73.908,5.77],[-73.909,5.767],[-73.925,5.759]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":88,"s":[{"i":[[1.022,-0.318],[22.981,-7.154],[-0.472,0.148],[-2.964,0.923],[0,0],[-1.781,0.552],[1.404,-0.438],[11.142,-3.471]],"o":[[-10.983,3.42],[-1.629,0.507],[0.69,-0.217],[0,0],[2.899,-0.903],[0.745,-0.231],[-23.438,7.303],[-1.036,0.323]],"v":[[-4.379,-19.464],[-72.465,1.734],[-74.732,2.437],[-70.385,1.081],[65.996,-41.397],[70.589,-42.826],[69.197,-42.389],[-1.196,-20.456]],"c":true}]},{"i":{"x":0,"y":1},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[1.113,-0.027],[12.953,-2.709],[-0.38,0.852],[-2.964,0.923],[0,0],[-2.227,-0.88],[0.429,-0.708],[10.164,-6.61]],"o":[[-11.959,0.286],[-0.893,-0.313],[0.756,-1.716],[0,0],[2.9,-0.903],[0.932,0.368],[-12.428,5.23],[-0.945,0.614]],"v":[[4.34,8.53],[-73.957,7.989],[-75.054,6.191],[-70.014,2.274],[66.374,-40.184],[72.789,-39.988],[73.406,-38.058],[7.523,7.538]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":94,"s":[{"i":[[1.212,0.293],[-0.052,1.063],[-0.465,1.092],[-2.964,0.923],[0,0],[-2.408,-1.462],[-0.75,-0.938],[9.093,-10.048]],"o":[[-13.028,-3.146],[0.06,-1.209],[1.127,-2.65],[0,0],[2.9,-0.902],[1.008,0.612],[0.699,0.874],[-0.845,0.934]],"v":[[11.967,33.017],[-77.516,10.287],[-76.716,6.809],[-70.385,1.081],[66.005,-41.368],[74.313,-40.349],[76.976,-38.011],[15.15,32.025]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":1,"y":0},"t":106,"s":[{"i":[[1.212,0.293],[-0.052,1.063],[-0.465,1.092],[-2.964,0.923],[0,0],[-2.408,-1.462],[-0.75,-0.938],[9.093,-10.048]],"o":[[-13.028,-3.146],[0.06,-1.209],[1.127,-2.65],[0,0],[2.9,-0.902],[1.008,0.612],[0.699,0.874],[-0.845,0.934]],"v":[[11.967,33.017],[-77.516,10.287],[-76.716,6.809],[-70.385,1.081],[66.005,-41.368],[74.313,-40.349],[76.976,-38.011],[15.15,32.025]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":111,"s":[{"i":[[1.113,-0.027],[12.953,-2.709],[-0.38,0.852],[-2.964,0.923],[0,0],[-2.227,-0.88],[0.429,-0.708],[10.164,-6.61]],"o":[[-11.959,0.286],[-0.893,-0.313],[0.756,-1.716],[0,0],[2.9,-0.903],[0.932,0.368],[-12.428,5.23],[-0.945,0.614]],"v":[[4.34,8.53],[-73.957,7.989],[-75.054,6.191],[-70.014,2.274],[66.374,-40.184],[72.789,-39.988],[73.406,-38.058],[7.523,7.538]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":112,"s":[{"i":[[1.022,-0.318],[22.981,-7.154],[-0.472,0.148],[-2.964,0.923],[0,0],[-1.781,0.552],[1.404,-0.438],[11.142,-3.471]],"o":[[-10.983,3.42],[-1.629,0.507],[0.69,-0.217],[0,0],[2.899,-0.903],[0.745,-0.231],[-23.438,7.303],[-1.036,0.323]],"v":[[-4.379,-19.464],[-72.465,1.734],[-74.732,2.437],[-70.385,1.081],[65.996,-41.397],[70.589,-42.826],[69.197,-42.389],[-1.196,-20.456]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":113,"s":[{"i":[[0,0],[0.003,-0.005],[-0.001,-0.001],[0,0],[0,0],[0.001,0.003],[0,0],[0.002,0]],"o":[[-0.001,0.002],[0,0],[0.001,0.002],[0,0],[0,0],[0,-0.001],[-0.005,-0.003],[0,0]],"v":[[-73.926,5.759],[-73.934,5.774],[-73.933,5.778],[-73.932,5.781],[-73.908,5.773],[-73.908,5.77],[-73.909,5.767],[-73.925,5.759]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":188,"s":[{"i":[[0,0],[0.003,-0.005],[-0.001,-0.001],[0,0],[0,0],[0.001,0.003],[0,0],[0.002,0]],"o":[[-0.001,0.002],[0,0],[0.001,0.002],[0,0],[0,0],[0,-0.001],[-0.005,-0.003],[0,0]],"v":[[-73.926,5.759],[-73.934,5.774],[-73.933,5.778],[-73.932,5.781],[-73.908,5.773],[-73.908,5.77],[-73.909,5.767],[-73.925,5.759]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":189,"s":[{"i":[[1.022,-0.318],[22.981,-7.154],[-0.472,0.148],[-2.964,0.923],[0,0],[-1.781,0.552],[1.404,-0.438],[11.142,-3.471]],"o":[[-10.983,3.42],[-1.629,0.507],[0.69,-0.217],[0,0],[2.899,-0.903],[0.745,-0.231],[-23.438,7.303],[-1.036,0.323]],"v":[[-4.379,-19.464],[-72.465,1.734],[-74.732,2.437],[-70.385,1.081],[65.996,-41.397],[70.589,-42.826],[69.197,-42.389],[-1.196,-20.456]],"c":true}]},{"i":{"x":0,"y":1},"o":{"x":0.167,"y":0.167},"t":190,"s":[{"i":[[1.113,-0.027],[12.953,-2.709],[-0.38,0.852],[-2.964,0.923],[0,0],[-2.227,-0.88],[0.429,-0.708],[10.164,-6.61]],"o":[[-11.959,0.286],[-0.893,-0.313],[0.756,-1.716],[0,0],[2.9,-0.903],[0.932,0.368],[-12.428,5.23],[-0.945,0.614]],"v":[[4.34,8.53],[-73.957,7.989],[-75.054,6.191],[-70.014,2.274],[66.374,-40.184],[72.789,-39.988],[73.406,-38.058],[7.523,7.538]],"c":true}]},{"t":195,"s":[{"i":[[1.212,0.293],[-0.052,1.063],[-0.465,1.092],[-2.964,0.923],[0,0],[-2.408,-1.462],[-0.75,-0.938],[9.093,-10.048]],"o":[[-13.028,-3.146],[0.06,-1.209],[1.127,-2.65],[0,0],[2.9,-0.902],[1.008,0.612],[0.699,0.874],[-0.845,0.934]],"v":[[11.967,33.017],[-77.516,10.287],[-76.716,6.809],[-70.385,1.081],[66.005,-41.368],[74.313,-40.349],[76.976,-38.011],[15.15,32.025]],"c":true}]}],"ix":2},"nm":"Path 6","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.401014983654,0.635570943356,0.964705884457,1],"ix":4},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":87,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":88,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":111,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":112,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":188,"s":[0]},{"t":189,"s":[100]}],"ix":5},"r":1,"bm":0,"nm":"Fill 6","mn":"ADBE Vector Graphic - Fill","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,0],[4.2,-1.308],[0,0],[2.5,3.352],[0,0],[-2.717,0.813],[0,0]],"o":[[-0.18,4.162],[0,0],[-4.22,1.315],[0,0],[0,0],[2.717,-0.813],[0,0]],"v":[[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[5.902,26.402],[11.27,22.465],[17.541,22.927]],"c":true},"ix":2},"nm":"Path 5","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.349019616842,0.592156887054,0.933333396912,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 5","mn":"ADBE Vector Graphic - Fill","hd":false},{"ind":4,"ty":"sh","ix":5,"ks":{"a":0,"k":{"i":[[0,0],[4.2,-1.308],[0,0],[2.53,3.45],[0,0],[-2.717,0.812],[0,0]],"o":[[-0.149,4.26],[0,0],[-4.22,1.315],[0,0],[0,0],[2.717,-0.812],[0,0]],"v":[[103.573,47.134],[96.485,56.565],[-39.905,99.013],[-51.118,95.217],[5.469,25.011],[10.82,21.02],[17.108,21.54]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.1254902035,0.1254902035,0.1254902035,1],"ix":4},"o":{"a":0,"k":30,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ind":6,"ty":"sh","ix":7,"ks":{"a":0,"k":{"i":[[-1.69,-5.354],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-6.374,-1.622],[0,0],[0,0]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-1.67,-5.372],[0,0],[0,0],[4.139,-4.613]],"v":[[78.958,-33.99],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-76.841,14.537],[-68.329,7.683],[12.317,29.181],[68.061,-34.766]],"c":true},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.227450996637,0.509803950787,0.95294123888,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 4","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1801.139,934.164],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[594,594],"ix":3},"r":{"a":0,"k":17.3,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"front","np":8,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[1741.752,1155.556],[1787.515,1141.378]],"c":false},"ix":2},"nm":"Path 6","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[1746.435,1170.589],[1844.124,1140.151]],"c":false},"ix":2},"nm":"Path 5","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"rp","c":{"a":0,"k":4,"ix":1},"o":{"a":0,"k":0,"ix":2},"m":1,"ix":3,"tr":{"ty":"tr","p":{"a":0,"k":[5,15],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":4},"so":{"a":0,"k":100,"ix":5},"eo":{"a":0,"k":100,"ix":6},"nm":"Transform"},"nm":"Repeater 1","mn":"ADBE Vector Filter - Repeater","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":17,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":46,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":105,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":106,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":118,"s":[0]},{"t":147,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":2,"ix":4,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.750726640224,0.811723232269,0.886274516582,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-1792,-1153],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":5,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-1.152,-3.69],[0,0],[3.69,-1.152],[0,0],[1.152,3.69],[0,0],[-3.69,1.152],[0,0]],"o":[[0,0],[1.152,3.69],[0,0],[-3.69,1.152],[0,0],[-1.152,-3.69],[0,0],[3.69,-1.152]],"v":[[1861.208,1111.749],[1886.121,1191.558],[1881.524,1200.325],[1753.446,1240.409],[1744.678,1235.813],[1719.765,1156.005],[1724.362,1147.237],[1852.441,1107.153]],"c":true},"ix":2},"nm":"Path 5","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.926412940025,0.952468276024,0.984313726425,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 3","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-1792,-1153],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":14,"s":[1801.139,977.164],"to":[0,-35.5],"ti":[0,35.5]},{"t":29,"s":[1801.139,764.164],"h":1},{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":72,"s":[1801.139,764.164],"to":[0,35.5],"ti":[0,-35.5]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":87,"s":[1801.139,977.164],"to":[0,35.5],"ti":[0,0]},{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":115,"s":[1801.139,977.164],"to":[0,-35.5],"ti":[0,35.5]},{"t":130,"s":[1801.139,764.164],"h":1},{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":173,"s":[1801.139,764.164],"to":[0,35.5],"ti":[0,-35.5]},{"t":188,"s":[1801.139,977.164]}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[594,594],"ix":3},"r":{"a":0,"k":17.3,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":10,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-14.49,5.196],[-2.265,0.603],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.197,-0.788],[14.919,-3.974],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-6.935,-15.232],[0.09,-17.423],[74.435,-40.273]],"c":true}]},{"i":{"x":0,"y":1},"o":{"x":0.167,"y":0.167},"t":11,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-13.999,6.772],[-2.338,0.367],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.122,-1.027],[15.402,-2.421],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-11.418,-32.357],[-4.394,-34.553],[74.435,-40.273]],"c":true}]},{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":18,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-11.355,15.263],[-2.734,-0.902],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[1.721,-2.314],[18.008,5.945],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-21.154,-67.958],[-14.138,-70.18],[74.435,-40.273]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":1,"y":0},"t":81,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-11.355,15.263],[-2.734,-0.902],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[1.721,-2.314],[18.008,5.945],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-21.154,-67.958],[-14.138,-70.18],[74.435,-40.273]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":88,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-13.999,6.772],[-2.338,0.367],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.122,-1.027],[15.402,-2.421],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-11.418,-32.357],[-4.394,-34.553],[74.435,-40.273]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":89,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-14.49,5.196],[-2.265,0.603],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.197,-0.788],[14.919,-3.974],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-6.935,-15.232],[0.09,-17.423],[74.435,-40.273]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":111,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-14.49,5.196],[-2.265,0.603],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.197,-0.788],[14.919,-3.974],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-6.935,-15.232],[0.09,-17.423],[74.435,-40.273]],"c":true}]},{"i":{"x":0,"y":1},"o":{"x":0.167,"y":0.167},"t":112,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-13.999,6.772],[-2.338,0.367],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.122,-1.027],[15.402,-2.421],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-11.418,-32.357],[-4.394,-34.553],[74.435,-40.273]],"c":true}]},{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":119,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-11.355,15.263],[-2.734,-0.902],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[1.721,-2.314],[18.008,5.945],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-21.154,-67.958],[-14.138,-70.18],[74.435,-40.273]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":1,"y":0},"t":182,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-11.355,15.263],[-2.734,-0.902],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[1.721,-2.314],[18.008,5.945],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-21.154,-67.958],[-14.138,-70.18],[74.435,-40.273]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":189,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-13.999,6.772],[-2.338,0.367],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.122,-1.027],[15.402,-2.421],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-11.418,-32.357],[-4.394,-34.553],[74.435,-40.273]],"c":true}]},{"t":190,"s":[{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-14.49,5.196],[-2.265,0.603],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.197,-0.788],[14.919,-3.974],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-6.935,-15.232],[0.09,-17.423],[74.435,-40.273]],"c":true}]}],"ix":2},"nm":"Path 4","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.174548238516,0.452203959227,0.890196084976,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 3","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1801.139,934.164],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[594,594],"ix":3},"r":{"a":0,"k":17.3,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"back 2","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-0.758,-2.403],[0,0],[0.05,-1.142],[4.2,-1.308],[0,0],[2.5,3.352],[0.36,1.136],[0,0],[-0.788,2.059],[-14.49,5.196],[-2.265,0.603],[-2.109,-1.321]],"o":[[0,0],[0.36,1.157],[-0.18,4.162],[0,0],[-4.22,1.315],[-0.68,-0.903],[0,0],[-0.707,-2.273],[0.933,-2.439],[2.197,-0.788],[14.919,-3.974],[1.978,1.239]],"v":[[78.735,-34.708],[103.175,43.824],[103.625,47.302],[96.485,56.565],[-39.905,99.014],[-51.065,95.387],[-52.635,92.318],[-77.065,13.819],[-76.632,7.874],[-6.935,-15.232],[0.09,-17.423],[74.435,-40.273]],"c":true},"ix":2},"nm":"Path 5","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0,0,0,1],"ix":4},"o":{"a":0,"k":20,"ix":5},"r":1,"bm":0,"nm":"Fill 4","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1785.139,947.164],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[594,594],"ix":3},"r":{"a":0,"k":17.3,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"back","np":2,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":204,"st":-23,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"leaf 3","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[196,-240,0],"ix":2},"a":{"a":0,"k":[1500,1000.5,0],"ix":1},"s":{"a":0,"k":[56,56,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0.52,-0.05],[15.722,26.95],[21.544,0],[35.341,11.01],[18.56,8.32],[-0.503,0.85],[-32.295,-10.05],[-43.131,0],[-18.804,6.14],[53.395,57.16],[53.175,-22.35],[0.078,1.07],[-71.736,2.18],[38.007,34.2],[6.59,0],[35.162,-3.87],[48.565,-16.86],[-0.019,0.91],[-0.004,0.17],[-36.694,4.03],[-15.873,0],[-2.179,-0.05],[16.272,13.78],[46.573,31.822],[-0.864,0.642],[-96.288,-81.53],[-30.176,-28.64],[6.164,41.47],[-7.044,39.89],[-0.121,0],[-0.907,-0.082],[-5.287,-35.58],[-1.283,-4.23],[-31.301,-36.06],[-0.128,-0.15],[-21.362,32.62],[-1.781,2.47],[-0.853,-0.58],[1.847,-2.83],[1.964,-100.57],[-22.107,-33.25],[-15.101,23.19],[0,0],[18.79,-15.47],[3.888,-2.83],[-14.087,-36.96],[0.026,-0.05],[0,0],[0,-0.01],[0,0],[0,0],[0,0]],"o":[[-10.195,-26.92],[-19.289,6.4],[-43.533,0],[-32.442,-10.1],[0.365,-0.95],[18.478,8.28],[35.105,10.92],[21.011,0],[-36.858,-62.52],[-71.572,1.17],[-0.162,-0.99],[52.999,-22.24],[-37.629,-40.07],[-1.487,-0.05],[-15.789,0],[-36.753,4.04],[-0.049,-0.86],[0.005,-0.17],[48.538,-16.74],[35.271,-3.88],[3.726,0],[-17.173,-15.39],[-96.765,-81.94],[0.811,-0.642],[46.879,32.06],[34.528,29.24],[-3.716,-13],[-5.304,-35.69],[0.12,-0.001],[0.865,0],[-7.072,39.563],[7.363,49.53],[39.468,37.69],[0.128,0.14],[2.483,-99.44],[1.801,-2.75],[0.83,0.52],[-1.835,2.53],[-21.321,32.64],[30.334,35.22],[25.708,-18.68],[0,0],[-13.962,21.17],[-3.711,3.06],[26.711,40.49],[-0.569,1.13],[0,0],[-0.001,0],[0,0],[0,0],[0,0],[-0.521,0.05]],"v":[[255.766,316.768],[216.528,235.818],[154.584,244.798],[30.443,223.238],[-48.814,193.138],[-47.513,190.428],[31.384,220.388],[154.559,241.798],[214.959,233.168],[75.055,51.768],[-160.378,118.009],[-160.74,114.908],[72.293,48.818],[-42.731,-63.242],[-55.102,-63.412],[-135.027,-58.872],[-269.658,-29.702],[-269.711,-32.362],[-269.694,-32.862],[-135.302,-61.862],[-55.074,-66.412],[-46.17,-66.332],[-96.486,-110.152],[-330.557,-288.331],[-328.049,-290.257],[-94.545,-112.452],[2.575,-25.562],[-16.736,-115.122],[-18.595,-236.622],[-18.232,-236.624],[-15.577,-236.505],[-13.763,-115.562],[6.917,-21.432],[113.245,89.358],[113.624,89.799],[163.293,-101.922],[168.667,-109.731],[171.191,-108.082],[165.664,-100.052],[116.551,93.188],[195.269,195.978],[255.786,131.648],[257.595,134.588],[208.343,189.678],[196.939,198.499],[258.235,314.828],[257.327,316.628],[257.327,316.628],[257.326,316.638],[257.326,316.638],[257.324,316.638],[257.324,316.638]],"c":true},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.038554400206,0.285618126392,0.427450984716,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,0],[-2.571,34.45],[3.084,43.56],[43.327,-9.15],[-2.839,43.58],[-8.23,67.7],[47.117,-4.25],[-0.732,34.35],[63,43.63],[-107.885,59.634],[-44.46,-49.284],[-10.216,5.71],[-45.98,-14.077],[-28.6,19.15],[-50.281,-35.21],[-9.916,-48.08],[0,0],[0,0]],"o":[[-165.658,16.23],[2.57,-34.44],[-1.788,-25.24],[-50.433,10.65],[3.265,-50.13],[4.104,-33.77],[-47.371,4.27],[1.46,-68.54],[-62.999,-43.63],[59.009,-32.618],[36.825,40.819],[22.549,-12.62],[45.98,14.074],[28.598,-19.16],[50.281,35.21],[9.916,48.09],[52.841,-105.53],[101.227,31.71]],"v":[[257.324,316.638],[-50.066,198.838],[26.462,137.818],[-59.895,130.878],[-160.749,108.598],[-85.562,18.719],[-174.718,-6.552],[-269.711,-32.362],[-213.42,-133.032],[-312.32,-300.251],[-150.486,-234.301],[-76.158,-158.372],[-7.843,-234.946],[82.178,-44.552],[171.887,-107.612],[174.944,60.078],[224.503,87.928],[298.275,6.368]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.086274512112,0.355340272188,0.509803950787,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[776.13,1559.822],"ix":2},"a":{"a":0,"k":[252.778,308.333],"ix":1},"s":{"a":0,"k":[79,79],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[8]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":32,"s":[3]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":77,"s":[8]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":136,"s":[3]},{"t":199,"s":[8]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 62","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[241.821,-10.2],[0,0],[0,0],[95.49,-34.36],[-51.858,-9.43],[0,0],[0,0],[-116.191,13.01]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[116.191,-13.01]],"v":[[315.451,8.94],[-55.948,-125.96],[-13.934,-68.12],[-81.586,-121.09],[-317.278,-74.91],[-59.935,42.4],[9.245,21.74],[-28.17,73.51],[201.087,123.15]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.363921552896,0.909803926945,0.864848911762,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[818.171,1522.597],"ix":2},"a":{"a":0,"k":[277.778,30.556],"ix":1},"s":{"a":0,"k":[79,79],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[-24]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":47,"s":[-21]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":99,"s":[-24]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":155,"s":[-21]},{"t":199,"s":[-24]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 59","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-17.327,35.044],[0,0],[13.781,7.041],[0,0],[0,0],[11.484,9.404],[0,0],[0,0],[0,0],[38.997,-12.255],[0,0],[10.845,-4.755],[0,0],[-1.813,0.784],[0,0],[0,0],[-2.051,0.78],[0,0],[0,0],[-2.342,0.994],[0,0],[-8.098,-1.219],[-0.749,1.903],[0.27,-1.086],[-35.396,-15.316],[6.324,21.338],[6.24,10.822],[-19.384,-14.548],[-30.872,-23.95],[0,0],[-4.168,6.938],[-11.2,-4.394]],"o":[[17.328,-35.045],[0,0],[-13.781,-7.041],[-22.754,-17.704],[0,0],[-11.484,-9.404],[-28.863,-10.099],[-60.399,15.696],[0,0],[0,0],[0,0],[0,0],[1.79,-0.952],[0,0],[0,0],[1.17,-0.457],[0,0],[0,0],[4.779,-1.609],[0,0],[14.735,-5.227],[0,0],[-0.248,0.993],[-6.196,24.92],[35.396,15.316],[-6.326,-21.344],[-6.235,-10.813],[19.375,14.541],[30.871,23.95],[-28.077,-9.588],[4.167,-6.937],[11.2,4.394]],"v":[[206.56,28.825],[151.299,-38.985],[88.847,-36.229],[118.427,-55.556],[11.44,-91.41],[-27.295,-61.107],[-2.416,-91.742],[-73.512,-95.551],[-98.836,-16.496],[-156.585,16.364],[-156.585,16.364],[-175.312,22.938],[-168.851,27.142],[-163.456,24.528],[-163.419,24.516],[-163.339,24.482],[-158.445,22.598],[-158.442,22.593],[-158.427,22.592],[-147.889,18.725],[-147.868,18.715],[-105.019,8.428],[-104.254,5.309],[-105.019,8.428],[-23.419,87.593],[-1.44,66.332],[-30.523,33.292],[-7.472,36.239],[26.276,84.231],[127.822,77.943],[103.376,52.768],[147.656,62.865]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.086274512112,0.35686275363,0.509803950787,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1548.444,1508.903],"ix":2},"a":{"a":0,"k":[-158.915,-25.046],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[5.188]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":29,"s":[-2.812]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":61,"s":[5.188]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":97,"s":[-2.812]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":138,"s":[5.188]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":172,"s":[-2.812]},{"t":199,"s":[5.188]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 23","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-72.4,84.95],[0,0],[0,0],[-45.06,19.7],[0,0],[0,0],[0,0],[0,0]],"o":[[27.07,8.84],[0,0],[39.11,-38.04],[21.97,25.45],[0,0],[122.11,-40.8],[-139.9,286.28],[-19.88,-196.75]],"v":[[-156.955,-120.915],[-89.775,-98.475],[-132.365,-147.195],[-3.555,-232.345],[44.065,-178.565],[40.695,-249.335],[275.615,-280.815],[-255.735,290.135]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.161906957626,0.703410863876,0.737254917622,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[7.46,89.64],[0,0],[-141.07,232.92],[0,0],[-40.31,-0.18]],"o":[[-139.9,286.28],[0,0],[0,0],[0,0],[29.92,-64.87]],"v":[[275.615,-280.815],[-255.735,290.135],[214.515,-6.555],[149.255,-49.445],[237.575,-49.925]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.113725498319,0.600000023842,0.674509823322,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1601.536,1460.812],"ix":2},"a":{"a":0,"k":[-258.333,261.111],"ix":1},"s":{"a":0,"k":[67,67],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[-14]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":43,"s":[-22]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":113,"s":[-14]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":158,"s":[-22]},{"t":199,"s":[-14]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 65","np":4,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":200,"st":0,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"leaf","parent":1,"sr":1,"ks":{"o":{"a":0,"k":85,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[196,-64,0],"ix":2},"a":{"a":0,"k":[1500,1000.5,0],"ix":1},"s":{"a":0,"k":[56,56,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[14.25,30.88],[0,0],[-9.27,5.98],[0,0],[0,0],[-7.51,8.07],[0,0],[0,0],[0,0],[-28.13,-11.24],[0,0],[-7.9,-4.3],[0,0],[1.32,0.71],[0,0],[0,0],[1.49,0.71],[0,0],[1.71,0.9],[0,0],[5.61,-0.95],[0.64,1.67],[-0.26,-0.95],[23.97,-12.92],[-3.18,18.58],[-3.75,9.38],[12.76,-12.46],[20.28,-20.53],[0,0],[3.34,6.12],[7.61,-3.69]],"o":[[-14.25,-30.89],[0,0],[9.27,-5.97],[14.94,-15.18],[0,0],[7.52,-8.07],[19.69,-8.44],[43.38,14.54],[0,0],[0,0],[0,0],[0,0],[-1.31,-0.86],[0,0],[0,0],[-0.85,-0.41],[0,0],[-3.46,-1.47],[0,0],[-10.67,-4.77],[0,0],[0.23,0.87],[5.82,21.89],[-23.97,12.92],[3.18,-18.59],[3.74,-9.37],[-12.76,12.46],[-20.27,20.53],[19.17,-8],[-3.34,-6.13],[-7.61,3.7]],"v":[[-126.89,19.835],[-92.06,-38.735],[-17.07,-24.785],[-69.93,-52.795],[3.14,-82.705],[51.04,-37.175],[12.86,-82.815],[62.6,-85.185],[85.06,-15.695],[127.59,13.825],[127.6,13.825],[141.14,19.825],[136.85,23.415],[132.9,21.065],[132.88,21.045],[132.82,21.015],[129.27,19.305],[129.26,19.295],[121.62,15.775],[121.61,15.765],[90.89,6.185],[90.16,3.455],[90.89,6.185],[38.21,74.335],[21.51,55.445],[53.59,10.635],[31.53,18.385],[3.08,70.725],[-68.66,63.855],[-32.57,18.515],[-83.48,50.395]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.364705890417,0.909803986549,0.866666734219,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1214.31,722.068],"ix":2},"a":{"a":0,"k":[119.643,-16.071],"ix":1},"s":{"a":0,"k":[-136,136],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[-53]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":56,"s":[-50]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":98,"s":[-53]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":157,"s":[-50]},{"t":199,"s":[-53]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 15","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[81.51,-75.711],[-128.3,56.065],[-2.22,0],[-2.69,-5.596],[5.2,-10.51]],"o":[[0,0],[2.13,-0.931],[5.93,0],[2.42,5.01],[-13.86,28.028]],"v":[[-86.345,139.385],[136.205,-73.643],[142.805,-75.019],[157.135,-66.244],[154.665,-43.433]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[45.26,-73.28],[-81,69.392],[-3.32,0],[1.78,-10.112],[0.05,-0.238]],"o":[[0,0],[2.96,-2.541],[8.7,0],[-0.04,0.233],[-4.57,23.769]],"v":[[-72.085,102.304],[45.575,-98.64],[55.255,-102.304],[70.305,-84.442],[70.175,-83.736]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.086274512112,0.35686275363,0.509803950787,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[904.419,751.634],"ix":2},"a":{"a":0,"k":[-71.429,132.143],"ix":1},"s":{"a":0,"k":[-100,100],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[1]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":36,"s":[-4]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":117,"s":[1]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":174,"s":[-4]},{"t":199,"s":[1]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 22","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-26.95,51.06],[0,0],[-17.61,-6.33],[0,0],[0,0],[-18.63,-1.8],[0,0],[0,0],[0,0],[-13.85,-49.51],[0,0],[-2.34,-15.09],[0,0],[0.4,2.51],[0,0],[0,0],[0.57,2.74],[0,0],[0,0],[0.54,3.23],[0,0],[7.08,6.58],[-1.59,2.59],[1.03,-1.33],[42.13,18.99],[-28.33,14.87],[-16.51,4.62],[29.96,4.35],[48.58,6.17],[0,0],[-4.81,10.82],[12.83,6.46]],"o":[[26.94,-51.05],[0,0],[17.61,6.34],[35.87,4.49],[0,0],[18.63,1.81],[31.68,17.83],[25.13,73.47],[0,0],[0,0],[0,0],[0,0],[-0.2,-2.65],[0,0],[0,0],[-0.31,-1.57],[0,0],[0,0],[-1.58,-6.18],[0,0],[-4.57,-19.29],[0,0],[-0.93,1.21],[-23.49,30.41],[-42.13,-18.99],[28.33,-14.87],[16.5,-4.62],[-29.95,-4.36],[-48.58,-6.16],[30.55,17.57],[4.81,-10.81],[-12.82,-6.45]],"v":[[-167.555,-155.115],[-52.705,-168.555],[5.815,-53.085],[-10.935,-153.215],[104.725,-85.575],[92.745,25.945],[114.895,-72.575],[169.375,-7.985],[98.875,93.955],[102.935,181.715],[102.935,181.715],[108.815,206.165],[99.555,204.085],[98.655,196.335],[98.645,196.285],[98.625,196.175],[97.285,189.625],[97.275,189.615],[97.275,189.605],[94.155,175.675],[94.155,175.645],[75.375,124.365],[78.315,120.565],[75.375,124.365],[-70.795,123.625],[-62.545,81.635],[30.925,78.675],[-2.265,56.945],[-102.145,72.555],[-166.855,-31.215],[-68.525,-29.345],[-164.005,-65.085]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.086274512112,0.35686275363,0.509803950787,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1619.554,1158.989],"ix":2},"a":{"a":0,"k":[100,194.643],"ix":1},"s":{"a":0,"k":[-100,100],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[25]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":34,"s":[33]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":71,"s":[22.806]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":124,"s":[29.894]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":167,"s":[20.914]},{"t":199,"s":[25]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 16","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[12.16,44.9],[0,0],[14.97,-1.87],[0,0],[0,0],[14.96,1.87],[0,0],[0,0],[0,0],[19.64,-36.49],[0,0],[4.5,-11.45],[0,0],[-0.76,1.9],[0,0],[0,0],[-0.94,2.04],[0,0],[0,0],[-1,2.44],[0,0],[-6.73,3.93],[0.79,2.32],[-0.57,-1.22],[-36.49,7.48],[19.65,16.7],[12.16,6.55],[-24.32,-1.87],[-39.28,-3.74],[0,0],[1.87,9.36],[-11.22,2.8]],"o":[[-12.16,-44.9],[0,0],[-14.97,1.87],[-29,-2.8],[0,0],[-14.97,-1.87],[-28.07,8.41],[-32.74,53.32],[0,0],[0,0],[0,0],[0,0],[0.62,-2.05],[0,0],[0,0],[0.53,-1.18],[0,0],[0,0],[2.34,-4.58],[0,0],[7.01,-14.37],[0,0],[0.52,1.12],[13.1,28.07],[36.48,-7.48],[-19.64,-16.71],[-12.16,-6.55],[24.32,1.87],[39.3,3.74],[-27.13,8.42],[-1.88,-9.35],[11.23,-2.81]],"v":[[155.755,-95.155],[67.825,-126.025],[1.405,-45.575],[32.275,-121.355],[-70.625,-88.605],[-80.915,1.195],[-80.915,-80.185],[-135.175,-39.025],[-97.755,53.585],[-116.465,121.875],[-116.465,121.875],[-125.405,140.055],[-117.755,140.055],[-115.685,134.125],[-115.665,134.085],[-115.635,134.005],[-113.415,129.095],[-113.415,129.085],[-113.405,129.075],[-108.495,118.675],[-108.485,118.655],[-84.665,81.645],[-86.295,78.135],[-84.665,81.645],[30.405,106.905],[31.335,72.435],[-41.625,53.585],[-11.695,42.355],[64.075,72.295],[133.305,2.135],[55.665,-13.775],[137.045,-24.995]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.160784319043,0.701960802078,0.737254917622,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[764.819,817.631],"ix":2},"a":{"a":0,"k":[-144.643,121.429],"ix":1},"s":{"a":0,"k":[-100,100],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[-2]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":45,"s":[-7]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":104,"s":[-2]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":164,"s":[-7]},{"t":199,"s":[-2]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 17","np":2,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-24.431,75.331],[0,0],[-25.3,-4.431],[0,0],[0,0],[-25.61,1.932],[0,0],[0,0],[0,0],[-30.35,-63.672],[0,0],[-6.71,-19.844],[0,0],[1.14,3.301],[0,0],[0,0],[1.43,3.563],[0,0],[0,0],[1.5,4.233],[0,0],[11.12,7.237],[-1.54,3.876],[1.08,-2.032],[61.4,15.771],[-34.8,26.756],[-21.22,10.119],[41.51,-1.15],[67.11,-3.082],[0,0],[-3.96,15.75],[18.85,5.709]],"o":[[24.42,-75.331],[0,0],[25.29,4.431],[49.54,-2.35],[0,0],[25.6,-1.931],[47.02,16.658],[51.22,93.396],[0,0],[0,0],[0,0],[0,0],[-0.89,-3.541],[0,0],[0,0],[-0.79,-2.054],[0,0],[0,0],[-3.59,-7.976],[0,0],[-10.71,-25.008],[0,0],[-0.98,1.859],[-24.61,46.623],[-61.41,-15.771],[34.79,-26.755],[21.23,-10.118],[-41.51,1.15],[-67.12,3.081],[45.42,16.58],[3.96,-15.749],[-18.85,-5.709]],"v":[[-258.38,-174.038],[-106.3,-219.183],[-0.079,-76.85],[-46.239,-208.262],[125.98,-144],[135.98,9.551],[142.771,-128.826],[231.59,-54.312],[160.231,100.028],[186.341,217.7],[186.341,217.702],[200.031,249.369],[187.021,248.73],[183.98,238.459],[183.96,238.399],[183.91,238.256],[180.551,229.716],[180.551,229.703],[180.54,229.686],[173.051,211.598],[173.04,211.559],[135.62,146.651],[138.701,140.821],[135.62,146.651],[-62.12,179.988],[-60.829,121.293],[64.8,95.341],[14.84,73.755],[-116.49,118.326],[-228.33,-6.746],[-94.989,-27.302],[-232.429,-53.184]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.364705890417,0.909803986549,0.866666734219,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1129.909,779.904],"ix":2},"a":{"a":0,"k":[225,232.143],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[13]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":43,"s":[5]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":92,"s":[13]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":144,"s":[5]},{"t":199,"s":[13]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 18","np":2,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[16.94,0],[0,0],[-43.77,0],[-1.63,-21.67]],"o":[[-53.5,0],[0,0],[51.94,0],[0.73,9.6]],"v":[[46.76,70.935],[-120.69,34.595],[-36.86,31.075],[73.86,57.695]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[-46.77,0],[-3.84,-3.7]],"o":[[0,0],[8.15,0],[25.89,24.98]],"v":[[-97.42,26.005],[53.12,-26.005],[71.53,-20.725]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.113725498319,0.600000023842,0.674509823322,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[1571.15,766.592],"ix":2},"a":{"a":0,"k":[-109.821,33.036],"ix":1},"s":{"a":0,"k":[145,145],"ix":3},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":0,"s":[-49.641]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":52,"s":[-46.641]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":99,"s":[-49.641]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":153,"s":[-46.641]},{"t":199,"s":[-49.641]}],"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 20","np":3,"cix":2,"bm":0,"ix":6,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":200,"st":0,"bm":0},{"ddd":0,"ind":9,"ty":4,"nm":"Shape Layer 10","sr":1,"ks":{"o":{"a":0,"k":36,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[512,530,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[905,66],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"fl","c":{"a":0,"k":[0.04705881979,0.077508642159,0.176470588235,1],"ix":4},"o":{"a":0,"k":20,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[-3,249],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":200,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Animations/empty.json b/src/Artemis.UI/Assets/Animations/empty.json new file mode 100644 index 000000000..f79bf38aa --- /dev/null +++ b/src/Artemis.UI/Assets/Animations/empty.json @@ -0,0 +1 @@ +{"v":"4.7.0","fr":25,"ip":0,"op":50,"w":120,"h":120,"nm":"Comp 1","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"ruoi","ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.967]},"o":{"x":[0.167],"y":[0.033]},"n":["0p833_0p967_0p167_0p033"],"t":35,"s":[100],"e":[0]},{"t":49}]},"r":{"a":0,"k":0},"p":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0,"y":0},"n":"0p833_0p833_0_0","t":0,"s":[57.361,61.016,0],"e":[57.699,41.796,0],"to":[-4.67500305175781,-4.12800598144531,0],"ti":[-13.9099960327148,5.27300262451172,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"n":"0p833_0p833_0p167_0p167","t":10.219,"s":[57.699,41.796,0],"e":[79.084,33.982,0],"to":[12.8159942626953,-4.85800170898438,0],"ti":[-4.54498291015625,3.73400115966797,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"n":"0p833_0p833_0p167_0p167","t":19.445,"s":[79.084,33.982,0],"e":[59.691,9.121,0],"to":[6.61601257324219,-5.43799591064453,0],"ti":[20.0290069580078,1.20700073242188,0]},{"t":35}]},"a":{"a":0,"k":[60.531,10.945,0]},"s":{"a":0,"k":[100,100,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-0.994,0],[0,-0.994],[0.995,0],[0,0.994]],"o":[[0.995,0],[0,0.994],[-0.994,0],[0,-0.994]],"v":[[-0.001,-1.801],[1.801,-0.001],[-0.001,1.801],[-1.801,-0.001]],"c":true}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"fl","c":{"a":0,"k":[0.529,0.529,0.529,1]},"o":{"a":0,"k":100},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill"},{"ty":"tr","p":{"a":0,"k":[62.4,13.144],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group"},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-1.422,0],[0,-1.422],[1.421,0],[0,1.422]],"o":[[1.421,0],[0,1.422],[-1.422,0],[0,-1.422]],"v":[[0.001,-2.574],[2.574,0],[0.001,2.574],[-2.574,0]],"c":true}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"st","c":{"a":0,"k":[0.529,0.529,0.529,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":0.7},"lc":1,"lj":1,"ml":10,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke"},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill"},{"ty":"tr","p":{"a":0,"k":[64.145,9.606],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":3,"cix":2,"ix":2,"mn":"ADBE Vector Group"},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-1.996,0],[0,-1.996],[1.996,0],[0,1.996]],"o":[[1.996,0],[0,1.996],[-1.996,0],[0,-1.996]],"v":[[0,-3.614],[3.614,0],[0,3.614],[-3.614,0]],"c":true}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"st","c":{"a":0,"k":[0.529,0.529,0.529,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":0.7},"lc":1,"lj":1,"ml":10,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke"},{"ty":"fl","c":{"a":0,"k":[1,1,1,1]},"o":{"a":0,"k":100},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill"},{"ty":"tr","p":{"a":0,"k":[57.957,10.552],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":3,"cix":2,"ix":3,"mn":"ADBE Vector Group"},{"ty":"tr","p":{"a":0,"k":[60.531,10.941],"ix":2},"a":{"a":0,"k":[60.531,10.941],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"ruoi","np":3,"cix":2,"ix":1,"mn":"ADBE Vector Group"}],"ip":0,"op":50,"st":0,"bm":0,"sr":1},{"ddd":0,"ind":2,"ty":4,"nm":"Shape Layer 2","ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.967]},"o":{"x":[0.167],"y":[0.033]},"n":["0p833_0p967_0p167_0p033"],"t":35,"s":[100],"e":[0]},{"t":49}]},"r":{"a":0,"k":0},"p":{"a":0,"k":[-0.75,-0.75,0]},"a":{"a":0,"k":[0,0,0]},"s":{"a":0,"k":[100,100,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-13.91,5.273],[-4.545,3.734],[20.029,1.207]],"o":[[-4.675,-4.128],[12.816,-4.858],[6.616,-5.438],[0,0]],"v":[[-7.383,24.76],[-7.046,5.54],[14.34,-2.273],[-3.178,-24.76]],"c":false}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"st","c":{"a":0,"k":[0.627,0.627,0.627,1]},"o":{"a":0,"k":100},"w":{"a":0,"k":1},"lc":2,"lj":2,"d":[{"n":"d","nm":"dash","v":{"a":0,"k":2.028}},{"n":"g","nm":"gap","v":{"a":0,"k":2.028}},{"n":"o","nm":"offset","v":{"a":0,"k":0}}],"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke"},{"ty":"tr","p":{"a":0,"k":[67.87,37.631],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group"},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.953]},"o":{"x":[0.167],"y":[0.033]},"n":["0p833_0p953_0p167_0p033"],"t":0,"s":[0],"e":[100]},{"t":35}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim"}],"ip":0,"op":50,"st":0,"bm":0,"sr":1},{"ddd":0,"ind":3,"ty":4,"nm":"im_emptyBox Outlines","ks":{"o":{"a":0,"k":100},"r":{"a":0,"k":0},"p":{"a":0,"k":[60,60,0]},"a":{"a":0,"k":[60,60,0]},"s":{"a":0,"k":[100,100,100]}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-0.001,-16.607],[-32.143,-0.002],[-0.001,16.607],[32.144,-0.002]],"c":true}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"fl","c":{"a":0,"k":[0.8,0.82,0.851,1]},"o":{"a":0,"k":100},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill"},{"ty":"tr","p":{"a":0,"k":[60,55.75],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 7","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group"},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[12.856,-23.249],[0,-16.605],[-12.857,-23.249],[-45,-6.641],[-32.144,0.001],[-45,6.645],[-12.857,23.249],[0,16.609],[12.856,23.249],[45,6.645],[32.143,0.001],[45,-6.641]],"c":true}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"fl","c":{"a":0,"k":[0.957,0.957,0.957,1]},"o":{"a":0,"k":100},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill"},{"ty":"tr","p":{"a":0,"k":[60,55.748],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 8","np":2,"cix":2,"ix":2,"mn":"ADBE Vector Group"},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-16.072,24.171],[16.072,11.312],[16.072,-24.171],[-16.072,-24.171]],"c":true}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"fl","c":{"a":0,"k":[0.902,0.914,0.929,1]},"o":{"a":0,"k":100},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill"},{"ty":"tr","p":{"a":0,"k":[76.072,83.33],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 9","np":2,"cix":2,"ix":3,"mn":"ADBE Vector Group"},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[-32.143,-24.171],[-32.143,11.311],[-0.001,24.171],[32.144,11.311],[32.144,-24.171]],"c":true}},"nm":"Path 1","mn":"ADBE Vector Shape - Group"},{"ty":"fl","c":{"a":0,"k":[0.8,0.82,0.851,1]},"o":{"a":0,"k":100},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill"},{"ty":"tr","p":{"a":0,"k":[60,83.33],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 10","np":2,"cix":2,"ix":4,"mn":"ADBE Vector Group"},{"ty":"tr","p":{"a":0,"k":[60,60.186],"ix":2},"a":{"a":0,"k":[60,60.186],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"box","np":4,"cix":2,"ix":1,"mn":"ADBE Vector Group"}],"ip":0,"op":50,"st":0,"bm":0,"sr":1}]} \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Animations/login-pending.json b/src/Artemis.UI/Assets/Animations/login-pending.json new file mode 100644 index 000000000..7e5d7187a --- /dev/null +++ b/src/Artemis.UI/Assets/Animations/login-pending.json @@ -0,0 +1 @@ +{"v":"5.8.1","fr":30,"ip":0,"op":101,"w":500,"h":500,"nm":"Login_and_Sign_up ","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Group 1 :M","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[218.999,313.27,0],"ix":2,"l":2},"a":{"a":0,"k":[218.999,313.27,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.567,-0.343],[0.567,-0.3],[3.803,-0.729],[3.803,0.729],[0.567,0.343],[0.567,0.386],[2.624,2.956],[1.274,3.728],[-0.011,0.729],[-0.054,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.803,0.729],[-3.803,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.317,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[250.124,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[250.124,313.27],"ix":2},"a":{"a":0,"k":[250.124,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":63,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":64,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":84,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":364,"s":[100]},{"t":365,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":1,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.568,-0.343],[0.568,-0.3],[3.802,-0.729],[3.802,0.729],[0.568,0.343],[0.568,0.386],[2.625,2.956],[1.275,3.728],[-0.01,0.729],[-0.053,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.802,0.729],[-3.802,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.318,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[241.231,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[241.231,313.27],"ix":2},"a":{"a":0,"k":[241.231,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":61,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":62,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":82,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":362,"s":[100]},{"t":363,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":1,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.568,-0.343],[0.568,-0.3],[3.803,-0.729],[3.803,0.729],[0.568,0.343],[0.568,0.386],[2.625,2.956],[1.275,3.728],[-0.011,0.729],[-0.054,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.803,0.729],[-3.803,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.317,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[232.338,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[232.338,313.27],"ix":2},"a":{"a":0,"k":[232.338,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":59,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":60,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":80,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":360,"s":[100]},{"t":361,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":1,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.568,-0.343],[0.568,-0.3],[3.803,-0.729],[3.803,0.729],[0.568,0.343],[0.568,0.386],[2.625,2.956],[1.275,3.728],[-0.011,0.729],[-0.054,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.803,0.729],[-3.803,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.317,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[223.445,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[223.445,313.27],"ix":2},"a":{"a":0,"k":[223.445,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":57,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":58,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":78,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":358,"s":[100]},{"t":359,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":1,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.568,-0.343],[0.568,-0.3],[3.803,-0.729],[3.803,0.729],[0.568,0.343],[0.568,0.386],[2.625,2.956],[1.275,3.728],[-0.011,0.729],[-0.054,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.803,0.729],[-3.803,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.317,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[214.553,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[214.553,313.27],"ix":2},"a":{"a":0,"k":[214.553,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":55,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":56,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":76,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":356,"s":[100]},{"t":357,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":1,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.568,-0.343],[0.568,-0.3],[3.803,-0.729],[3.803,0.729],[0.568,0.343],[0.568,0.386],[2.625,2.956],[1.275,3.728],[-0.011,0.729],[-0.054,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.803,0.729],[-3.803,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.317,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[205.66,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[205.66,313.27],"ix":2},"a":{"a":0,"k":[205.66,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":53,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":54,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":74,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":354,"s":[100]},{"t":355,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":1,"cix":2,"bm":0,"ix":6,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.568,-0.343],[0.568,-0.3],[3.803,-0.729],[3.803,0.729],[0.568,0.343],[0.568,0.386],[2.625,2.956],[1.275,3.728],[-0.011,0.729],[-0.054,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.803,0.729],[-3.803,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.317,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[196.767,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 7","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[196.767,313.27],"ix":2},"a":{"a":0,"k":[196.767,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":51,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":52,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":72,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":352,"s":[100]},{"t":353,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 7","np":1,"cix":2,"bm":0,"ix":7,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[2.603,-3.021],[0.568,-0.343],[0.568,-0.3],[3.803,-0.729],[3.803,0.729],[0.568,0.343],[0.568,0.386],[2.625,2.956],[1.275,3.728],[-0.011,0.729],[-0.054,0.729],[-1.446,3.749],[-2.667,2.978],[-0.632,0.364],[-0.632,0.321],[-3.803,0.729],[-3.803,-0.729],[-0.653,-0.321],[-0.653,-0.364],[-2.667,-2.978],[-1.36,-3.728],[-0.032,-0.75],[0.011,-0.75],[1.317,-3.749]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[187.874,313.27],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 8","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[187.874,313.27],"ix":2},"a":{"a":0,"k":[187.874,313.27],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":49,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":50,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":70,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":350,"s":[100]},{"t":351,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 8","np":1,"cix":2,"bm":0,"ix":8,"mn":"ADBE Vector Group","hd":false}],"ip":50,"op":364,"st":50,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"User name","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":20,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[183.429,270.817,0],"ix":2,"l":2},"a":{"a":0,"k":[-55.929,0,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0,0,0],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":30,"s":[0,100,100]},{"t":60,"s":[100,100,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[111.857,9.6],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254901961,0.666666666667,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"User name","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":30,"op":300,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Login & Sign up Button","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":12,"s":[0]},{"t":29,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":9,"s":[250,461.026,0],"to":[0,-16.667,0],"ti":[0,16.667,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.333,"y":0.333},"t":29,"s":[250,361.026,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":69,"s":[250,361.026,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":74,"s":[250,366.026,0],"to":[0,0,0],"ti":[0,0,0]},{"t":79,"s":[250,361.026,0]}],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":1,"k":[{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":69,"s":[100,100,100]},{"i":{"x":[0.667,0.667,0.667],"y":[1,1,1]},"o":{"x":[0.333,0.333,0.333],"y":[0,0,0]},"t":74,"s":[95,95,100]},{"t":79,"s":[100,100,100]}],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.633,0],[0,0],[0,7.633],[0,0],[-7.633,0],[0,0],[0,-7.633],[0,0]],"o":[[0,0],[-7.633,0],[0,0],[0,-7.633],[0,0],[7.633,0],[0,0],[0,7.633]],"v":[[34.696,13.878],[-34.696,13.878],[-48.574,0],[-48.574,0],[-34.696,-13.878],[34.696,-13.878],[48.574,0],[48.574,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254911661,0.666666686535,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Login & Sign up Button","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":9,"op":309,"st":9,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Password Box","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":7,"s":[0]},{"t":27,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":7,"s":[250,412.452,0],"to":[0,-16.667,0],"ti":[0,16.667,0]},{"t":27,"s":[250,312.452,0]}],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[-76.33,-13.878],[76.33,-13.878],[76.33,13.878],[-76.33,13.878]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Password Box","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":7,"op":307,"st":7,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"User name Box","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":5,"s":[0]},{"t":25,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":5,"s":[250,370.817,0],"to":[0,-16.667,0],"ti":[0,16.667,0]},{"t":25,"s":[250,270.817,0]}],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[152.661,27.757],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"User name Box","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":5,"op":305,"st":5,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Use icon","parent":8,"sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":20,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[0,-55.513,0],"ix":2,"l":2},"a":{"a":0,"k":[250,194.487,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-10.409,0],[-7.633,6.939],[13.184,0],[2.776,-11.103]],"o":[[10.409,0],[-2.776,-11.103],[-13.184,0],[6.939,6.939]],"v":[[0,12.837],[27.063,2.429],[0,-12.837],[-27.063,2.429]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.015686275437,0.470588237047,0.929411768913,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[250,223.284],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-9.198,0],[0,-9.198],[9.198,0],[0,9.198]],"o":[[9.198,0],[0,9.198],[-9.198,0],[0,-9.198]],"v":[[0,-16.654],[16.654,0],[0,16.654],[-16.654,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[250,188.936],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[22.899,0],[0,-22.899],[-22.899,0],[0,22.899]],"o":[[-22.899,0],[0,22.899],[22.899,0],[0,-23.593]],"v":[[0,-41.635],[-41.635,0],[0,41.635],[41.635,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.337254911661,0.666666686535,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[250,194.487],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":300,"st":0,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"Top Button","parent":8,"sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":20,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-97.148,-124.904,0],"ix":2,"l":2},"a":{"a":0,"k":[152.852,125.096,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[13.878,13.878],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[125.096,125.096],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[13.878,13.878],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[152.852,125.096],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[13.878,13.878],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[180.609,125.096],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":300,"st":0,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"Tab","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":20,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[250,450,0],"to":[0,-33.333,0],"ti":[0,33.333,0]},{"t":20,"s":[250,250,0]}],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.633,0],[0,0],[0,-7.633],[0,0],[-7.633,0],[0,0],[0,7.633],[0,0]],"o":[[0,0],[-7.633,0],[0,0],[0,7.633],[0,0],[7.633,0],[0,0],[0,-7.633]],"v":[[145.722,-159.6],[-145.722,-159.6],[-159.6,-145.722],[-159.6,145.722],[-145.722,159.6],[145.722,159.6],[159.6,145.722],[159.6,-145.722]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.850980401039,0.86274510622,0.882352948189,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Tab","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":300,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Animations/password.json b/src/Artemis.UI/Assets/Animations/password.json new file mode 100644 index 000000000..33d3a9b4a --- /dev/null +++ b/src/Artemis.UI/Assets/Animations/password.json @@ -0,0 +1 @@ +{"v":"5.7.8","fr":30,"ip":0,"op":301,"w":300,"h":300,"nm":"free 04","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":3,"nm":"Null 9","sr":1,"ks":{"o":{"a":0,"k":0,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[150,170,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[191,191,100],"ix":6,"l":2}},"ao":0,"ip":0,"op":301,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Shapes on screen 2","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-26,94.218,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0.762],[0.762,0],[0,-0.762],[-0.762,0]],"o":[[0,-0.762],[-0.762,0],[0,0.762],[0.762,0]],"v":[[40.188,-146.941],[38.809,-148.32],[37.429,-146.941],[38.809,-145.561]],"c":true},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0.762],[-0.762,0],[0,-0.762],[0.762,0]],"o":[[0,-0.762],[0.762,0],[0,0.762],[-0.762,0]],"v":[[15.077,-146.941],[16.457,-148.32],[17.836,-146.941],[16.457,-145.561]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.807843148708,0.827450990677,0.972549021244,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[11.008,-159.04],[6.02,-164.027]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[9.345,-152.39],[2.695,-152.39]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":103,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":107,"s":[0]},{"t":113,"s":[100],"h":1},{"t":130,"s":[0],"h":1},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":250,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":254,"s":[0]},{"t":260,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":103,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":107,"s":[100]},{"t":113,"s":[100],"h":1},{"t":130,"s":[0],"h":1},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":250,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":254,"s":[100]},{"t":260,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":3,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.988235294819,0.729411780834,0.023529412225,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":2,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":1,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[27.633,-136.939],[27.633,-133.004]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.258823543787,0.243137255311,0.686274528503,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":2,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,-1.894],[1.894,0],[0,1.894],[-1.894,0]],"o":[[0,1.894],[-1.894,0],[0,-1.894],[1.894,0]],"v":[[31.062,-139.433],[27.633,-136.003],[24.203,-139.433],[27.633,-142.862]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.258823543787,0.243137255311,0.686274528503,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":1,"y":0},"t":94,"s":[{"i":[[0,0],[0,4.507],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.482,-150.336],[19.453,-155.153],[27.614,-163.315],[35.776,-155.153],[35.776,-150.501]],"c":false}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":113,"s":[{"i":[[0,0],[0,0.341],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.5,-155.992],[19.471,-159.403],[27.633,-167.565],[35.794,-159.403],[35.794,-150.727]],"c":false}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":147,"s":[{"i":[[0,0],[0,0.341],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.5,-155.992],[19.471,-159.403],[27.633,-167.565],[35.794,-159.403],[35.794,-150.727]],"c":false}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":151,"s":[{"i":[[0,0],[0,4.507],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.482,-150.336],[19.453,-155.153],[27.614,-163.315],[35.776,-155.153],[35.776,-150.501]],"c":false}]},{"i":{"x":0,"y":1},"o":{"x":1,"y":0},"t":241,"s":[{"i":[[0,0],[0,4.507],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.482,-150.336],[19.453,-155.153],[27.614,-163.315],[35.776,-155.153],[35.776,-150.501]],"c":false}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":260,"s":[{"i":[[0,0],[0,0.341],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.5,-155.992],[19.471,-159.403],[27.633,-167.565],[35.794,-159.403],[35.794,-150.727]],"c":false}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":296,"s":[{"i":[[0,0],[0,0.341],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.5,-155.992],[19.471,-159.403],[27.633,-167.565],[35.794,-159.403],[35.794,-150.727]],"c":false}]},{"t":300,"s":[{"i":[[0,0],[0,4.507],[-4.507,0],[0,-4.507],[0,0]],"o":[[0,0],[0,-4.507],[4.507,0],[0,0],[0,0]],"v":[[19.482,-150.336],[19.453,-155.153],[27.614,-163.315],[35.776,-155.153],[35.776,-150.501]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.258823543787,0.243137255311,0.686274528503,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":3,"ix":5},"lc":1,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 3","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[8.264,0],[0,8.264],[0,0],[0,0]],"o":[[0,8.264],[-8.264,0],[0,0],[0,0],[0,0]],"v":[[42.595,-136.003],[27.633,-121.041],[12.67,-136.003],[12.67,-150.727],[42.595,-150.727]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.501960813999,0.490196079016,0.984313726425,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 4","np":2,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":4,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 7","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":301,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Shapes on screen","parent":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[-26,-105.782,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-2.521,99.321],[6.883,104.75]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[6.883,99.321],[-2.521,104.75]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[2.181,107.465],[2.181,96.606]],"c":false},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.988235294819,0.729411780834,0.023529412225,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":2,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":4,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":9,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":18,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":50,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":55,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":64,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":104,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":109,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":143,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":145,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":154,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":159,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":168,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":200,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":205,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":214,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":254,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":259,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":293,"s":[100]},{"t":295,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":4,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-2.521,99.321],[6.883,104.75]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[6.883,99.321],[-2.521,104.75]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[2.181,107.465],[2.181,96.606]],"c":false},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.988235294819,0.729411780834,0.023529412225,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":2,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[17,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":14,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":19,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":28,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":60,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":65,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":74,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":106,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":111,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":143,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":145,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":164,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":169,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":178,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":210,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":215,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":224,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":256,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":261,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":293,"s":[100]},{"t":295,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 7","np":4,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-2.521,99.321],[6.883,104.75]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[6.883,99.321],[-2.521,104.75]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[2.181,107.465],[2.181,96.606]],"c":false},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.988235294819,0.729411780834,0.023529412225,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":2,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[33.5,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":24,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":29,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":38,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":70,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":75,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":84,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":109,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":114,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":143,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":145,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":174,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":179,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":188,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":220,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":225,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":234,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":259,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":264,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":293,"s":[100]},{"t":295,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 8","np":4,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-2.521,99.321],[6.883,104.75]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[6.883,99.321],[-2.521,104.75]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[2.181,107.465],[2.181,96.606]],"c":false},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.988235294819,0.729411780834,0.023529412225,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":2,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[50.5,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":36,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":41,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":50,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":82,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":87,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":96,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":112,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":117,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":143,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":145,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":186,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":191,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":200,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":232,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":237,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":246,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":262,"s":[0]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":267,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":293,"s":[100]},{"t":295,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":4,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-2.521,99.321],[6.883,104.75]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[6.883,99.321],[-2.521,104.75]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[2.181,107.465],[2.181,96.606]],"c":false},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ind":3,"ty":"sh","ix":4,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[14.479,99.321],[23.883,104.75]],"c":false},"ix":2},"nm":"Path 4","mn":"ADBE Vector Shape - Group","hd":false},{"ind":4,"ty":"sh","ix":5,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[23.883,99.321],[14.479,104.75]],"c":false},"ix":2},"nm":"Path 5","mn":"ADBE Vector Shape - Group","hd":false},{"ind":5,"ty":"sh","ix":6,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[19.181,107.465],[19.181,96.606]],"c":false},"ix":2},"nm":"Path 6","mn":"ADBE Vector Shape - Group","hd":false},{"ind":6,"ty":"sh","ix":7,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[30.979,99.321],[40.383,104.75]],"c":false},"ix":2},"nm":"Path 7","mn":"ADBE Vector Shape - Group","hd":false},{"ind":7,"ty":"sh","ix":8,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[40.383,99.321],[30.979,104.75]],"c":false},"ix":2},"nm":"Path 8","mn":"ADBE Vector Shape - Group","hd":false},{"ind":8,"ty":"sh","ix":9,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[35.681,107.465],[35.681,96.606]],"c":false},"ix":2},"nm":"Path 9","mn":"ADBE Vector Shape - Group","hd":false},{"ind":9,"ty":"sh","ix":10,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[47.979,99.321],[57.383,104.75]],"c":false},"ix":2},"nm":"Path 10","mn":"ADBE Vector Shape - Group","hd":false},{"ind":10,"ty":"sh","ix":11,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[57.383,99.321],[47.979,104.75]],"c":false},"ix":2},"nm":"Path 11","mn":"ADBE Vector Shape - Group","hd":false},{"ind":11,"ty":"sh","ix":12,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[52.681,107.465],[52.681,96.606]],"c":false},"ix":2},"nm":"Path 12","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.827450990677,0.827450990677,0.827450990677,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":2,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 12","np":13,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":5,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[1.587,0]],"v":[[-19.753,127.51],[74.55,127.51]],"c":false},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[0.412,0.115]],"v":[[-19.716,141.635],[27.75,141.519]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[1.587,0]],"v":[[46.534,141.519],[74.588,141.635]],"c":false},"ix":2},"nm":"Path 4","mn":"ADBE Vector Shape - Group","hd":false},{"ind":3,"ty":"sh","ix":4,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[1.587,0]],"v":[[-19.716,155.635],[74.588,155.635]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":86,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":148,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":149,"s":[0]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.167],"y":[0.167]},"t":235,"s":[0]},{"t":297,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":86,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":148,"s":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":149,"s":[0]},{"i":{"x":[0],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":235,"s":[100]},{"t":297,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":2,"ix":5,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.2901960784313726,0.39215686274509803,0.8,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":6.5,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 6","np":6,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[1.587,0]],"v":[[-19.753,127.51],[74.55,127.51]],"c":false},"ix":2},"nm":"Path 3","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[0.412,0.115]],"v":[[-19.716,141.635],[27.75,141.519]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[1.587,0]],"v":[[46.534,141.519],[74.588,141.635]],"c":false},"ix":2},"nm":"Path 4","mn":"ADBE Vector Shape - Group","hd":false},{"ind":3,"ty":"sh","ix":4,"ks":{"a":0,"k":{"i":[[-1.587,0],[0,0]],"o":[[0,0],[1.587,0]],"v":[[-19.716,155.635],[74.588,155.635]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.831372618675,0.831372618675,0.831372618675,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":6.5,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 5","np":5,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 2","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":301,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Animations/success.json b/src/Artemis.UI/Assets/Animations/success.json new file mode 100644 index 000000000..f274c145e --- /dev/null +++ b/src/Artemis.UI/Assets/Animations/success.json @@ -0,0 +1 @@ +{"v":"5.0.1","fr":29.9700012207031,"ip":0,"op":45.0000018328876,"w":512,"h":512,"nm":"Comp 1","ddd":0,"assets":[{"id":"comp_0","layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.572],"y":[0.556]},"o":{"x":[0.167],"y":[0.167]},"n":["0p572_0p556_0p167_0p167"],"t":7,"s":[100],"e":[92.154]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.641],"y":[0.056]},"n":["0p833_0p833_0p641_0p056"],"t":13,"s":[92.154],"e":[30]},{"t":17.0000006924242}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[256,256,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-230,4],[214,4]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.23137254902,0.741176470588,0.36862745098,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":70,"ix":5},"lc":2,"lj":1,"ml":4,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.588],"y":[-51709.363]},"o":{"x":[0.167],"y":[0.167]},"n":["0p588_-51709p363_0p167_0p167"],"t":7,"s":[0],"e":[0]},{"i":{"x":[0.696],"y":[0.999]},"o":{"x":[0.509],"y":[0.003]},"n":["0p696_0p999_0p509_0p003"],"t":10,"s":[0],"e":[100]},{"t":16.0000006516934}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.566],"y":[0.999]},"o":{"x":[0.457],"y":[0.063]},"n":["0p566_0p999_0p457_0p063"],"t":7,"s":[0],"e":[100]},{"t":16.0000006516934}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":150.000006109625,"st":0,"bm":0}]},{"id":"comp_1","layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[256,256,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0,0],"y":[0.997,0.997]},"o":{"x":[0.167,0.167],"y":[0.167,0.167]},"n":["0_0p997_0p167_0p167","0_0p997_0p167_0p167"],"t":24,"s":[40,40],"e":[90,90]},{"i":{"x":[0.833,0.833],"y":[0.833,0.833]},"o":{"x":[0.574,0.574],"y":[-0.004,-0.004]},"n":["0p833_0p833_0p574_-0p004","0p833_0p833_0p574_-0p004"],"t":27,"s":[90,90],"e":[18.394,18.394]},{"t":38.0000015477717}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"n":"0p667_1_0p167_0p167","t":24,"s":[-181.074,-5.414],"e":[200,-5.414],"to":[34.0465698242188,0],"ti":[-26.72825050354,0]},{"t":38.0000015477717}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"n":["0p833_0p833_0p167_0p167"],"t":24,"s":[0],"e":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"n":["0p833_0p833_0p167_0p167"],"t":25,"s":[100],"e":[100]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"n":["0p833_0p833_0p167_0p167"],"t":29,"s":[100],"e":[0]},{"t":38.0000015477717}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":150.000006109625,"st":0,"bm":0}]},{"id":"comp_2","layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.572],"y":[0.556]},"o":{"x":[0.167],"y":[0.167]},"n":["0p572_0p556_0p167_0p167"],"t":10,"s":[100],"e":[92.154]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.641],"y":[0.056]},"n":["0p833_0p833_0p641_0p056"],"t":16,"s":[92.154],"e":[30]},{"t":20.0000008146167}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[256,256,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-230,4],[214,4]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.23137254902,0.741176470588,0.36862745098,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":70,"ix":5},"lc":2,"lj":1,"ml":4,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.588],"y":[-51709.363]},"o":{"x":[0.167],"y":[0.167]},"n":["0p588_-51709p363_0p167_0p167"],"t":10,"s":[0],"e":[0]},{"i":{"x":[0.696],"y":[0.999]},"o":{"x":[0.509],"y":[0.003]},"n":["0p696_0p999_0p509_0p003"],"t":13,"s":[0],"e":[100]},{"t":19.0000007738859}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.566],"y":[0.999]},"o":{"x":[0.457],"y":[0.063]},"n":["0p566_0p999_0p457_0p063"],"t":10,"s":[0],"e":[100]},{"t":19.0000007738859}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":150.000006109625,"st":0,"bm":0}]}],"layers":[{"ddd":0,"ind":1,"ty":0,"nm":"trait","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":90,"ix":10},"p":{"a":0,"k":[263.334,471.109,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[15,15,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":0,"nm":"trait","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-180,"ix":10},"p":{"a":0,"k":[51.641,253.275,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[15,15,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":0,"nm":"trait","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-90,"ix":10},"p":{"a":0,"k":[266.322,44.315,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[15,15,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":0,"nm":"trait","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[469.91,258.792,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[15,15,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":5,"ty":0,"nm":"firefly","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-18.097,"ix":10},"p":{"a":0,"k":[400.635,189.708,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[20,20,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":6,"ty":0,"nm":"firefly","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-72.471,"ix":10},"p":{"a":0,"k":[359.413,150.912,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[20,20,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":7,"ty":0,"nm":"firefly","refId":"comp_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-45.707,"ix":10},"p":{"a":0,"k":[396.894,150.961,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[30,30,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":8,"ty":0,"nm":"trait 2","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-135.205,"ix":10},"p":{"a":0,"k":[410.865,406.53,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[-19.512,19.512,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":9,"ty":0,"nm":"trait 2","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-45.606,"ix":10},"p":{"a":0,"k":[105.535,402.598,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[-19.512,19.512,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":10,"ty":0,"nm":"trait 2","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-135.205,"ix":10},"p":{"a":0,"k":[104.864,111.71,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[19.512,19.512,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":11,"ty":0,"nm":"trait 2","refId":"comp_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":-45.606,"ix":10},"p":{"a":0,"k":[416.722,113.206,0],"ix":2},"a":{"a":0,"k":[256,256,0],"ix":1},"s":{"a":0,"k":[19.512,19.512,100],"ix":6}},"ao":0,"w":512,"h":512,"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":12,"ty":4,"nm":"Shape Layer 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[236.888,240.258,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[69.59,69.59,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-76.426,37.999],[12.056,114.074],[169.991,-68.635]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":35,"ix":5},"lc":2,"lj":1,"ml":4,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-7,11],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[-2.986]},"o":{"x":[0.167],"y":[0]},"n":["0p833_-2p986_0p167_0"],"t":0,"s":[0],"e":[0]},{"i":{"x":[0],"y":[0.973]},"o":{"x":[0.167],"y":[0.042]},"n":["0_0p973_0p167_0p042"],"t":14.791,"s":[0],"e":[32]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.828],"y":[0.011]},"n":["0p833_0p833_0p828_0p011"],"t":19.791,"s":[32],"e":[100]},{"t":24.7912510097683}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":150.000006109625,"st":0,"bm":0},{"ddd":0,"ind":13,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[256,256,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[80,80,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":1,"k":[{"i":{"x":[0.976,0.976],"y":[0.968,0.968]},"o":{"x":[0.654,0.654],"y":[0.007,0.007]},"n":["0p976_0p968_0p654_0p007","0p976_0p968_0p654_0p007"],"t":0,"s":[0,0],"e":[401.025,401.025]},{"i":{"x":[0.468,0.468],"y":[1.057,1.057]},"o":{"x":[0.346,0.346],"y":[-4.83,-4.83]},"n":["0p468_1p057_0p346_-4p83","0p468_1p057_0p346_-4p83"],"t":7,"s":[401.025,401.025],"e":[372.7,372.7]},{"i":{"x":[0.375,0.375],"y":[1.543,1.543]},"o":{"x":[0.364,0.364],"y":[0.031,0.031]},"n":["0p375_1p543_0p364_0p031","0p375_1p543_0p364_0p031"],"t":12,"s":[372.7,372.7],"e":[401.025,401.025]},{"i":{"x":[0.833,0.833],"y":[1,1]},"o":{"x":[0.327,0.327],"y":[-8.038,-8.038]},"n":["0p833_1_0p327_-8p038","0p833_1_0p327_-8p038"],"t":16,"s":[401.025,401.025],"e":[401.025,401.025]},{"t":20.0000008146167}],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"fl","c":{"a":0,"k":[0.229886716955,0.739552696078,0.369435897528,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[5.992,3.49],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":2,"cix":2,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":150.000006109625,"st":0,"bm":0}]} \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Animations/workshop-wizard-welcome.json b/src/Artemis.UI/Assets/Animations/workshop-wizard-welcome.json new file mode 100644 index 000000000..02a22bcc3 --- /dev/null +++ b/src/Artemis.UI/Assets/Animations/workshop-wizard-welcome.json @@ -0,0 +1 @@ +{"v": "5.5.7", "meta": {"g": "LottieFiles AE 0.1.20", "a": "", "k": "", "d": "", "tc": ""}, "fr": 24, "ip": 0, "op": 36, "w": 1600, "h": 1200, "nm": "graphic", "ddd": 0, "assets": [{"id": "image_0", "w": 163, "h": 319, "u": "", "p": "", "e": 1}, {"id": "image_1", "w": 372, "h": 216, "u": "", "p": "", "e": 1}, {"id": "image_2", "w": 149, "h": 87, "u": "", "p": "", "e": 1}, {"id": "image_3", "w": 222, "h": 159, "u": "", "p": "", "e": 1}, {"id": "image_4", "w": 245, "h": 74, "u": "", "p": "", "e": 1}, {"id": "image_5", "w": 586, "h": 145, "u": "", "p": "", "e": 1}, {"id": "image_6", "w": 444, "h": 288, "u": "", "p": "", "e": 1}, {"id": "image_7", "w": 706, "h": 410, "u": "", "p": "", "e": 1}, {"id": "image_8", "w": 771, "h": 376, "u": "", "p": "", "e": 1}], "layers": [{"ddd": 0, "ind": 1, "ty": 2, "nm": "pen 2", "refId": "image_0", "sr": 1, "ks": {"o": {"a": 0, "k": 100, "ix": 11}, "r": {"a": 1, "k": [{"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 0, "s": [0]}, {"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 5, "s": [-30]}, {"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 10, "s": [-30]}, {"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 14, "s": [-15]}, {"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 18, "s": [-30]}, {"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 22, "s": [-10]}, {"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 26, "s": [-30]}, {"t": 30, "s": [0]}], "ix": 10}, "p": {"a": 1, "k": [{"i": {"x": 0.242, "y": 1}, "o": {"x": 0.333, "y": 0}, "t": 0, "s": [938.408, 323.652, 0], "to": [-84.752, -8.713, 0], "ti": [84.752, 8.713, 0]}, {"i": {"x": 0.667, "y": 0.667}, "o": {"x": 0.333, "y": 0.333}, "t": 5, "s": [429.893, 271.375, 0], "to": [0, 0, 0], "ti": [0, 0, 0]}, {"i": {"x": 0.667, "y": 0.569}, "o": {"x": 0.333, "y": 0}, "t": 10, "s": [429.893, 271.375, 0], "to": [21.151, 13.176, 0], "ti": [-44.46, -22.024, 0]}, {"i": {"x": 0.667, "y": 0.864}, "o": {"x": 0.333, "y": 0.237}, "t": 14, "s": [523.553, 301.389, 0], "to": [29.096, 14.413, 0], "ti": [-45.818, -50.49, 0]}, {"i": {"x": 0.667, "y": 0.893}, "o": {"x": 0.333, "y": 0.072}, "t": 18, "s": [663.591, 412.357, 0], "to": [1.634, 1.801, 0], "ti": [-48.996, 15.884, 0]}, {"i": {"x": 0.667, "y": 0.885}, "o": {"x": 0.333, "y": 0.189}, "t": 22, "s": [1002.192, 426.09, 0], "to": [49.726, -16.12, 0], "ti": [35.512, 24.795, 0]}, {"i": {"x": 0.667, "y": 1}, "o": {"x": 0.333, "y": 0.317}, "t": 26, "s": [946.246, 264.763, 0], "to": [-19.812, -13.833, 0], "ti": [-2.406, 1.06, 0]}, {"t": 30, "s": [938.408, 323.652, 0]}], "ix": 2}, "a": {"a": 0, "k": [81.222, 159.292, 0], "ix": 1}, "s": {"a": 0, "k": [100, 100, 100], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 3, "ty": 4, "nm": "line", "sr": 1, "ks": {"o": {"a": 0, "k": 100, "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 0, "k": [793.313, 548.688, 0], "ix": 2}, "a": {"a": 0, "k": [353.419, 144.6, 0], "ix": 1}, "s": {"a": 0, "k": [100, 100, 100], "ix": 6}}, "ao": 0, "shapes": [{"ty": "gr", "it": [{"ind": 0, "ty": "sh", "ix": 1, "ks": {"a": 0, "k": {"i": [[0, 0], [-116.849, -66.994], [-14.03, -0.479], [-4.382, 1.989], [-95.037, 54.53], [0.648, 5.836], [2.663, 2.088], [82.608, 45.053], [12.674, -7.604], [4.048, -15.548], [28.044, 0]], "o": [[116.849, 66.994], [4.046, 2.796], [10.696, 0.365], [95.037, -54.53], [2.35, -3.116], [-0.798, -7.183], [-11.764, -9.226], [-42.065, -9.348], [-23.37, 14.022], [-3.751, 14.41], [0, 0]], "v": [[-346.198, -77.303], [4.35, 123.678], [32.394, 133.026], [55.764, 128.352], [340.876, -35.238], [345.55, -49.26], [336.202, -63.281], [233.374, -124.043], [191.309, -91.325], [121.199, -91.325], [65.112, -58.607]], "c": false}, "ix": 2}, "nm": "Path 1", "mn": "ADBE Vector Shape - Group", "hd": false}, {"ty": "st", "c": {"a": 0, "k": [1, 1, 1, 1], "ix": 3}, "o": {"a": 0, "k": 100, "ix": 4}, "w": {"a": 0, "k": 3, "ix": 5}, "lc": 2, "lj": 2, "bm": 0, "nm": "Stroke 1", "mn": "ADBE Vector Graphic - Stroke", "hd": false}, {"ty": "tr", "p": {"a": 0, "k": [353.698, 140.891], "ix": 2}, "a": {"a": 0, "k": [0, 0], "ix": 1}, "s": {"a": 0, "k": [100, 100], "ix": 3}, "r": {"a": 0, "k": 0, "ix": 6}, "o": {"a": 0, "k": 100, "ix": 7}, "sk": {"a": 0, "k": 0, "ix": 4}, "sa": {"a": 0, "k": 0, "ix": 5}, "nm": "Transform"}], "nm": "Group 1", "np": 2, "cix": 2, "bm": 0, "ix": 1, "mn": "ADBE Vector Group", "hd": false}, {"ty": "tm", "s": {"a": 0, "k": 0, "ix": 1}, "e": {"a": 1, "k": [{"i": {"x": [0.667], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 0, "s": [100]}, {"i": {"x": [0.544], "y": [1]}, "o": {"x": [0.333], "y": [0]}, "t": 5, "s": [0]}, {"i": {"x": [0.561], "y": [1]}, "o": {"x": [0.644], "y": [0]}, "t": 10, "s": [0]}, {"t": 30, "s": [100]}], "ix": 2}, "o": {"a": 0, "k": 0, "ix": 3}, "m": 1, "ix": 2, "nm": "Trim Paths 1", "mn": "ADBE Vector Filter - Trim", "hd": false}], "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 4, "ty": 2, "nm": "rule", "refId": "image_1", "sr": 1, "ks": {"o": {"a": 0, "k": 100, "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 1, "k": [{"i": {"x": 0.411, "y": 1}, "o": {"x": 0.536, "y": 0}, "t": 6, "s": [869.05, 575.269, 0], "to": [0, -12.161, 0], "ti": [0, 0, 0]}, {"i": {"x": 0.325, "y": 1}, "o": {"x": 0.603, "y": 0}, "t": 14, "s": [869.05, 502.3, 0], "to": [0, 0, 0], "ti": [0, -12.161, 0]}, {"t": 20, "s": [869.05, 575.269, 0]}], "ix": 2}, "a": {"a": 0, "k": [185.76, 107.761, 0], "ix": 1}, "s": {"a": 1, "k": [{"i": {"x": [0.411, 0.411, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.536, 0.536, 0.333], "y": [0, 0, 0]}, "t": 6, "s": [100, 100, 100]}, {"i": {"x": [0.325, 0.325, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.603, 0.603, 0.333], "y": [0, 0, 0]}, "t": 14, "s": [105, 105, 100]}, {"t": 20, "s": [100, 100, 100]}], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 5, "ty": 2, "nm": "circle", "refId": "image_2", "sr": 1, "ks": {"o": {"a": 0, "k": 100, "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 1, "k": [{"i": {"x": 0.356, "y": 1}, "o": {"x": 0.533, "y": 0}, "t": 4, "s": [829.036, 709.273, 0], "to": [0, -12.162, 0], "ti": [0, 0, 0]}, {"i": {"x": 0.252, "y": 1}, "o": {"x": 0.6, "y": 0}, "t": 12, "s": [829.036, 636.3, 0], "to": [0, 0, 0], "ti": [0, -12.162, 0]}, {"t": 18, "s": [829.036, 709.273, 0]}], "ix": 2}, "a": {"a": 0, "k": [74.402, 43.421, 0], "ix": 1}, "s": {"a": 1, "k": [{"i": {"x": [0.382, 0.382, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.533, 0.533, 0.333], "y": [0, 0, 0]}, "t": 4, "s": [100, 100, 100]}, {"i": {"x": [0.287, 0.287, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.6, 0.6, 0.333], "y": [0, 0, 0]}, "t": 12, "s": [105, 105, 100]}, {"t": 18, "s": [100, 100, 100]}], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 6, "ty": 2, "nm": "circleshadow/graphic.ai", "cl": "ai", "refId": "image_3", "sr": 1, "ks": {"o": {"a": 1, "k": [{"i": {"x": [0.833], "y": [0.833]}, "o": {"x": [0.167], "y": [0.167]}, "t": 4, "s": [0]}, {"i": {"x": [0.833], "y": [0.833]}, "o": {"x": [0.167], "y": [0.167]}, "t": 12, "s": [25]}, {"t": 18, "s": [0]}], "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 0, "k": [829.374, 702.742, 0], "ix": 2}, "a": {"a": 0, "k": [111, 79.5, 0], "ix": 1}, "s": {"a": 0, "k": [96.125, 96.125, 100], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 7, "ty": 2, "nm": "trangle", "refId": "image_4", "sr": 1, "ks": {"o": {"a": 0, "k": 100, "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 1, "k": [{"i": {"x": 0.38, "y": 1}, "o": {"x": 0.531, "y": 0}, "t": 2, "s": [658.173, 632.703, 0], "to": [0, -12.167, 0], "ti": [0, 0, 0]}, {"i": {"x": 0.284, "y": 1}, "o": {"x": 0.597, "y": 0}, "t": 10, "s": [658.173, 559.7, 0], "to": [0, 0, 0], "ti": [0, -12.167, 0]}, {"t": 16, "s": [658.173, 632.703, 0]}], "ix": 2}, "a": {"a": 0, "k": [122.162, 36.615, 0], "ix": 1}, "s": {"a": 1, "k": [{"i": {"x": [0.485, 0.485, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.61, 0.61, 0.333], "y": [0, 0, 0]}, "t": 2, "s": [100, 100, 100]}, {"i": {"x": [0.425, 0.425, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.703, 0.703, 0.333], "y": [0, 0, 0]}, "t": 10, "s": [105, 105, 100]}, {"t": 16, "s": [100, 100, 100]}], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 8, "ty": 2, "nm": "trangleshadow/graphic.ai", "cl": "ai", "refId": "image_5", "sr": 1, "ks": {"o": {"a": 1, "k": [{"i": {"x": [0.833], "y": [0.833]}, "o": {"x": [0.167], "y": [0.167]}, "t": 2, "s": [0]}, {"i": {"x": [0.833], "y": [0.833]}, "o": {"x": [0.167], "y": [0.167]}, "t": 10, "s": [25]}, {"t": 16, "s": [0]}], "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 0, "k": [520.816, 622.041, 0], "ix": 2}, "a": {"a": 0, "k": [293, 72.5, 0], "ix": 1}, "s": {"a": 0, "k": [100, 100, 100], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 9, "ty": 2, "nm": "ruleshadow/graphic.ai", "cl": "ai", "refId": "image_6", "sr": 1, "ks": {"o": {"a": 1, "k": [{"i": {"x": [0.833], "y": [0.833]}, "o": {"x": [0.167], "y": [0.167]}, "t": 6, "s": [0]}, {"i": {"x": [0.833], "y": [0.833]}, "o": {"x": [0.167], "y": [0.167]}, "t": 14, "s": [25]}, {"t": 20, "s": [0]}], "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 0, "k": [868.571, 577.959, 0], "ix": 2}, "a": {"a": 0, "k": [222, 144, 0], "ix": 1}, "s": {"a": 0, "k": [100, 100, 100], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 10, "ty": 2, "nm": "board", "refId": "image_7", "sr": 1, "ks": {"o": {"a": 0, "k": 100, "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 1, "k": [{"i": {"x": 0.536, "y": 1}, "o": {"x": 0.502, "y": 0}, "t": 0, "s": [804.165, 619.963, 0], "to": [0, -2.5, 0], "ti": [0, 0, 0]}, {"i": {"x": 0.493, "y": 1}, "o": {"x": 0.333, "y": 0}, "t": 8, "s": [804.165, 604.963, 0], "to": [0, 0, 0], "ti": [0, -2.5, 0]}, {"t": 14, "s": [804.165, 619.963, 0]}], "ix": 2}, "a": {"a": 0, "k": [352.643, 204.691, 0], "ix": 1}, "s": {"a": 1, "k": [{"i": {"x": [0.483, 0.483, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.529, 0.529, 0.333], "y": [0, 0, 0]}, "t": 0, "s": [100, 100, 100]}, {"i": {"x": [0.422, 0.422, 0.667], "y": [1, 1, 1]}, "o": {"x": [0.594, 0.594, 0.333], "y": [0, 0, 0]}, "t": 8, "s": [105, 105, 100]}, {"t": 14, "s": [100, 100, 100]}], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}, {"ddd": 0, "ind": 11, "ty": 2, "nm": "shadow/graphic.ai", "cl": "ai", "parent": 10, "refId": "image_8", "sr": 1, "ks": {"o": {"a": 0, "k": 59, "ix": 11}, "r": {"a": 0, "k": 0, "ix": 10}, "p": {"a": 0, "k": [348.478, 435.728, 0], "ix": 2}, "a": {"a": 0, "k": [385.5, 188, 0], "ix": 1}, "s": {"a": 0, "k": [100, 100, 100], "ix": 6}}, "ao": 0, "ip": 0, "op": 480, "st": 0, "bm": 0}], "markers": []} \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Animations/workshop-wizard.json b/src/Artemis.UI/Assets/Animations/workshop-wizard.json new file mode 100644 index 000000000..248ce1370 --- /dev/null +++ b/src/Artemis.UI/Assets/Animations/workshop-wizard.json @@ -0,0 +1 @@ +{"v":"5.7.6","fr":25,"ip":0,"op":150,"w":320,"h":200,"nm":"03_Expertise 02","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Line top","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,26.042,0],"ix":2,"l":2},"a":{"a":0,"k":[150,176.042,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[149.984,176.037],[150.031,176.037]],"c":false}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":22,"s":[{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[105,176.074],[195.028,176.085]],"c":false}]},{"t":32,"s":[{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[149.984,176.037],[150.031,176.037]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":12,"op":32,"st":12,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Picto check","parent":6,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[55,161,0],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[43.023,137.816],[50.776,145.569],[67.023,129.321]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":20,"op":150,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Right tube stroke","parent":6,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[55,161,0],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[11,-3.828],[11,-41.5],[-11,-41.5],[-11,-3.828],[-24,17.5],[0,41.5],[24,17.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[55,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass stroke","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":40,"op":190,"st":40,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Liquid green mask","parent":6,"td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[55,161,0],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[66,115.672],[66,78],[44,78],[44,115.672],[31,137],[55,161],[79,137]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false}],"ip":70,"op":220,"st":70,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Liquid green","tt":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[265,161,0],"ix":2,"l":2},"a":{"a":0,"k":[0.49,27.328,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":1,"k":[{"i":{"x":[0.667,0.667],"y":[1,1]},"o":{"x":[0.333,0.333],"y":[0,0]},"t":70,"s":[80,0]},{"t":100,"s":[80,74]}],"ix":2},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":70,"s":[0,27],"to":[0,-6.167],"ti":[0,6.167]},{"t":100,"s":[0,-10]}],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[0.125490196078,0.113725490196,0.113725490196,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.525490196078,0.8,0,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0.49,0.328],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":70,"op":220,"st":70,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Right tube","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":20,"s":[0]},{"i":{"x":[0.258],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":34,"s":[-20]},{"t":44,"s":[0]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.197,"y":1},"o":{"x":0.333,"y":0},"t":20,"s":[355,161,0],"to":[-15,0,0],"ti":[15,0,0]},{"t":34,"s":[265,161,0]}],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[2.761,0],[0,0],[0,2.761],[-2.761,0],[0,0],[0,-2.761]],"o":[[0,0],[-2.761,0],[0,-2.761],[0,0],[2.761,0],[0,2.761]],"v":[[10,5],[-10,5],[-15,0],[-10,-5],[10,-5],[15,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[55,73],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Plug stoke","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,2.761],[-2.761,0],[0,0],[0,-2.761],[-2.761,0],[0,0]],"o":[[0,-2.761],[0,0],[-2.761,0],[0,2.761],[0,0],[-2.761,0]],"v":[[1.5,0],[6.5,-5],[-1.5,-5],[-6.5,0],[-1.5,5],[6.5,5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":30,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[46.5,73],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Plug light","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[2.761,0],[0,0],[0,2.761],[-2.761,0],[0,0],[0,-2.761]],"o":[[0,0],[-2.761,0],[0,-2.761],[0,0],[2.761,0],[0,2.761]],"v":[[10,5],[-10,5],[-15,0],[-10,-5],[10,-5],[15,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[55,73],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Plug bg","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0]],"v":[[-11.977,0.816],[-4.224,8.569],[12.023,-7.679]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[55,137],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Cross","np":2,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[11,-3.828],[11,-41.5],[-11,-41.5],[-11,-3.828],[-24,17.5],[0,41.5],[24,17.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[55,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass stroke","np":2,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,-3.59],[5.523,0],[0,3.59],[-5.523,0]],"o":[[0,3.59],[-5.523,0],[0,-3.59],[5.523,0]],"v":[[10,0],[0,6.5],[-10,0],[0,-6.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":30,"ix":5},"r":1,"bm":3,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[55,127.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass light","np":2,"cix":2,"bm":0,"ix":7,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[11,-3.828],[11,-41.5],[-11,-41.5],[-11,-3.828],[-24,17.5],[0,41.5],[24,17.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[55,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass bg","np":3,"cix":2,"bm":0,"ix":10,"mn":"ADBE Vector Group","hd":false}],"ip":20,"op":150,"st":0,"bm":0},{"ddd":0,"ind":7,"ty":4,"nm":"Picto cross","parent":11,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[55,161,0],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[46.876,145.124],[63.124,128.876]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[46.876,128.876],[63.124,145.124]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":20,"op":170,"st":20,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"Left tube stroke","parent":11,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[55,161,0],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[66,115.672],[66,78],[44,78],[44,115.672],[31,137],[55,161],[79,137]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":40,"op":190,"st":40,"bm":0},{"ddd":0,"ind":9,"ty":4,"nm":"Liquid red mask","parent":11,"td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[55,161,0],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[66,115.672],[66,78],[44,78],[44,115.672],[31,137],[55,161],[79,137]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false}],"ip":70,"op":220,"st":70,"bm":0},{"ddd":0,"ind":10,"ty":4,"nm":"Liquid red","tt":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[55,161,0],"ix":2,"l":2},"a":{"a":0,"k":[0.49,27.328,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":1,"k":[{"i":{"x":[0.667,0.667],"y":[1,1]},"o":{"x":[0.333,0.333],"y":[0,0]},"t":70,"s":[80,0]},{"t":100,"s":[80,28]}],"ix":2},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":70,"s":[0,27],"to":[0,-2.333],"ti":[0,2.333]},{"t":100,"s":[0,13]}],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"st","c":{"a":0,"k":[0.125490196078,0.113725490196,0.113725490196,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":0,"k":[0.776470588235,0.035294117647,0,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0.49,0.328],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":70,"op":220,"st":70,"bm":0},{"ddd":0,"ind":11,"ty":4,"nm":"Left tube","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":20,"s":[0]},{"i":{"x":[0.258],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":34,"s":[20]},{"t":44,"s":[0]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.197,"y":1},"o":{"x":0.333,"y":0},"t":20,"s":[-35,161,0],"to":[10.983,0,0],"ti":[-4.017,0,0]},{"t":34,"s":[55,161,0]}],"ix":2,"l":2},"a":{"a":0,"k":[55,161,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[2.761,0],[0,0],[0,2.761],[-2.761,0],[0,0],[0,-2.761]],"o":[[0,0],[-2.761,0],[0,-2.761],[0,0],[2.761,0],[0,2.761]],"v":[[10,5],[-10,5],[-15,0],[-10,-5],[10,-5],[15,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[55,73],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Plug stoke","np":2,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,2.761],[-2.761,0],[0,0],[0,-2.761],[-2.761,0],[0,0]],"o":[[0,-2.761],[0,0],[-2.761,0],[0,2.761],[0,0],[-2.761,0]],"v":[[1.5,0],[6.5,-5],[-1.5,-5],[-6.5,0],[-1.5,5],[6.5,5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":30,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[46.5,73],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Plug light","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[2.761,0],[0,0],[0,2.761],[-2.761,0],[0,0],[0,-2.761]],"o":[[0,0],[-2.761,0],[0,-2.761],[0,0],[2.761,0],[0,2.761]],"v":[[10,5],[-10,5],[-15,0],[-10,-5],[10,-5],[15,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[55,73],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Plug bg","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-8.124,8.124],[8.124,-8.124]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[-8.124,-8.124],[8.124,8.124]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":5,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[55,137],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Cross","np":3,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[11,-3.828],[11,-41.5],[-11,-41.5],[-11,-3.828],[-24,17.5],[0,41.5],[24,17.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[55,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass stroke","np":2,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,-3.59],[5.523,0],[0,3.59],[-5.523,0]],"o":[[0,3.59],[-5.523,0],[0,-3.59],[5.523,0]],"v":[[10,0],[0,6.5],[-10,0],[0,-6.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":30,"ix":5},"r":1,"bm":3,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[55,127.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass light","np":2,"cix":2,"bm":0,"ix":7,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[7.72,3.989],[0,0],[0,0],[0,0],[0,-9.289],[-13.255,0],[0,13.255]],"o":[[0,0],[0,0],[0,0],[-7.719,3.989],[0,13.255],[13.255,0],[0,-9.289]],"v":[[11,-3.828],[11,-41.5],[-11,-41.5],[-11,-3.828],[-24,17.5],[0,41.5],[24,17.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 2","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[55,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass bg","np":3,"cix":2,"bm":0,"ix":10,"mn":"ADBE Vector Group","hd":false}],"ip":20,"op":170,"st":20,"bm":0},{"ddd":0,"ind":12,"ty":4,"nm":"Liquid stroke","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[159.97,144.056,0],"ix":2,"l":2},"a":{"a":0,"k":[-0.03,44.056,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":20,"s":[{"i":[[0,2.065],[1.271,0],[0.492,0],[0,0],[0,0]],"o":[[-1.226,0],[0.689,0],[0,0.698],[0,0],[0,0]],"v":[[15.077,32.985],[15.008,33.017],[15.055,33.002],[14.982,33.223],[15.001,33.219]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":23,"s":[{"i":[[0,9.375],[14.825,0],[-0.638,0],[0,0],[0,0]],"o":[[-9.96,0.057],[-15.008,0],[0,11.582],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":33,"s":[{"i":[[0,9.375],[15.075,6.983],[10.029,-7.085],[0,0],[0,0]],"o":[[-9.793,7.099],[-15.011,-6.954],[0,11.625],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":43,"s":[{"i":[[0,9.375],[15.023,-6.995],[9.925,7.019],[0,0],[0,0]],"o":[[-9.877,-6.984],[-14.998,6.983],[0,11.625],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":53,"s":[{"i":[[0,9.375],[15.008,-4.233],[9.904,4.248],[0,0],[0,0]],"o":[[-9.752,-4.026],[-15.008,4.233],[0,11.625],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":63,"s":[{"i":[[0,9.375],[14.825,0],[-0.638,0],[0,0],[0,0]],"o":[[-9.96,0.057],[-15.008,0],[0,11.582],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,9.375],[14.825,0],[-0.638,0],[0,0],[0,0]],"o":[[-9.96,0.057],[-15.008,0],[0,11.582],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"t":79,"s":[{"i":[[0.085,0.165],[-0.035,-0.119],[0.027,0.022],[0,0],[0,0]],"o":[[0.045,0.084],[-0.096,-0.14],[-0.034,-0.04],[0,0],[0,0]],"v":[[-21.779,32.896],[-21.679,33.14],[-21.68,33.019],[-21.609,32.74],[-21.712,32.629]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":0,"k":36,"ix":2},"o":{"a":0,"k":4,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.1254902035,0.113725490868,0.113725490868,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":20,"op":79,"st":20,"bm":0},{"ddd":0,"ind":13,"ty":4,"nm":"Liquid","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[159.97,144.056,0],"ix":2,"l":2},"a":{"a":0,"k":[-0.03,44.056,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":20,"s":[{"i":[[0,2.065],[1.271,0],[0.492,0],[0,0],[0,0]],"o":[[-1.226,0],[0.689,0],[0,0.698],[0,0],[0,0]],"v":[[15.077,32.985],[15.008,33.017],[15.055,33.002],[15.065,56.556],[15.084,56.552]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":23,"s":[{"i":[[0,9.375],[14.825,0],[-0.638,0],[0,0],[0,0]],"o":[[-9.96,0.057],[-15.008,0],[0,11.582],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":33,"s":[{"i":[[0,9.375],[15.075,6.983],[10.029,-7.085],[0,0],[0,0]],"o":[[-9.793,7.099],[-15.011,-6.954],[0,11.625],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":43,"s":[{"i":[[0,9.375],[15.023,-6.995],[9.925,7.019],[0,0],[0,0]],"o":[[-9.877,-6.984],[-14.998,6.983],[0,11.625],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":53,"s":[{"i":[[0,9.375],[15.008,-4.233],[9.904,4.248],[0,0],[0,0]],"o":[[-9.752,-4.026],[-15.008,4.233],[0,11.625],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":63,"s":[{"i":[[0,9.375],[14.825,0],[-0.638,0],[0,0],[0,0]],"o":[[-9.96,0.057],[-15.008,0],[0,11.582],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":70,"s":[{"i":[[0,9.375],[14.825,0],[-0.638,0],[0,0],[0,0]],"o":[[-9.96,0.057],[-15.008,0],[0,11.582],[0,0],[0,0]],"v":[[38.877,32.984],[0.008,33.017],[-38.946,33.002],[-38.936,56.556],[38.883,56.551]],"c":true}]},{"t":79,"s":[{"i":[[0.085,0.165],[-0.035,-0.119],[0.027,0.022],[0,0],[0,0]],"o":[[0.045,0.084],[-0.096,-0.14],[-0.034,-0.04],[0,0],[0,0]],"v":[[-21.779,32.896],[-21.679,33.14],[-21.68,33.019],[-21.67,56.573],[-21.773,56.463]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":54,"s":[0.007843137719,0.835294127464,0.933333337307,1]},{"t":78,"s":[1,0.341176480055,0.180392161012,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"gr","it":[{"ty":"fl","c":{"a":0,"k":[1,0,0,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Group 1","np":1,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false}],"ip":20,"op":79,"st":20,"bm":0},{"ddd":0,"ind":14,"ty":4,"nm":"Pad mask","td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,84,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[320,112],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":14,"op":164,"st":14,"bm":0},{"ddd":0,"ind":15,"ty":4,"nm":"Pad","tt":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.82],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":24,"s":[40]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":44,"s":[-10]},{"t":74,"s":[25]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":14,"s":[160,-10,0],"to":[0,23.5,0],"ti":[0,-21.667,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":24,"s":[160,131,0],"to":[0,21.667,0],"ti":[0,-0.167,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":34,"s":[160,120,0],"to":[0,0.167,0],"ti":[0,-1.167,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":44,"s":[160,132,0],"to":[0,1.167,0],"ti":[0,0.833,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.333,"y":0.333},"t":56,"s":[160,127,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":61,"s":[160,127,0],"to":[0,7.5,0],"ti":[0,-7.5,0]},{"t":81,"s":[160,172,0]}],"ix":2,"l":2},"a":{"a":0,"k":[160,134,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,-1.657],[1.657,0],[0,1.657],[-1.657,0]],"o":[[0,1.657],[-1.657,0],[0,-1.657],[1.657,0]],"v":[[3,0],[0,3],[-3,0],[0,-3]],"c":true},"ix":2},"nm":"Button top","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[1.657,0],[0,1.657],[-1.657,0],[0,-1.657]],"o":[[-1.657,0],[0,-1.657],[1.657,0],[0,1.657]],"v":[[6,9],[3,6],[6,3],[9,6]],"c":true},"ix":2},"nm":"Button right","mn":"ADBE Vector Shape - Group","hd":false},{"ind":2,"ty":"sh","ix":3,"ks":{"a":0,"k":{"i":[[0,-1.657],[1.657,0],[0,1.657],[-1.657,0]],"o":[[0,1.657],[-1.657,0],[0,-1.657],[1.657,0]],"v":[[3,12],[0,15],[-3,12],[0,9]],"c":true},"ix":2},"nm":"Button bottom","mn":"ADBE Vector Shape - Group","hd":false},{"ind":3,"ty":"sh","ix":4,"ks":{"a":0,"k":{"i":[[1.657,0],[0,1.657],[-1.657,0],[0,-1.657]],"o":[[-1.657,0],[0,-1.657],[1.657,0],[0,1.657]],"v":[[-6,9],[-9,6],[-6,3],[-3,6]],"c":true},"ix":2},"nm":"Button left","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[176.5,128],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Pad buttons","np":5,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[4.4,0],[0,0],[0,4.4],[-4.4,0],[0,-4.4]],"o":[[0,0],[-4.4,0],[0,-4.4],[4.4,0],[0,4.4]],"v":[[0,8],[0,8],[-8,0],[0,-8],[8,0]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[143.5,134],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Pad joystick","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[3.85,0],[0,0],[0,3.85],[0,0],[-3.85,0],[0,0],[0,-3.85],[0,0]],"o":[[0,0],[-3.85,0],[0,0],[0,-3.85],[0,0],[3.85,0],[0,0],[0,3.85]],"v":[[26,17],[-26,17],[-33,10],[-33,-10],[-26,-17],[26,-17],[33,-10],[33,10]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[160,134],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Pad stroke","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[160,117],[160,151]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Pad central line","np":2,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[0,3.85],[0,0],[-3.85,0],[0,0]],"o":[[0,0],[-3.85,0],[0,0],[0,-3.85],[0,0],[0,0]],"v":[[16.5,17],[-9.5,17],[-16.5,10],[-16.5,-10],[-9.5,-17],[16.5,-17]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.004000000393,0.263000009574,0.395999983245,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[143.5,134],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Pad left side bg","np":2,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[3.85,0],[0,0],[0,0],[0,0],[0,-3.85],[0,0]],"o":[[0,0],[0,0],[0,0],[3.85,0],[0,0],[0,3.85]],"v":[[9.5,17],[-16.5,17],[-16.5,-17],[9.5,-17],[16.5,-10],[16.5,10]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,0.340999977261,0.180000005984,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[176.5,134],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Pad right side bg","np":2,"cix":2,"bm":0,"ix":6,"mn":"ADBE Vector Group","hd":false}],"ip":14,"op":164,"st":14,"bm":0},{"ddd":0,"ind":16,"ty":4,"nm":"Cup liquid mask","td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,140,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[320,120],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":4,"op":154,"st":4,"bm":0},{"ddd":0,"ind":17,"ty":4,"nm":"Cup liquid","tt":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.178,"y":1},"o":{"x":0.333,"y":0},"t":4,"s":[160,10,0],"to":[0,15,0],"ti":[0,-15,0]},{"t":14,"s":[160,100,0]}],"ix":2,"l":2},"a":{"a":0,"k":[160,100,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":22,"s":[{"i":[[5.522,0],[0,0],[0,5.522],[0,0],[-20.597,0],[-14.083,0],[0,0]],"o":[[0,0],[-5.523,0],[0,0],[15.999,0],[21.879,0],[0,0],[0,5.522]],"v":[[200,161],[120,161],[110,151],[110.001,120.006],[160.013,120.025],[209.999,120.024],[210,151]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":30,"s":[{"i":[[5.522,0],[0,0],[0,5.522],[0,0],[-20.705,-7.069],[-12.005,4.802],[0,0]],"o":[[0,0],[-5.523,0],[0,0],[14.304,-5.722],[20.705,7.069],[0,0],[0,5.522]],"v":[[200,161],[120,161],[110,151],[110.001,120.006],[160.013,120.025],[209.999,120.024],[210,151]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":46,"s":[{"i":[[5.522,0],[0,0],[0,5.522],[0,0],[-20.763,10.059],[-11.499,-5.94],[0,0]],"o":[[0,0],[-5.523,0],[0,0],[14.082,5.994],[19.69,-9.539],[0,0],[0,5.522]],"v":[[200,161],[120,161],[110,151],[110.001,120.006],[160.013,120.025],[209.999,120.024],[210,151]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":60,"s":[{"i":[[5.522,0],[0,0],[0,5.522],[0,0],[-20.597,0],[-14.083,0],[0,0]],"o":[[0,0],[-5.523,0],[0,0],[15.999,0],[21.879,0],[0,0],[0,5.522]],"v":[[200,161],[120,161],[110,151],[110.001,120.006],[160.013,120.025],[209.999,120.024],[210,151]],"c":true}]},{"t":90,"s":[{"i":[[5.522,0],[0,0],[0,5.522],[0,0],[-20.597,0],[-14.083,0],[0,0]],"o":[[0,0],[-5.523,0],[0,0],[15.999,0],[21.879,0],[0,0],[0,5.522]],"v":[[200,161],[120,161],[110,151],[110.001,138.006],[160.013,138.025],[209.999,138.024],[210,151]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.1254902035,0.113725490868,0.113725490868,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":54,"s":[0.007843137719,0.835294127464,0.933333337307,1]},{"t":78,"s":[1,0.341176480055,0.180392161012,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Liquid","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":4,"op":154,"st":4,"bm":0},{"ddd":0,"ind":18,"ty":4,"nm":"Straw right","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,100,0],"ix":2,"l":2},"a":{"a":0,"k":[160,100,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[-4.694,0],[0,-4.694],[0,0],[-4.694,0],[0,4.694],[0,0],[-4.694,0],[0,-4.694],[0,0],[-4.694,0],[0,4.694],[0,0],[-4.697,-0.004],[0,-4.692],[0,0]],"o":[[0,0],[0,-4.694],[4.694,0],[0,0],[0,4.694],[4.694,0],[0,0],[0,-4.694],[4.694,0],[0,0],[0,4.694],[4.694,0],[0,0],[0,-4.697],[4.691,0.004],[0,0],[0,0]],"v":[[-42.5,74],[-42.5,-48],[-34,-56.5],[-25.5,-48],[-25.5,-33],[-17,-24.5],[-8.5,-33],[-8.5,-48],[0,-56.5],[8.5,-48],[8.5,-33],[17,-24.5],[25.5,-33],[25.5,-47.993],[34.007,-56.493],[42.5,-47.993],[42.5,12.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":75,"s":[0]},{"t":110,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":45,"s":[0]},{"t":75,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.525,0.8,0,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 2","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[222.5,75.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Straw color","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[-4.694,0],[0,-4.694],[0,0],[-4.694,0],[0,4.694],[0,0],[-4.694,0],[0,-4.694],[0,0],[-4.694,0],[0,4.694],[0,0],[-4.697,-0.004],[0,-4.692],[0,0]],"o":[[0,0],[0,-4.694],[4.694,0],[0,0],[0,4.694],[4.694,0],[0,0],[0,-4.694],[4.694,0],[0,0],[0,4.694],[4.694,0],[0,0],[0,-4.697],[4.691,0.004],[0,0],[0,0]],"v":[[-42.5,74],[-42.5,-48],[-34,-56.5],[-25.5,-48],[-25.5,-33],[-17,-24.5],[-8.5,-33],[-8.5,-48],[0,-56.5],[8.5,-48],[8.5,-33],[17,-24.5],[25.5,-33],[25.5,-47.993],[34.007,-56.493],[42.5,-47.993],[42.5,12.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":43,"s":[0]},{"t":73,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.780392169952,0.921568632126,0.976470589638,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":4,"bm":0,"nm":"Stroke 2","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[222.5,75.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Straw bg","np":4,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false}],"ip":43,"op":193,"st":43,"bm":0},{"ddd":0,"ind":19,"ty":4,"nm":"Straw left","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,100,0],"ix":2,"l":2},"a":{"a":0,"k":[160,100,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[4.418,0],[0,0],[0,-4.418],[-4.418,0],[0,0],[0,-4.418],[0,0],[4.418,0],[0,0],[0,-4.419],[0,0]],"o":[[0,0],[0,-4.418],[0,0],[-4.418,0],[0,4.418],[0,0],[4.418,0],[0,0],[0,4.418],[0,0],[-4.418,0],[0,0],[0,0]],"v":[[42.5,74],[42.5,-48.5],[34.5,-56.5],[-34.5,-56.5],[-42.5,-48.5],[-34.5,-40.5],[18.5,-40.5],[26.5,-32.5],[26.5,-32.321],[18.5,-24.321],[-34.5,-24.321],[-42.5,-16.321],[-42.5,12.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":65,"s":[0]},{"t":100,"s":[100]}],"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":45,"s":[0]},{"t":75,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.776000019148,0.035000000748,0,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 2","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[97.5,75.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Straw color","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0],[4.418,0],[0,0],[0,-4.418],[-4.418,0],[0,0],[0,-4.418],[0,0],[4.418,0],[0,0],[0,-4.419],[0,0]],"o":[[0,0],[0,-4.418],[0,0],[-4.418,0],[0,4.418],[0,0],[4.418,0],[0,0],[0,4.418],[0,0],[-4.418,0],[0,0],[0,0]],"v":[[42.5,74],[42.5,-48.5],[34.5,-56.5],[-34.5,-56.5],[-42.5,-48.5],[-34.5,-40.5],[18.5,-40.5],[26.5,-32.5],[26.5,-32.321],[18.5,-24.321],[-34.5,-24.321],[-42.5,-16.321],[-42.5,12.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":43,"s":[0]},{"t":73,"s":[100]}],"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[0.780392169952,0.921568632126,0.976470589638,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 2","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":12,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[97.5,75.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Straw bg","np":4,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false}],"ip":43,"op":193,"st":43,"bm":0},{"ddd":0,"ind":20,"ty":4,"nm":"Top line","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"t":1,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,100,0],"ix":2,"l":2},"a":{"a":0,"k":[160,100,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.333,"y":0},"t":0,"s":[{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[159,78],[161,78]],"c":false}]},{"t":8,"s":[{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[104,78],[216,78]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":0,"op":150,"st":0,"bm":0},{"ddd":0,"ind":21,"ty":4,"nm":"Cup mask","td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,140,0],"ix":2,"l":2},"a":{"a":0,"k":[0,0,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ty":"rc","d":1,"s":{"a":0,"k":[320,120],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"r":{"a":0,"k":0,"ix":4},"nm":"Rectangle Path 1","mn":"ADBE Vector Shape - Rect","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Rectangle 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":4,"op":154,"st":4,"bm":0},{"ddd":0,"ind":22,"ty":4,"nm":"Cup","tt":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.178,"y":1},"o":{"x":0.333,"y":0},"t":4,"s":[160,10,0],"to":[0,15,0],"ti":[0,-15,0]},{"t":14,"s":[160,100,0]}],"ix":2,"l":2},"a":{"a":0,"k":[160,100,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[200,85.569],[200,99.569]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ind":1,"ty":"sh","ix":2,"ks":{"a":0,"k":{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[200,107.791],[200,111.791]],"c":false},"ix":2},"nm":"Path 2","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Reflect","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[5.522,0],[0,0],[0,5.522],[0,0],[0,0],[0,0]],"o":[[0,0],[-5.523,0],[0,0],[0,0],[0,0],[0,5.522]],"v":[[40,41.5],[-40,41.5],[-50,31.5],[-50,-41.5],[50,-41.5],[50,31.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.125,0.113999998803,0.113999998803,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":1,"lj":1,"ml":10,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[160,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass stroke","np":2,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,5.522],[0,0],[0,0],[0,0],[-5.523,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,5.522],[0,0],[-5.523,0]],"v":[[2,31.5],[2,-41.5],[-12,-41.5],[-12,31.5],[-2,41.5],[12,41.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":20,"ix":5},"r":1,"bm":3,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[122,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Light bg","np":2,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[5.522,0],[0,0],[0,5.522],[0,0],[0,0],[0,0]],"o":[[0,0],[-5.523,0],[0,0],[0,0],[0,0],[0,5.522]],"v":[[40,41.5],[-40,41.5],[-50,31.5],[-50,-41.5],[50,-41.5],[50,31.5]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.779999976065,0.922000002394,0.976000019148,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[160,119.5],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Glass bg","np":2,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false}],"ip":4,"op":154,"st":4,"bm":0},{"ddd":0,"ind":23,"ty":4,"nm":"Shadow","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[0]},{"t":5,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160,176.042,0],"ix":2,"l":2},"a":{"a":0,"k":[150,176.042,0],"ix":1,"l":2},"s":{"a":0,"k":[100,100,100],"ix":6,"l":2}},"ao":0,"shapes":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":4,"s":[{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[149.25,176.042],[150.75,176.042]],"c":false}]},{"t":14,"s":[{"i":[[0,0],[0,0]],"o":[[0,0],[0,0]],"v":[[113,176.042],[187,176.042]],"c":false}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"st","c":{"a":0,"k":[0.917647063732,0.917647063732,0.917647063732,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":4,"ix":5},"lc":2,"lj":2,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false}],"ip":4,"op":149,"st":4,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/src/Artemis.UI/Assets/Fonts/RobotoMono-Bold.ttf b/src/Artemis.UI/Assets/Fonts/RobotoMono-Bold.ttf new file mode 100644 index 000000000..8eff26c1e Binary files /dev/null and b/src/Artemis.UI/Assets/Fonts/RobotoMono-Bold.ttf differ diff --git a/src/Artemis.UI/Assets/Fonts/RobotoMono-BoldItalic.ttf b/src/Artemis.UI/Assets/Fonts/RobotoMono-BoldItalic.ttf new file mode 100644 index 000000000..99299640b Binary files /dev/null and b/src/Artemis.UI/Assets/Fonts/RobotoMono-BoldItalic.ttf differ diff --git a/src/Artemis.UI/Assets/Fonts/RobotoMono-Italic.ttf b/src/Artemis.UI/Assets/Fonts/RobotoMono-Italic.ttf new file mode 100644 index 000000000..cf2b5b39c Binary files /dev/null and b/src/Artemis.UI/Assets/Fonts/RobotoMono-Italic.ttf differ diff --git a/src/Artemis.UI/Assets/Fonts/RobotoMono-Regular.ttf b/src/Artemis.UI/Assets/Fonts/RobotoMono-Regular.ttf new file mode 100644 index 000000000..d9371a1bd Binary files /dev/null and b/src/Artemis.UI/Assets/Fonts/RobotoMono-Regular.ttf differ diff --git a/src/Artemis.UI/Assets/Fonts/RobotoMono-SemiBold.ttf b/src/Artemis.UI/Assets/Fonts/RobotoMono-SemiBold.ttf new file mode 100644 index 000000000..06aea97f3 Binary files /dev/null and b/src/Artemis.UI/Assets/Fonts/RobotoMono-SemiBold.ttf differ diff --git a/src/Artemis.UI/Assets/Fonts/RobotoMono-SemiBoldItalic.ttf b/src/Artemis.UI/Assets/Fonts/RobotoMono-SemiBoldItalic.ttf new file mode 100644 index 000000000..72b0fc6d9 Binary files /dev/null and b/src/Artemis.UI/Assets/Fonts/RobotoMono-SemiBoldItalic.ttf differ diff --git a/src/Artemis.UI/Converters/DateTimeConverter.cs b/src/Artemis.UI/Converters/DateTimeConverter.cs new file mode 100644 index 000000000..7baf68ebb --- /dev/null +++ b/src/Artemis.UI/Converters/DateTimeConverter.cs @@ -0,0 +1,34 @@ +using System; +using System.Globalization; +using Artemis.WebClient.Workshop; +using Avalonia.Data.Converters; +using Humanizer; + +namespace Artemis.UI.Converters; + +public class DateTimeConverter : IValueConverter +{ + public object? Convert(object? value, Type targetType, object? parameter, CultureInfo culture) + { + if (value is DateTimeOffset dateTimeOffset) + { + return parameter?.ToString() == "humanize" + ? dateTimeOffset.ToLocalTime().Humanize() + : dateTimeOffset.ToLocalTime().ToString(parameter?.ToString() ?? "g"); + } + + if (value is DateTime dateTime) + { + return parameter?.ToString() == "humanize" + ? dateTime.ToLocalTime().Humanize() + : dateTime.ToLocalTime().ToString(parameter?.ToString() ?? "g"); + } + + return value; + } + + public object? ConvertBack(object? value, Type targetType, object? parameter, CultureInfo culture) + { + return value; + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Converters/EntryIconUriConverter.cs b/src/Artemis.UI/Converters/EntryIconUriConverter.cs new file mode 100644 index 000000000..fdd185fec --- /dev/null +++ b/src/Artemis.UI/Converters/EntryIconUriConverter.cs @@ -0,0 +1,19 @@ +using System; +using System.Globalization; +using Artemis.WebClient.Workshop; +using Avalonia.Data.Converters; + +namespace Artemis.UI.Converters; + +public class EntryIconUriConverter : IValueConverter +{ + public object? Convert(object? value, Type targetType, object? parameter, CultureInfo culture) + { + return $"{WorkshopConstants.WORKSHOP_URL}/entries/{value}/icon"; + } + + public object? ConvertBack(object? value, Type targetType, object? parameter, CultureInfo culture) + { + return value; + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Extensions/Bitmap.cs b/src/Artemis.UI/Extensions/Bitmap.cs new file mode 100644 index 000000000..e30bc754f --- /dev/null +++ b/src/Artemis.UI/Extensions/Bitmap.cs @@ -0,0 +1,48 @@ +using System; +using System.IO; +using System.Threading.Tasks; +using Avalonia.Media.Imaging; +using SkiaSharp; + +namespace Artemis.UI.Extensions; + +public class BitmapExtensions +{ + public static Bitmap LoadAndResize(string file, int size) + { + using SKBitmap source = SKBitmap.Decode(file); + return Resize(source, size); + } + + public static Bitmap LoadAndResize(Stream stream, int size) + { + stream.Seek(0, SeekOrigin.Begin); + using MemoryStream copy = new(); + stream.CopyTo(copy); + copy.Seek(0, SeekOrigin.Begin); + using SKBitmap source = SKBitmap.Decode(copy); + return Resize(source, size); + } + + private static Bitmap Resize(SKBitmap source, int size) + { + // Get smaller dimension. + int minDim = Math.Min(source.Width, source.Height); + + // Calculate crop rectangle position for center crop. + int deltaX = (source.Width - minDim) / 2; + int deltaY = (source.Height - minDim) / 2; + + // Create crop rectangle. + SKRectI rect = new(deltaX, deltaY, deltaX + minDim, deltaY + minDim); + + // Do the actual cropping of the bitmap. + using SKBitmap croppedBitmap = new(minDim, minDim); + source.ExtractSubset(croppedBitmap, rect); + + // Resize to the desired size after cropping. + using SKBitmap resizedBitmap = croppedBitmap.Resize(new SKImageInfo(size, size), SKFilterQuality.High); + + return new Bitmap(resizedBitmap.Encode(SKEncodedImageFormat.Png, 100).AsStream()); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Extensions/IActivatableViewModelExtensions.cs b/src/Artemis.UI/Extensions/IActivatableViewModelExtensions.cs new file mode 100644 index 000000000..8186b0222 --- /dev/null +++ b/src/Artemis.UI/Extensions/IActivatableViewModelExtensions.cs @@ -0,0 +1,15 @@ +using System; +using System.Reactive.Disposables; +using System.Threading.Tasks; +using Avalonia.Threading; +using ReactiveUI; + +namespace Artemis.UI.Extensions; + +public static class ActivatableViewModelExtensions +{ + public static void WhenActivatedAsync(this IActivatableViewModel item, Func block) + { + item.WhenActivated(d => Dispatcher.UIThread.InvokeAsync(async () => await block(d))); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Extensions/MaterialIconKindExtensions.cs b/src/Artemis.UI/Extensions/MaterialIconKindExtensions.cs new file mode 100644 index 000000000..c594ce7bc --- /dev/null +++ b/src/Artemis.UI/Extensions/MaterialIconKindExtensions.cs @@ -0,0 +1,35 @@ +using System; +using System.IO; +using Material.Icons; +using SkiaSharp; + +namespace Artemis.UI.Extensions; + +public static class MaterialIconKindExtensions +{ + public static Stream EncodeToBitmap(this MaterialIconKind icon, int size, int margin, SKColor color) + { + string geometrySource = MaterialIconDataProvider.GetData(icon); + + SKBitmap bitmap = new(size, size); + using (SKCanvas canvas = new(bitmap)) + { + canvas.Clear(SKColors.Transparent); + + // Parse and render the geometry data using SkiaSharp's SKPath + using SKPath path = SKPath.ParseSvgPathData(geometrySource); + using SKPaint paint = new() {Color = color, IsAntialias = true,}; + + // Calculate scaling and translation to fit the icon in the 100x100 area with 14 pixels margin + float scale = Math.Min(size / path.Bounds.Width, size / path.Bounds.Height); + path.Transform(SKMatrix.CreateTranslation(path.Bounds.Left * -1, path.Bounds.Top * -1)); + path.Transform(SKMatrix.CreateScale(scale, scale)); + canvas.Scale((size - margin * 2) / (float) size, (size - margin * 2) / (float) size, size / 2f, size / 2f); + canvas.DrawPath(path, paint); + } + + MemoryStream stream = new(); + bitmap.Encode(stream, SKEncodedImageFormat.Png, 100); + return stream; + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Extensions/ZipArchiveExtensions.cs b/src/Artemis.UI/Extensions/ZipArchiveExtensions.cs index 0d1fc507d..ad1900caa 100644 --- a/src/Artemis.UI/Extensions/ZipArchiveExtensions.cs +++ b/src/Artemis.UI/Extensions/ZipArchiveExtensions.cs @@ -2,6 +2,7 @@ using System; using System.IO; using System.IO.Compression; using System.Threading; +using Artemis.UI.Shared.Utilities; namespace Artemis.UI.Extensions; @@ -16,7 +17,7 @@ public static class ZipArchiveExtensions /// A boolean indicating whether to override existing files /// The progress to report to. /// A cancellation token - public static void ExtractToDirectory(this ZipArchive source, string destinationDirectoryName, bool overwriteFiles, IProgress progress, CancellationToken cancellationToken) + public static void ExtractToDirectory(this ZipArchive source, string destinationDirectoryName, bool overwriteFiles, IProgress progress, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); @@ -28,7 +29,7 @@ public static class ZipArchiveExtensions { ZipArchiveEntry entry = source.Entries[index]; entry.ExtractRelativeToDirectory(destinationDirectoryName, overwriteFiles); - progress.Report((index + 1f) / source.Entries.Count * 100f); + progress.Report(new StreamProgress(index + 1, source.Entries.Count)); cancellationToken.ThrowIfCancellationRequested(); } } diff --git a/src/Artemis.UI/MainWindow.axaml b/src/Artemis.UI/MainWindow.axaml index f91d7815a..70f2b258b 100644 --- a/src/Artemis.UI/MainWindow.axaml +++ b/src/Artemis.UI/MainWindow.axaml @@ -12,6 +12,9 @@ MinWidth="600" MinHeight="400" PointerReleased="InputElement_OnPointerReleased"> + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Settings/SettingsView.axaml.cs b/src/Artemis.UI/Screens/Settings/SettingsView.axaml.cs index 75d3d5b12..6e04c8c28 100644 --- a/src/Artemis.UI/Screens/Settings/SettingsView.axaml.cs +++ b/src/Artemis.UI/Screens/Settings/SettingsView.axaml.cs @@ -3,8 +3,7 @@ using System.Reactive.Disposables; using Artemis.UI.Shared; using Avalonia.ReactiveUI; using Avalonia.Threading; -using FluentAvalonia.UI.Media.Animation; -using FluentAvalonia.UI.Navigation; +using FluentAvalonia.UI.Controls; using ReactiveUI; namespace Artemis.UI.Screens.Settings; @@ -19,6 +18,11 @@ public partial class SettingsView : ReactiveUserControl private void Navigate(ViewModelBase viewModel) { - Dispatcher.UIThread.Invoke(() => TabFrame.NavigateFromObject(viewModel, new FrameNavigationOptions {TransitionInfoOverride = new SlideNavigationTransitionInfo()})); + Dispatcher.UIThread.Invoke(() => TabFrame.NavigateFromObject(viewModel)); + } + + private void NavigationView_OnBackRequested(object? sender, NavigationViewBackRequestedEventArgs e) + { + ViewModel?.GoBack(); } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Settings/SettingsViewModel.cs b/src/Artemis.UI/Screens/Settings/SettingsViewModel.cs index 2a747e5f9..3391aeab8 100644 --- a/src/Artemis.UI/Screens/Settings/SettingsViewModel.cs +++ b/src/Artemis.UI/Screens/Settings/SettingsViewModel.cs @@ -4,39 +4,40 @@ using System.Linq; using System.Reactive.Disposables; using System.Threading; using System.Threading.Tasks; +using Artemis.UI.Routing; using Artemis.UI.Shared; using Artemis.UI.Shared.Routing; using ReactiveUI; namespace Artemis.UI.Screens.Settings; -public class SettingsViewModel : RoutableScreen, IMainScreenViewModel +public class SettingsViewModel : RoutableHostScreen, IMainScreenViewModel { private readonly IRouter _router; - private SettingsTab? _selectedTab; + private RouteViewModel? _selectedTab; public SettingsViewModel(IRouter router) { _router = router; - SettingTabs = new ObservableCollection + SettingTabs = new ObservableCollection { - new("general", "General"), - new("plugins", "Plugins"), - new("devices", "Devices"), - new("releases", "Releases"), - new("about", "About"), + new("General", "settings/general"), + new("Plugins", "settings/plugins"), + new("Devices", "settings/devices"), + new("Releases", "settings/releases"), + new("About", "settings/about"), }; // Navigate on tab change this.WhenActivated(d => this.WhenAnyValue(vm => vm.SelectedTab) .WhereNotNull() - .Subscribe(s => _router.Navigate($"settings/{s.Path}", new RouterNavigationOptions {IgnoreOnPartialMatch = true})) + .Subscribe(s => _router.Navigate(s.Path, new RouterNavigationOptions {IgnoreOnPartialMatch = true})) .DisposeWith(d)); } - public ObservableCollection SettingTabs { get; } + public ObservableCollection SettingTabs { get; } - public SettingsTab? SelectedTab + public RouteViewModel? SelectedTab { get => _selectedTab; set => RaiseAndSetIfChanged(ref _selectedTab, value); @@ -52,6 +53,11 @@ public class SettingsViewModel : RoutableScreen, IMain // Always show a tab, if there is none forward to the first if (SelectedTab == null) - await _router.Navigate($"settings/{SettingTabs.First().Path}"); + await _router.Navigate(SettingTabs.First().Path); + } + + public void GoBack() + { + _router.Navigate("workshop"); } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Settings/Tabs/AboutTabView.axaml b/src/Artemis.UI/Screens/Settings/Tabs/AboutTabView.axaml index 8016a6b79..506c960d7 100644 --- a/src/Artemis.UI/Screens/Settings/Tabs/AboutTabView.axaml +++ b/src/Artemis.UI/Screens/Settings/Tabs/AboutTabView.axaml @@ -5,20 +5,21 @@ xmlns:controls="clr-namespace:FluentAvalonia.UI.Controls;assembly=FluentAvalonia" xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" xmlns:vm="clr-namespace:Artemis.UI.Screens.Settings;assembly=Artemis.UI" + xmlns:il="clr-namespace:AsyncImageLoader;assembly=AsyncImageLoader.Avalonia" x:DataType="vm:AboutTabViewModel" mc:Ignorable="d" d:DesignWidth="1000" d:DesignHeight="1400" x:Class="Artemis.UI.Screens.Settings.AboutTabView"> - + RenderOptions.BitmapInterpolationMode="HighQuality" /> Artemis 2 @@ -52,17 +53,9 @@ - + - + @@ -81,17 +74,9 @@ - + - + @@ -110,17 +95,9 @@ - + - + @@ -139,17 +116,9 @@ - + - + diff --git a/src/Artemis.UI/Screens/Settings/Tabs/AboutTabViewModel.cs b/src/Artemis.UI/Screens/Settings/Tabs/AboutTabViewModel.cs index 8fa53983e..984187e6f 100644 --- a/src/Artemis.UI/Screens/Settings/Tabs/AboutTabViewModel.cs +++ b/src/Artemis.UI/Screens/Settings/Tabs/AboutTabViewModel.cs @@ -1,75 +1,15 @@ -using System; -using System.Reactive.Disposables; -using System.Reflection; -using System.Threading.Tasks; -using Artemis.Core; -using Artemis.UI.Shared; -using Avalonia.Media.Imaging; -using Flurl.Http; -using ReactiveUI; +using Artemis.Core; +using Artemis.UI.Shared.Routing; namespace Artemis.UI.Screens.Settings; -public class AboutTabViewModel : ActivatableViewModelBase +public class AboutTabViewModel : RoutableScreen { - private Bitmap? _darthAffeProfileImage; - private Bitmap? _drMeteorProfileImage; - private Bitmap? _kaiProfileImage; - private Bitmap? _robertProfileImage; - private string? _version; - public AboutTabViewModel() { DisplayName = "About"; - this.WhenActivated((CompositeDisposable _) => Task.Run(Activate)); - } - - public string? Version - { - get => _version; - set => RaiseAndSetIfChanged(ref _version, value); - } - - public Bitmap? RobertProfileImage - { - get => _robertProfileImage; - set => RaiseAndSetIfChanged(ref _robertProfileImage, value); - } - - public Bitmap? DarthAffeProfileImage - { - get => _darthAffeProfileImage; - set => RaiseAndSetIfChanged(ref _darthAffeProfileImage, value); - } - - public Bitmap? DrMeteorProfileImage - { - get => _drMeteorProfileImage; - set => RaiseAndSetIfChanged(ref _drMeteorProfileImage, value); - } - - public Bitmap? KaiProfileImage - { - get => _kaiProfileImage; - set => RaiseAndSetIfChanged(ref _kaiProfileImage, value); - } - - private async Task Activate() - { - AssemblyInformationalVersionAttribute? versionAttribute = typeof(AboutTabViewModel).Assembly.GetCustomAttribute(); Version = $"Version {Constants.CurrentVersion}"; - - try - { - RobertProfileImage = new Bitmap(await "https://avatars.githubusercontent.com/u/8858506".GetStreamAsync()); - RobertProfileImage = new Bitmap(await "https://avatars.githubusercontent.com/u/8858506".GetStreamAsync()); - DarthAffeProfileImage = new Bitmap(await "https://avatars.githubusercontent.com/u/1094841".GetStreamAsync()); - DrMeteorProfileImage = new Bitmap(await "https://avatars.githubusercontent.com/u/29486064".GetStreamAsync()); - KaiProfileImage = new Bitmap(await "https://i.imgur.com/8mPWY1j.png".GetStreamAsync()); - } - catch (Exception) - { - // ignored, unluckyyyy - } } + + public string Version { get; } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Settings/Tabs/DevicesTabViewModel.cs b/src/Artemis.UI/Screens/Settings/Tabs/DevicesTabViewModel.cs index 620da173e..e6c1366c1 100644 --- a/src/Artemis.UI/Screens/Settings/Tabs/DevicesTabViewModel.cs +++ b/src/Artemis.UI/Screens/Settings/Tabs/DevicesTabViewModel.cs @@ -8,7 +8,7 @@ using Artemis.Core; using Artemis.Core.Services; using Artemis.UI.DryIoc.Factories; using Artemis.UI.Screens.Device; -using Artemis.UI.Shared; +using Artemis.UI.Shared.Routing; using Artemis.UI.Shared.Services; using Avalonia.Threading; using DynamicData; @@ -16,7 +16,7 @@ using ReactiveUI; namespace Artemis.UI.Screens.Settings; -public class DevicesTabViewModel : ActivatableViewModelBase +public class DevicesTabViewModel : RoutableScreen { private readonly IDeviceVmFactory _deviceVmFactory; private readonly IRgbService _rgbService; diff --git a/src/Artemis.UI/Screens/Settings/Tabs/GeneralTabViewModel.cs b/src/Artemis.UI/Screens/Settings/Tabs/GeneralTabViewModel.cs index 110c948e8..74adce108 100644 --- a/src/Artemis.UI/Screens/Settings/Tabs/GeneralTabViewModel.cs +++ b/src/Artemis.UI/Screens/Settings/Tabs/GeneralTabViewModel.cs @@ -13,8 +13,8 @@ using Artemis.Core.Services; using Artemis.UI.Screens.StartupWizard; using Artemis.UI.Services.Interfaces; using Artemis.UI.Services.Updating; -using Artemis.UI.Shared; using Artemis.UI.Shared.Providers; +using Artemis.UI.Shared.Routing; using Artemis.UI.Shared.Services; using Artemis.UI.Shared.Services.Builders; using Avalonia.Threading; @@ -26,7 +26,7 @@ using Serilog.Events; namespace Artemis.UI.Screens.Settings; -public class GeneralTabViewModel : ActivatableViewModelBase +public class GeneralTabViewModel : RoutableScreen { private readonly IAutoRunProvider? _autoRunProvider; private readonly IDebugService _debugService; diff --git a/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabView.axaml b/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabView.axaml index 5e1b9c246..b9f28f4a6 100644 --- a/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabView.axaml +++ b/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabView.axaml @@ -25,14 +25,19 @@ - - + + + + + + + - + - - - + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabViewModel.cs b/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabViewModel.cs index 217ca1383..6113a20b4 100644 --- a/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabViewModel.cs +++ b/src/Artemis.UI/Screens/Settings/Tabs/PluginsTabViewModel.cs @@ -9,18 +9,17 @@ using Artemis.Core; using Artemis.Core.Services; using Artemis.UI.DryIoc.Factories; using Artemis.UI.Screens.Plugins; -using Artemis.UI.Shared; +using Artemis.UI.Shared.Routing; using Artemis.UI.Shared.Services; using Artemis.UI.Shared.Services.Builders; using Avalonia.ReactiveUI; -using Avalonia.Threading; using DynamicData; using DynamicData.Binding; using ReactiveUI; namespace Artemis.UI.Screens.Settings; -public class PluginsTabViewModel : ActivatableViewModelBase +public class PluginsTabViewModel : RoutableScreen { private readonly INotificationService _notificationService; private readonly IPluginManagementService _pluginManagementService; diff --git a/src/Artemis.UI/Screens/Settings/Tabs/ReleasesTabViewModel.cs b/src/Artemis.UI/Screens/Settings/Tabs/ReleasesTabViewModel.cs index fe6a1a532..d316b62a8 100644 --- a/src/Artemis.UI/Screens/Settings/Tabs/ReleasesTabViewModel.cs +++ b/src/Artemis.UI/Screens/Settings/Tabs/ReleasesTabViewModel.cs @@ -22,7 +22,7 @@ using StrawberryShake; namespace Artemis.UI.Screens.Settings; -public class ReleasesTabViewModel : RoutableScreen +public class ReleasesTabViewModel : RoutableHostScreen { private readonly ILogger _logger; private readonly IUpdateService _updateService; diff --git a/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsView.axaml b/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsView.axaml index 61fe4a5c1..ef4b22704 100644 --- a/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsView.axaml +++ b/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsView.axaml @@ -3,22 +3,23 @@ xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" xmlns:updating="clr-namespace:Artemis.UI.Screens.Settings.Updating" - xmlns:avalonia="clr-namespace:Markdown.Avalonia;assembly=Markdown.Avalonia" - xmlns:avalonia1="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" - xmlns:converters="clr-namespace:Artemis.UI.Shared.Converters;assembly=Artemis.UI.Shared" - xmlns:converters1="clr-namespace:Artemis.UI.Converters" + xmlns:shared="clr-namespace:Artemis.UI.Shared.Converters;assembly=Artemis.UI.Shared" + xmlns:converters="clr-namespace:Artemis.UI.Converters" + xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" + xmlns:mdxaml="https://github.com/whistyun/Markdown.Avalonia.Tight" mc:Ignorable="d" d:DesignWidth="1000" d:DesignHeight="1400" x:Class="Artemis.UI.Screens.Settings.Updating.ReleaseDetailsView" x:DataType="updating:ReleaseDetailsViewModel"> - - + + + - @@ -103,16 +104,16 @@ - + Release date + Text="{CompiledBinding Release.CreatedAt, Converter={StaticResource DateTimeConverter}, FallbackValue=Loading...}" /> - + Source - + File size Release notes - - + + - - + + diff --git a/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsViewModel.cs b/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsViewModel.cs index 2ea6be462..2d9a1a413 100644 --- a/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsViewModel.cs +++ b/src/Artemis.UI/Screens/Settings/Updating/ReleaseDetailsViewModel.cs @@ -18,7 +18,7 @@ using StrawberryShake; namespace Artemis.UI.Screens.Settings.Updating; -public class ReleaseDetailsViewModel : RoutableScreen +public class ReleaseDetailsViewModel : RoutableScreen { private readonly ObservableAsPropertyHelper _fileSize; private readonly ILogger _logger; diff --git a/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml b/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml index 97e58bd65..47a02b961 100644 --- a/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml +++ b/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml @@ -4,13 +4,17 @@ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" xmlns:updating="clr-namespace:Artemis.UI.Screens.Settings.Updating" xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" + xmlns:converters="clr-namespace:Artemis.UI.Converters" mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450" x:Class="Artemis.UI.Screens.Settings.Updating.ReleaseView" x:DataType="updating:ReleaseViewModel"> + + + - + - + diff --git a/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml.cs b/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml.cs index 29e385e8a..02cd2454c 100644 --- a/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml.cs +++ b/src/Artemis.UI/Screens/Settings/Updating/ReleaseView.axaml.cs @@ -1,6 +1,4 @@ -using Avalonia; using Avalonia.Controls; -using Avalonia.Markup.Xaml; namespace Artemis.UI.Screens.Settings.Updating; @@ -10,9 +8,4 @@ public partial class ReleaseView : UserControl { InitializeComponent(); } - - private void InitializeComponent() - { - AvaloniaXamlLoader.Load(this); - } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Sidebar/Dialogs/ProfileConfigurationEditViewModel.cs b/src/Artemis.UI/Screens/Sidebar/Dialogs/ProfileConfigurationEditViewModel.cs index 0e5e5e8d4..032f93cbe 100644 --- a/src/Artemis.UI/Screens/Sidebar/Dialogs/ProfileConfigurationEditViewModel.cs +++ b/src/Artemis.UI/Screens/Sidebar/Dialogs/ProfileConfigurationEditViewModel.cs @@ -9,6 +9,7 @@ using Artemis.Core; using Artemis.Core.Modules; using Artemis.Core.Services; using Artemis.UI.DryIoc.Factories; +using Artemis.UI.Extensions; using Artemis.UI.Screens.VisualScripting; using Artemis.UI.Shared; using Artemis.UI.Shared.Services; @@ -52,9 +53,9 @@ public class ProfileConfigurationEditViewModel : DialogViewModelBase().First().ToString()) + + _profileConfiguration = profileConfiguration == ProfileConfiguration.Empty + ? profileService.CreateProfileConfiguration(profileCategory, "New profile", Enum.GetValues().First().ToString()) : profileConfiguration; _profileName = _profileConfiguration.Name; _iconType = _profileConfiguration.Icon.IconType; @@ -140,7 +141,7 @@ public class ProfileConfigurationEditViewModel : DialogViewModelBase Confirm { get; } public ReactiveCommand Delete { get; } public ReactiveCommand Cancel { get; } - + private async Task ExecuteDelete() { if (IsNew) @@ -148,7 +149,7 @@ public class ProfileConfigurationEditViewModel : DialogViewModelBase _iconType; set => RaiseAndSetIfChanged(ref _iconType, value); } - + public ProfileIconViewModel? SelectedMaterialIcon { get => _selectedMaterialIcon; @@ -244,7 +245,7 @@ public class ProfileConfigurationEditViewModel : DialogViewModelBase? _isCollapsed; private ObservableAsPropertyHelper? _isSuspended; private SidebarProfileConfigurationViewModel? _selectedProfileConfiguration; @@ -69,9 +69,9 @@ public class SidebarCategoryViewModel : ActivatableViewModelBase .WhereNotNull() .Subscribe(s => _router.Navigate($"profile-editor/{s.ProfileConfiguration.ProfileId}", new RouterNavigationOptions {IgnoreOnPartialMatch = true, RecycleScreens = false})) .DisposeWith(d); - + _router.CurrentPath.WhereNotNull().Subscribe(r => SelectedProfileConfiguration = ProfileConfigurations.FirstOrDefault(c => c.Matches(r))).DisposeWith(d); - + // Update the list of profiles whenever the category fires events Observable.FromEventPattern(x => profileCategory.ProfileConfigurationAdded += x, x => profileCategory.ProfileConfigurationAdded -= x) .Subscribe(e => profileConfigurations.Add(e.EventArgs.ProfileConfiguration)) @@ -89,12 +89,6 @@ public class SidebarCategoryViewModel : ActivatableViewModelBase _isCollapsed = ProfileCategory.WhenAnyValue(vm => vm.IsCollapsed).ToProperty(this, vm => vm.IsCollapsed).DisposeWith(d); _isSuspended = ProfileCategory.WhenAnyValue(vm => vm.IsSuspended).ToProperty(this, vm => vm.IsSuspended).DisposeWith(d); }); - - profileConfigurations.Edit(updater => - { - foreach (ProfileConfiguration profileConfiguration in profileCategory.ProfileConfigurations) - updater.Add(profileConfiguration); - }); } public ReactiveCommand ImportProfile { get; } @@ -144,7 +138,7 @@ public class SidebarCategoryViewModel : ActivatableViewModelBase { if (await _windowService.ShowConfirmContentDialog($"Delete {ProfileCategory.Name}", "Do you want to delete this category and all its profiles?")) { - if (ProfileCategory.ProfileConfigurations.Any(c => c.IsBeingEdited)) + if (ProfileCategory.ProfileConfigurations.Any(c => _profileService.FocusProfile == c)) await _router.Navigate("home"); _profileService.DeleteProfileCategory(ProfileCategory); } @@ -161,7 +155,7 @@ public class SidebarCategoryViewModel : ActivatableViewModelBase } private async Task ExecuteImportProfile() - { + { string[]? result = await _windowService.CreateOpenFileDialog() .HavingFilter(f => f.WithExtension("zip").WithExtension("json").WithName("Artemis profile")) .ShowAsync(); diff --git a/src/Artemis.UI/Screens/Sidebar/SidebarProfileConfigurationViewModel.cs b/src/Artemis.UI/Screens/Sidebar/SidebarProfileConfigurationViewModel.cs index 2d7a7712e..bfc179095 100644 --- a/src/Artemis.UI/Screens/Sidebar/SidebarProfileConfigurationViewModel.cs +++ b/src/Artemis.UI/Screens/Sidebar/SidebarProfileConfigurationViewModel.cs @@ -104,7 +104,7 @@ public class SidebarProfileConfigurationViewModel : ActivatableViewModelBase if (!await _windowService.ShowConfirmContentDialog("Delete profile", "Are you sure you want to permanently delete this profile?")) return; - if (ProfileConfiguration.IsBeingEdited) + if (_profileService.FocusProfile == ProfileConfiguration) await _router.Navigate("home"); _profileService.RemoveProfileConfiguration(ProfileConfiguration); } diff --git a/src/Artemis.UI/Screens/Sidebar/SidebarViewModel.cs b/src/Artemis.UI/Screens/Sidebar/SidebarViewModel.cs index 0e8f7141d..f49cc0ec3 100644 --- a/src/Artemis.UI/Screens/Sidebar/SidebarViewModel.cs +++ b/src/Artemis.UI/Screens/Sidebar/SidebarViewModel.cs @@ -38,13 +38,15 @@ public class SidebarViewModel : ActivatableViewModelBase SidebarScreen = new SidebarScreenViewModel(MaterialIconKind.Abacus, ROOT_SCREEN, "", null, new ObservableCollection() { new(MaterialIconKind.HomeOutline, "Home", "home"), - #if DEBUG new(MaterialIconKind.TestTube, "Workshop", "workshop", null, new ObservableCollection { - new(MaterialIconKind.FolderVideo, "Profiles", "workshop/profiles/1", "workshop/profiles"), - new(MaterialIconKind.KeyboardVariant, "Layouts", "workshop/layouts/1", "workshop/layouts"), + new(MaterialIconKind.FolderVideo, "Profiles", "workshop/entries/profiles/1", "workshop/entries/profiles"), +#if DEBUG + new(MaterialIconKind.KeyboardVariant, "Layouts", "workshop/entries/layouts/1", "workshop/entries/layouts"), +#endif + new(MaterialIconKind.Bookshelf, "Library", "workshop/library"), }), - #endif + new(MaterialIconKind.Devices, "Surface Editor", "surface-editor"), new(MaterialIconKind.SettingsOutline, "Settings", "settings") }); @@ -119,7 +121,7 @@ public class SidebarViewModel : ActivatableViewModelBase { if (_updating) return; - + Dispatcher.UIThread.Invoke(async () => { try diff --git a/src/Artemis.UI/Screens/SurfaceEditor/SurfaceEditorViewModel.cs b/src/Artemis.UI/Screens/SurfaceEditor/SurfaceEditorViewModel.cs index a6e28ea39..fadaaaf0e 100644 --- a/src/Artemis.UI/Screens/SurfaceEditor/SurfaceEditorViewModel.cs +++ b/src/Artemis.UI/Screens/SurfaceEditor/SurfaceEditorViewModel.cs @@ -10,6 +10,7 @@ using Artemis.Core.Services; using Artemis.UI.DryIoc.Factories; using Artemis.UI.Extensions; using Artemis.UI.Shared; +using Artemis.UI.Shared.Routing; using Artemis.UI.Shared.Services; using Avalonia; using ReactiveUI; @@ -17,7 +18,7 @@ using SkiaSharp; namespace Artemis.UI.Screens.SurfaceEditor; -public class SurfaceEditorViewModel : ActivatableViewModelBase, IMainScreenViewModel +public class SurfaceEditorViewModel : RoutableScreen, IMainScreenViewModel { private readonly IDeviceService _deviceService; private readonly IDeviceVmFactory _deviceVmFactory; diff --git a/src/Artemis.UI/Screens/VisualScripting/CableView.axaml b/src/Artemis.UI/Screens/VisualScripting/CableView.axaml index a54997cc1..31fedf0fa 100644 --- a/src/Artemis.UI/Screens/VisualScripting/CableView.axaml +++ b/src/Artemis.UI/Screens/VisualScripting/CableView.axaml @@ -41,7 +41,8 @@ Text="{CompiledBinding Converter={StaticResource SKColorToStringConverter}, Mode=OneWay}" VerticalAlignment="Center" HorizontalAlignment="Stretch" - FontFamily="Consolas" /> + FontFamily="{StaticResource RobotoMono}" + FontSize="13"/> - + - + - + - + diff --git a/src/Artemis.UI/Screens/Workshop/Categories/CategoriesView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Categories/CategoriesView.axaml.cs index 15ac17ab5..8cb505ab9 100644 --- a/src/Artemis.UI/Screens/Workshop/Categories/CategoriesView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/Categories/CategoriesView.axaml.cs @@ -1,6 +1,5 @@ using Avalonia; using Avalonia.Input; -using Avalonia.Markup.Xaml; using Avalonia.ReactiveUI; namespace Artemis.UI.Screens.Workshop.Categories; @@ -12,10 +11,6 @@ public partial class CategoriesView : ReactiveUserControl InitializeComponent(); } - private void InitializeComponent() - { - AvaloniaXamlLoader.Load(this); - } private void InputElement_OnPointerReleased(object? sender, PointerReleasedEventArgs e) { diff --git a/src/Artemis.UI/Screens/Workshop/Categories/CategoriesViewModel.cs b/src/Artemis.UI/Screens/Workshop/Categories/CategoriesViewModel.cs index a72737141..74d0f629e 100644 --- a/src/Artemis.UI/Screens/Workshop/Categories/CategoriesViewModel.cs +++ b/src/Artemis.UI/Screens/Workshop/Categories/CategoriesViewModel.cs @@ -46,16 +46,16 @@ public class CategoriesViewModel : ActivatableViewModelBase private IReadOnlyList? CreateFilter() { - List categories = Categories.Where(c => c.IsSelected).Select(c => (int?) c.Id).ToList(); + List categories = Categories.Where(c => c.IsSelected).Select(c => (long?) c.Id).ToList(); if (!categories.Any()) return null; List categoryFilters = new(); - foreach (int? category in categories) + foreach (long? category in categories) { categoryFilters.Add(new EntryFilterInput { - Categories = new ListFilterInputTypeOfCategoryFilterInput {Some = new CategoryFilterInput {Id = new IntOperationFilterInput {Eq = category}}} + Categories = new ListFilterInputTypeOfCategoryFilterInput {Some = new CategoryFilterInput {Id = new LongOperationFilterInput {Eq = category}}} }); } diff --git a/src/Artemis.UI/Screens/Workshop/Categories/CategoryViewModel.cs b/src/Artemis.UI/Screens/Workshop/Categories/CategoryViewModel.cs index 08d991009..328b43c2a 100644 --- a/src/Artemis.UI/Screens/Workshop/Categories/CategoryViewModel.cs +++ b/src/Artemis.UI/Screens/Workshop/Categories/CategoryViewModel.cs @@ -17,7 +17,7 @@ public class CategoryViewModel : ViewModelBase Icon = icon as MaterialIconKind? ?? MaterialIconKind.QuestionMarkCircle; } - public int Id { get; } + public long Id { get; } public string Name { get; } public MaterialIconKind Icon { get; } diff --git a/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserView.axaml b/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserView.axaml index 2b28230f7..0c14c271f 100644 --- a/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserView.axaml +++ b/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserView.axaml @@ -9,9 +9,9 @@ x:Class="Artemis.UI.Screens.Workshop.CurrentUser.CurrentUserView" x:DataType="currentUser:CurrentUserViewModel"> - + - + @@ -27,7 +27,7 @@ - + @@ -39,6 +39,7 @@ - - \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserViewModel.cs b/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserViewModel.cs index c204c3165..5d4fc9b80 100644 --- a/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserViewModel.cs +++ b/src/Artemis.UI/Screens/Workshop/CurrentUser/CurrentUserViewModel.cs @@ -5,9 +5,11 @@ using System.Reactive.Disposables; using System.Threading; using System.Threading.Tasks; using Artemis.UI.Shared; +using Artemis.UI.Shared.Services; using Artemis.WebClient.Workshop; using Artemis.WebClient.Workshop.Services; using Avalonia.Media.Imaging; +using FluentAvalonia.UI.Controls; using Flurl.Http; using ReactiveUI; using Serilog; @@ -16,67 +18,91 @@ namespace Artemis.UI.Screens.Workshop.CurrentUser; public class CurrentUserViewModel : ActivatableViewModelBase { - private readonly ILogger _logger; private readonly IAuthenticationService _authenticationService; - private bool _loading = true; + private readonly ObservableAsPropertyHelper _isAnonymous; + private readonly ILogger _logger; + private readonly IWindowService _windowService; + private bool _allowLogout = true; private Bitmap? _avatar; private string? _email; + private bool _loading = true; private string? _name; private string? _userId; - public CurrentUserViewModel(ILogger logger, IAuthenticationService authenticationService) + public CurrentUserViewModel(ILogger logger, IAuthenticationService authenticationService, IWindowService windowService) { _logger = logger; _authenticationService = authenticationService; + _windowService = windowService; Login = ReactiveCommand.CreateFromTask(ExecuteLogin); - this.WhenActivated(d => ReactiveCommand.CreateFromTask(ExecuteAutoLogin).Execute().Subscribe().DisposeWith(d)); + _isAnonymous = this.WhenAnyValue(vm => vm.Loading, vm => vm.Name, (l, n) => l || n == null).ToProperty(this, vm => vm.IsAnonymous); + + this.WhenActivated(d => + { + Task.Run(AutoLogin); + _authenticationService.IsLoggedIn.Subscribe(_ => Task.Run(LoadCurrentUser)).DisposeWith(d); + }); + } + + public bool IsAnonymous => _isAnonymous.Value; + + public bool AllowLogout + { + get => _allowLogout; + set => RaiseAndSetIfChanged(ref _allowLogout, value); } public bool Loading { get => _loading; - set => RaiseAndSetIfChanged(ref _loading, value); + private set => RaiseAndSetIfChanged(ref _loading, value); } public string? UserId { get => _userId; - set => RaiseAndSetIfChanged(ref _userId, value); + private set => RaiseAndSetIfChanged(ref _userId, value); } public string? Name { get => _name; - set => RaiseAndSetIfChanged(ref _name, value); + private set => RaiseAndSetIfChanged(ref _name, value); } public string? Email { get => _email; - set => RaiseAndSetIfChanged(ref _email, value); + private set => RaiseAndSetIfChanged(ref _email, value); } public Bitmap? Avatar { get => _avatar; - set => RaiseAndSetIfChanged(ref _avatar, value); + private set => RaiseAndSetIfChanged(ref _avatar, value); } public ReactiveCommand Login { get; } public void Logout() { - _authenticationService.Logout(); + if (AllowLogout) + _authenticationService.Logout(); } private async Task ExecuteLogin(CancellationToken cancellationToken) { - await _authenticationService.Login(); - await LoadCurrentUser(); + ContentDialogResult result = await _windowService.CreateContentDialog() + .WithViewModel(out WorkshopLoginViewModel _) + .WithTitle("Workshop login") + .ShowAsync(); + + if (result == ContentDialogResult.Primary) + await LoadCurrentUser(); } - private async Task ExecuteAutoLogin(CancellationToken cancellationToken) + private async Task AutoLogin() { try { @@ -95,21 +121,26 @@ public class CurrentUserViewModel : ActivatableViewModelBase private async Task LoadCurrentUser() { - if (!_authenticationService.IsLoggedIn) - return; - UserId = _authenticationService.Claims.FirstOrDefault(c => c.Type == "sub")?.Value; Name = _authenticationService.Claims.FirstOrDefault(c => c.Type == "name")?.Value; Email = _authenticationService.Claims.FirstOrDefault(c => c.Type == "email")?.Value; if (UserId != null) + { await LoadAvatar(UserId); + } + else + { + Avatar?.Dispose(); + Avatar = null; + } } private async Task LoadAvatar(string userId) { try { + Avatar?.Dispose(); Avatar = new Bitmap(await $"{WorkshopConstants.AUTHORITY_URL}/user/avatar/{userId}".GetStreamAsync()); } catch (Exception) diff --git a/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginView.axaml b/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginView.axaml new file mode 100644 index 000000000..538f4cccc --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginView.axaml @@ -0,0 +1,17 @@ + + + + A browser window has opened where you'll be able to create an account and/or log in. + + + Waiting for log in... + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginView.axaml.cs b/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginView.axaml.cs new file mode 100644 index 000000000..9d6092a7f --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.CurrentUser; + +public partial class WorkshopLoginView : ReactiveUserControl +{ + public WorkshopLoginView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginViewModel.cs b/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginViewModel.cs new file mode 100644 index 000000000..793fedc6d --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/CurrentUser/WorkshopLoginViewModel.cs @@ -0,0 +1,49 @@ +using System; +using System.Reactive.Disposables; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Shared; +using Artemis.UI.Shared.Services; +using Artemis.UI.Shared.Services.Builders; +using Artemis.WebClient.Workshop.Services; +using Avalonia.Threading; +using FluentAvalonia.UI.Controls; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.CurrentUser; + +public class WorkshopLoginViewModel : ContentDialogViewModelBase +{ + private readonly IAuthenticationService _authenticationService; + private readonly CancellationTokenSource _cts; + private readonly INotificationService _notificationService; + + public WorkshopLoginViewModel(IAuthenticationService authenticationService, INotificationService notificationService) + { + _authenticationService = authenticationService; + _notificationService = notificationService; + _cts = new CancellationTokenSource(); + + this.WhenActivated(d => + { + Dispatcher.UIThread.InvokeAsync(Login); + Disposable.Create(_cts, cts => cts.Cancel()).DisposeWith(d); + }); + } + + private async Task Login() + { + try + { + await _authenticationService.Login(_cts.Token); + _notificationService.CreateNotification().WithTitle("Login succeeded").WithSeverity(NotificationSeverity.Success).Show(); + ContentDialog?.Hide(ContentDialogResult.Primary); + } + catch (Exception e) + { + if (e is not TaskCanceledException) + _notificationService.CreateNotification().WithTitle("Login failed").WithMessage(e.Message).WithSeverity(NotificationSeverity.Error).Show(); + ContentDialog?.Hide(ContentDialogResult.Secondary); + } + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntriesView.axaml b/src/Artemis.UI/Screens/Workshop/Entries/EntriesView.axaml new file mode 100644 index 000000000..ecc131eb9 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntriesView.axaml @@ -0,0 +1,32 @@ + + + + + + + + + + + + + + + diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntriesView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Entries/EntriesView.axaml.cs new file mode 100644 index 000000000..3f2106645 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntriesView.axaml.cs @@ -0,0 +1,28 @@ +using System.Reactive.Disposables; +using Artemis.UI.Shared; +using Avalonia.ReactiveUI; +using Avalonia.Threading; +using FluentAvalonia.UI.Controls; +using ReactiveUI; +using System; + +namespace Artemis.UI.Screens.Workshop.Entries; + +public partial class EntriesView : ReactiveUserControl +{ + public EntriesView() + { + InitializeComponent(); + this.WhenActivated(d => { ViewModel.WhenAnyValue(vm => vm.Screen).WhereNotNull().Subscribe(Navigate).DisposeWith(d); }); + } + + private void Navigate(ViewModelBase viewModel) + { + Dispatcher.UIThread.Invoke(() => TabFrame.NavigateFromObject(viewModel)); + } + + private void NavigationView_OnBackRequested(object? sender, NavigationViewBackRequestedEventArgs e) + { + ViewModel?.GoBack(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntriesViewModel.cs b/src/Artemis.UI/Screens/Workshop/Entries/EntriesViewModel.cs new file mode 100644 index 000000000..4ebd77ecc --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntriesViewModel.cs @@ -0,0 +1,67 @@ +using System.Collections.ObjectModel; +using System.Linq; +using System.Reactive.Disposables; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Routing; +using Artemis.UI.Shared.Routing; +using ReactiveUI; +using System; +using System.Reactive.Linq; + +namespace Artemis.UI.Screens.Workshop.Entries; + +public class EntriesViewModel : RoutableHostScreen +{ + private readonly IRouter _router; + private RouteViewModel? _selectedTab; + private ObservableAsPropertyHelper? _viewingDetails; + + public EntriesViewModel(IRouter router) + { + _router = router; + + Tabs = new ObservableCollection + { + new("Profiles", "workshop/entries/profiles/1", "workshop/entries/profiles"), +#if DEBUG + new("Layouts", "workshop/entries/layouts/1", "workshop/entries/layouts") +#endif + }; + + this.WhenActivated(d => + { + // Show back button on details page + _viewingDetails = _router.CurrentPath.Select(p => p != null && p.Contains("details")).ToProperty(this, vm => vm.ViewingDetails).DisposeWith(d); + // Navigate on tab change + this.WhenAnyValue(vm => vm.SelectedTab) + .WhereNotNull() + .Subscribe(s => router.Navigate(s.Path, new RouterNavigationOptions {IgnoreOnPartialMatch = true, PartialMatchOverride = s.MatchPath})) + .DisposeWith(d); + }); + } + + public bool ViewingDetails => _viewingDetails?.Value ?? false; + public ObservableCollection Tabs { get; } + + public RouteViewModel? SelectedTab + { + get => _selectedTab; + set => RaiseAndSetIfChanged(ref _selectedTab, value); + } + + public override async Task OnNavigating(NavigationArguments args, CancellationToken cancellationToken) + { + SelectedTab = Tabs.FirstOrDefault(t => t.Matches(args.Path)); + if (SelectedTab == null) + await args.Router.Navigate(Tabs.First().Path); + } + + public void GoBack() + { + if (ViewingDetails) + _router.GoBack(); + else + _router.Navigate("workshop"); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogView.axaml b/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogView.axaml new file mode 100644 index 000000000..af75201c2 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogView.axaml @@ -0,0 +1,8 @@ + + Welcome to Avalonia! + diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogView.axaml.cs new file mode 100644 index 000000000..b228f4b0f --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogView.axaml.cs @@ -0,0 +1,13 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Markup.Xaml; + +namespace Artemis.UI.Screens.Workshop.Entries; + +public partial class EntryInstallationDialogView : UserControl +{ + public EntryInstallationDialogView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogViewModel.cs b/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogViewModel.cs new file mode 100644 index 000000000..7b7cbca67 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntryInstallationDialogViewModel.cs @@ -0,0 +1,8 @@ +using Artemis.UI.Shared; + +namespace Artemis.UI.Screens.Workshop.Entries; + +public class EntryInstallationDialogViewModel : ContentDialogViewModelBase +{ + +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntryListView.axaml b/src/Artemis.UI/Screens/Workshop/Entries/EntryListItemView.axaml similarity index 67% rename from src/Artemis.UI/Screens/Workshop/Entries/EntryListView.axaml rename to src/Artemis.UI/Screens/Workshop/Entries/EntryListItemView.axaml index 9af92602f..2831515f6 100644 --- a/src/Artemis.UI/Screens/Workshop/Entries/EntryListView.axaml +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntryListItemView.axaml @@ -3,42 +3,48 @@ xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" - xmlns:entries="clr-namespace:Artemis.UI.Screens.Workshop.Profile" xmlns:entries1="clr-namespace:Artemis.UI.Screens.Workshop.Entries" + xmlns:il="clr-namespace:AsyncImageLoader;assembly=AsyncImageLoader.Avalonia" + xmlns:converters="clr-namespace:Artemis.UI.Converters" mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="110" - x:Class="Artemis.UI.Screens.Workshop.Entries.EntryListView" - x:DataType="entries1:EntryListViewModel"> - + + + + + + + + + Icon required + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + At least one category is required + + + + + + + + + + + Synchronized scrolling + + + + + + + + + + + + + + + + + + + + + A description is required + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntrySpecificationsView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Entries/EntrySpecificationsView.axaml.cs new file mode 100644 index 000000000..b3289a3bf --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntrySpecificationsView.axaml.cs @@ -0,0 +1,100 @@ +using System.Linq; +using Artemis.UI.Shared.Extensions; +using Avalonia; +using Avalonia.Controls; +using Avalonia.Media; +using Avalonia.Media.Immutable; +using Avalonia.ReactiveUI; +using AvaloniaEdit.TextMate; +using ReactiveUI; +using TextMateSharp.Grammars; + +namespace Artemis.UI.Screens.Workshop.Entries; + +public partial class EntrySpecificationsView : ReactiveUserControl +{ + private ScrollViewer? _editorScrollViewer; + private ScrollViewer? _previewScrollViewer; + private bool _updating; + + public EntrySpecificationsView() + { + InitializeComponent(); + + DescriptionEditor.Options.AllowScrollBelowDocument = false; + RegistryOptions options = new(ThemeName.Dark); + TextMate.Installation? install = DescriptionEditor.InstallTextMate(options); + + install.SetGrammar(options.GetScopeByExtension(".md")); + + this.WhenActivated(_ => SetupScrollSync()); + } + + protected override void OnAttachedToVisualTree(VisualTreeAttachmentEventArgs e) + { + if (this.TryFindResource("SystemAccentColorLight3", out object? resource) && resource is Color color) + DescriptionEditor.TextArea.TextView.LinkTextForegroundBrush = new ImmutableSolidColorBrush(color); + + base.OnAttachedToVisualTree(e); + } + + private void SetupScrollSync() + { + if (_editorScrollViewer != null) + _editorScrollViewer.PropertyChanged -= EditorScrollViewerOnPropertyChanged; + if (_previewScrollViewer != null) + _previewScrollViewer.PropertyChanged -= PreviewScrollViewerOnPropertyChanged; + + _editorScrollViewer = DescriptionEditor.GetVisualChildrenOfType().FirstOrDefault(); + _previewScrollViewer = DescriptionPreview.GetVisualChildrenOfType().FirstOrDefault(); + + if (_editorScrollViewer != null) + _editorScrollViewer.PropertyChanged += EditorScrollViewerOnPropertyChanged; + if (_previewScrollViewer != null) + _previewScrollViewer.PropertyChanged += PreviewScrollViewerOnPropertyChanged; + } + + private void EditorScrollViewerOnPropertyChanged(object? sender, AvaloniaPropertyChangedEventArgs e) + { + if (e.Property.Name != nameof(ScrollViewer.Offset) || _updating || SynchronizedScrolling.IsChecked != true) + return; + + try + { + _updating = true; + SynchronizeScrollViewers(_editorScrollViewer, _previewScrollViewer); + } + finally + { + _updating = false; + } + } + + private void PreviewScrollViewerOnPropertyChanged(object? sender, AvaloniaPropertyChangedEventArgs e) + { + if (e.Property.Name != nameof(ScrollViewer.Offset) || _updating || SynchronizedScrolling.IsChecked != true) + return; + + try + { + _updating = true; + SynchronizeScrollViewers(_previewScrollViewer, _editorScrollViewer); + } + finally + { + _updating = false; + } + } + + private void SynchronizeScrollViewers(ScrollViewer? source, ScrollViewer? target) + { + if (source == null || target == null) + return; + + double sourceScrollableHeight = source.Extent.Height - source.Viewport.Height; + double targetScrollableHeight = target.Extent.Height - target.Viewport.Height; + + if (sourceScrollableHeight != 0) + target.Offset = new Vector(target.Offset.X, targetScrollableHeight * (source.Offset.Y / sourceScrollableHeight)); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/EntrySpecificationsViewModel.cs b/src/Artemis.UI/Screens/Workshop/Entries/EntrySpecificationsViewModel.cs new file mode 100644 index 000000000..cc9d289eb --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/EntrySpecificationsViewModel.cs @@ -0,0 +1,164 @@ +using System; +using System.Collections.Generic; +using System.Collections.ObjectModel; +using System.Linq; +using System.Reactive; +using System.Reactive.Disposables; +using System.Reactive.Linq; +using System.Threading.Tasks; +using Artemis.UI.Extensions; +using Artemis.UI.Screens.Workshop.Categories; +using Artemis.UI.Shared; +using Artemis.UI.Shared.Services; +using Artemis.WebClient.Workshop; +using Avalonia.Media.Imaging; +using AvaloniaEdit.Document; +using DynamicData; +using DynamicData.Aggregation; +using DynamicData.Binding; +using ReactiveUI; +using ReactiveUI.Validation.Extensions; +using ReactiveUI.Validation.Helpers; +using StrawberryShake; + +namespace Artemis.UI.Screens.Workshop.Entries; + +public class EntrySpecificationsViewModel : ValidatableViewModelBase +{ + private readonly ObservableAsPropertyHelper _categoriesValid; + private readonly ObservableAsPropertyHelper _iconValid; + private readonly ObservableAsPropertyHelper _descriptionValid; + private readonly IWorkshopClient _workshopClient; + private readonly IWindowService _windowService; + + private string _description = string.Empty; + private Bitmap? _iconBitmap; + private TextDocument? _markdownDocument; + private string _name = string.Empty; + private string _summary = string.Empty; + private bool _iconChanged; + + public EntrySpecificationsViewModel(IWorkshopClient workshopClient, IWindowService windowService) + { + _workshopClient = workshopClient; + _windowService = windowService; + SelectIcon = ReactiveCommand.CreateFromTask(ExecuteSelectIcon); + + Categories.ToObservableChangeSet() + .AutoRefresh(c => c.IsSelected) + .Filter(c => c.IsSelected) + .Transform(c => c.Id) + .Bind(out ReadOnlyObservableCollection selectedCategories) + .Subscribe(); + SelectedCategories = selectedCategories; + + this.ValidationRule(vm => vm.Name, s => !string.IsNullOrWhiteSpace(s), "Name is required"); + this.ValidationRule(vm => vm.Summary, s => !string.IsNullOrWhiteSpace(s), "Summary is required"); + ValidationHelper descriptionRule = this.ValidationRule(vm => vm.Description, s => !string.IsNullOrWhiteSpace(s), "Description is required"); + + // These don't use inputs that support validation messages, do so manually + ValidationHelper iconRule = this.ValidationRule(vm => vm.IconBitmap, s => s != null, "Icon required"); + ValidationHelper categoriesRule = this.ValidationRule(vm => vm.Categories, Categories.ToObservableChangeSet().AutoRefresh(c => c.IsSelected).Filter(c => c.IsSelected).IsNotEmpty(), + "At least one category must be selected" + ); + _iconValid = iconRule.ValidationChanged.Select(c => c.IsValid).ToProperty(this, vm => vm.IconValid); + _categoriesValid = categoriesRule.ValidationChanged.Select(c => c.IsValid).ToProperty(this, vm => vm.CategoriesValid); + _descriptionValid = descriptionRule.ValidationChanged.Select(c => c.IsValid).ToProperty(this, vm => vm.DescriptionValid); + + this.WhenActivatedAsync(async d => + { + // Load categories + await PopulateCategories(); + + MarkdownDocument = new TextDocument(new StringTextSource(Description)); + MarkdownDocument.TextChanged += MarkdownDocumentOnTextChanged; + Disposable.Create(() => + { + MarkdownDocument.TextChanged -= MarkdownDocumentOnTextChanged; + MarkdownDocument = null; + ClearIcon(); + }).DisposeWith(d); + }); + } + + public ReactiveCommand SelectIcon { get; } + + public ObservableCollection Categories { get; } = new(); + public ObservableCollection Tags { get; } = new(); + public ReadOnlyObservableCollection SelectedCategories { get; } + + public bool CategoriesValid => _categoriesValid.Value ; + public bool IconValid => _iconValid.Value; + public bool DescriptionValid => _descriptionValid.Value; + + public string Name + { + get => _name; + set => RaiseAndSetIfChanged(ref _name, value); + } + + public string Summary + { + get => _summary; + set => RaiseAndSetIfChanged(ref _summary, value); + } + + public string Description + { + get => _description; + set => RaiseAndSetIfChanged(ref _description, value); + } + + public Bitmap? IconBitmap + { + get => _iconBitmap; + set => RaiseAndSetIfChanged(ref _iconBitmap, value); + } + + public TextDocument? MarkdownDocument + { + get => _markdownDocument; + set => RaiseAndSetIfChanged(ref _markdownDocument, value); + } + + public bool IconChanged + { + get => _iconChanged; + private set => RaiseAndSetIfChanged(ref _iconChanged, value); + } + + public List PreselectedCategories { get; set; } = new(); + + private void MarkdownDocumentOnTextChanged(object? sender, EventArgs e) + { + Description = MarkdownDocument?.Text ?? string.Empty; + } + + private async Task ExecuteSelectIcon() + { + string[]? result = await _windowService.CreateOpenFileDialog() + .HavingFilter(f => f.WithExtension("png").WithExtension("jpg").WithExtension("bmp").WithName("Bitmap image")) + .ShowAsync(); + + if (result == null) + return; + + IconBitmap?.Dispose(); + IconBitmap = BitmapExtensions.LoadAndResize(result[0], 128); + IconChanged = true; + } + + private void ClearIcon() + { + IconBitmap?.Dispose(); + IconBitmap = null; + } + + private async Task PopulateCategories() + { + IOperationResult categories = await _workshopClient.GetCategories.ExecuteAsync(); + Categories.Clear(); + if (categories.Data != null) + Categories.AddRange(categories.Data.Categories.Select(c => new CategoryViewModel(c) {IsSelected = PreselectedCategories.Contains(c.Id)})); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListView.axaml b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListView.axaml new file mode 100644 index 000000000..575f24ffc --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListView.axaml @@ -0,0 +1,41 @@ + + + + + + Categories + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListView.axaml.cs new file mode 100644 index 000000000..6b574bb29 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Entries.Tabs; + +public partial class LayoutListView : ReactiveUserControl +{ + public LayoutListView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListViewModel.cs b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListViewModel.cs new file mode 100644 index 000000000..a32408d09 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/LayoutListViewModel.cs @@ -0,0 +1,43 @@ +using System; +using Artemis.UI.Screens.Workshop.Categories; +using Artemis.UI.Shared.Routing; +using Artemis.UI.Shared.Services; +using Artemis.WebClient.Workshop; + +namespace Artemis.UI.Screens.Workshop.Entries.Tabs; + +public class LayoutListViewModel : EntryListViewModel +{ + /// + public LayoutListViewModel(IWorkshopClient workshopClient, + IRouter router, + CategoriesViewModel categoriesViewModel, + INotificationService notificationService, + Func getEntryListViewModel) + : base(workshopClient, router, categoriesViewModel, notificationService, getEntryListViewModel) + { + } + + #region Overrides of EntryListBaseViewModel + + /// + protected override string GetPagePath(int page) + { + return $"workshop/entries/layouts/{page}"; + } + + /// + protected override EntryFilterInput GetFilter() + { + return new EntryFilterInput + { + And = new[] + { + new EntryFilterInput {EntryType = new EntryTypeOperationFilterInput {Eq = WebClient.Workshop.EntryType.Layout}}, + base.GetFilter() + } + }; + } + + #endregion +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListView.axaml b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListView.axaml new file mode 100644 index 000000000..e7b733741 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListView.axaml @@ -0,0 +1,42 @@ + + + + + + Categories + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListView.axaml.cs new file mode 100644 index 000000000..62adad88b --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Entries.Tabs; + +public partial class ProfileListView : ReactiveUserControl +{ + public ProfileListView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListViewModel.cs b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListViewModel.cs new file mode 100644 index 000000000..49df897db --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Entries/Tabs/ProfileListViewModel.cs @@ -0,0 +1,43 @@ +using System; +using Artemis.UI.Screens.Workshop.Categories; +using Artemis.UI.Shared.Routing; +using Artemis.UI.Shared.Services; +using Artemis.WebClient.Workshop; + +namespace Artemis.UI.Screens.Workshop.Entries.Tabs; + +public class ProfileListViewModel : EntryListViewModel +{ + /// + public ProfileListViewModel(IWorkshopClient workshopClient, + IRouter router, + CategoriesViewModel categoriesViewModel, + INotificationService notificationService, + Func getEntryListViewModel) + : base(workshopClient, router, categoriesViewModel, notificationService, getEntryListViewModel) + { + } + + #region Overrides of EntryListBaseViewModel + + /// + protected override string GetPagePath(int page) + { + return $"workshop/entries/profiles/{page}"; + } + + /// + protected override EntryFilterInput GetFilter() + { + return new EntryFilterInput + { + And = new[] + { + new EntryFilterInput {EntryType = new EntryTypeOperationFilterInput {Eq = WebClient.Workshop.EntryType.Profile}}, + base.GetFilter() + } + }; + } + + #endregion +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml b/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml index d0717860f..c97d51ee0 100644 --- a/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml +++ b/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml @@ -9,6 +9,8 @@ x:DataType="home:WorkshopHomeViewModel"> + + - + - - - - - + + + + - + Featured submissions Not yet implemented, here we'll show submissions we think are worth some extra attention. - + Recently updated Not yet implemented, here we'll a few of the most recent uploads/updates to the workshop. - \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml.cs index e28f6659b..4a143850a 100644 --- a/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/Home/WorkshopHomeView.axaml.cs @@ -1,6 +1,3 @@ -using Avalonia; -using Avalonia.Controls; -using Avalonia.Markup.Xaml; using Avalonia.ReactiveUI; namespace Artemis.UI.Screens.Workshop.Home; @@ -11,9 +8,4 @@ public partial class WorkshopHomeView : ReactiveUserControl(async r => await router.Navigate(r)); + + AddSubmission = ReactiveCommand.CreateFromTask(ExecuteAddSubmission, this.WhenAnyValue(vm => vm.WorkshopReachable)); + Navigate = ReactiveCommand.CreateFromTask(async r => await router.Navigate(r), this.WhenAnyValue(vm => vm.WorkshopReachable)); + + this.WhenActivatedAsync(async d => WorkshopReachable = await workshopService.ValidateWorkshopStatus(d.AsCancellationToken())); } +#if DEBUG + public bool ShowLayouts => true; +# else + public bool ShowLayouts => false; +#endif + public ReactiveCommand AddSubmission { get; } public ReactiveCommand Navigate { get; } + public bool WorkshopReachable + { + get => _workshopReachable; + private set => RaiseAndSetIfChanged(ref _workshopReachable, value); + } + private async Task ExecuteAddSubmission(CancellationToken arg) { await _windowService.ShowDialogAsync(); } - - public EntryType? EntryType => null; } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineView.axaml b/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineView.axaml new file mode 100644 index 000000000..b700c5375 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineView.axaml @@ -0,0 +1,36 @@ + + + + + + + + + + + + + Could not reach the workshop + + + + Please ensure you are connected to the internet. + If this keeps occuring, hit us up on Discord + + + + + + + diff --git a/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineView.axaml.cs new file mode 100644 index 000000000..a7d5eebdb --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineView.axaml.cs @@ -0,0 +1,14 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Markup.Xaml; +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Home; + +public partial class WorkshopOfflineView : ReactiveUserControl +{ + public WorkshopOfflineView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineViewModel.cs b/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineViewModel.cs new file mode 100644 index 000000000..8c72ea6bd --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Home/WorkshopOfflineViewModel.cs @@ -0,0 +1,55 @@ + +using System.Reactive; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Shared; +using Artemis.UI.Shared.Routing; +using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Services; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Home; + +public class WorkshopOfflineViewModel : RoutableScreen +{ + private readonly IRouter _router; + private readonly IWorkshopService _workshopService; + private string _message; + + /// + public WorkshopOfflineViewModel(IWorkshopService workshopService, IRouter router) + { + _workshopService = workshopService; + _router = router; + + Retry = ReactiveCommand.CreateFromTask(ExecuteRetry); + } + + public ReactiveCommand Retry { get; } + + public string Message + { + get => _message; + set => RaiseAndSetIfChanged(ref _message, value); + } + + public override Task OnNavigating(WorkshopOfflineParameters parameters, NavigationArguments args, CancellationToken cancellationToken) + { + Message = parameters.Message; + return base.OnNavigating(parameters, args, cancellationToken); + } + + private async Task ExecuteRetry(CancellationToken cancellationToken) + { + IWorkshopService.WorkshopStatus status = await _workshopService.GetWorkshopStatus(cancellationToken); + if (status.IsReachable) + await _router.Navigate("workshop"); + + Message = status.Message; + } +} + +public class WorkshopOfflineParameters +{ + public string Message { get; set; } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml b/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml index 4e48f869c..5e953bd10 100644 --- a/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml +++ b/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml @@ -2,24 +2,132 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" - xmlns:layout="clr-namespace:Artemis.UI.Screens.Workshop.Layout" - mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450" + xmlns:Layout="clr-namespace:Artemis.UI.Screens.Workshop.Layout" + xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" + xmlns:il="clr-namespace:AsyncImageLoader;assembly=AsyncImageLoader.Avalonia" + xmlns:converters="clr-namespace:Artemis.UI.Converters" + xmlns:mdxaml="https://github.com/whistyun/Markdown.Avalonia.Tight" + xmlns:mdsvg="https://github.com/whistyun/Markdown.Avalonia.Svg" + xmlns:ui="clr-namespace:Artemis.UI" + xmlns:converters1="clr-namespace:Artemis.UI.Shared.Converters;assembly=Artemis.UI.Shared" + mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="800" x:Class="Artemis.UI.Screens.Workshop.Layout.LayoutDetailsView" - x:DataType="layout:LayoutDetailsViewModel"> - - - - - - - - - Side panel + x:DataType="Layout:LayoutDetailsViewModel"> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + downloads + + + + + Created + + + + + Updated + + + - - Layout details panel + + + Latest release + + + - - + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml.cs index 3a529743e..57e93b1e1 100644 --- a/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/Layout/LayoutDetailsView.axaml.cs @@ -1,4 +1,3 @@ -using Avalonia.Markup.Xaml; using Avalonia.ReactiveUI; namespace Artemis.UI.Screens.Workshop.Layout; @@ -9,9 +8,4 @@ public partial class LayoutDetailsView : ReactiveUserControl, IWorkshopViewModel +public class LayoutDetailsViewModel : RoutableScreen { private readonly IWorkshopClient _client; + private readonly INotificationService _notificationService; + private readonly IWindowService _windowService; + private readonly ObservableAsPropertyHelper _updatedAt; private IGetEntryById_Entry? _entry; - public LayoutDetailsViewModel(IWorkshopClient client) + public LayoutDetailsViewModel(IWorkshopClient client, INotificationService notificationService, IWindowService windowService) { _client = client; + _notificationService = notificationService; + _windowService = windowService; + _updatedAt = this.WhenAnyValue(vm => vm.Entry).Select(e => e?.LatestRelease?.CreatedAt ?? e?.CreatedAt).ToProperty(this, vm => vm.UpdatedAt); + + DownloadLatestRelease = ReactiveCommand.CreateFromTask(ExecuteDownloadLatestRelease); } - public EntryType? EntryType => null; + public ReactiveCommand DownloadLatestRelease { get; } + + public DateTimeOffset? UpdatedAt => _updatedAt.Value; public IGetEntryById_Entry? Entry { get => _entry; - set => RaiseAndSetIfChanged(ref _entry, value); + private set => RaiseAndSetIfChanged(ref _entry, value); } public override async Task OnNavigating(WorkshopDetailParameters parameters, NavigationArguments args, CancellationToken cancellationToken) @@ -32,7 +48,7 @@ public class LayoutDetailsViewModel : RoutableScreen result = await _client.GetEntryById.ExecuteAsync(entryId, cancellationToken); if (result.IsErrorResult()) @@ -40,4 +56,9 @@ public class LayoutDetailsViewModel : RoutableScreen - - - - - Categories - - - - - - - - - - - - \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Layout/LayoutListViewModel.cs b/src/Artemis.UI/Screens/Workshop/Layout/LayoutListViewModel.cs deleted file mode 100644 index 17ac0ca9e..000000000 --- a/src/Artemis.UI/Screens/Workshop/Layout/LayoutListViewModel.cs +++ /dev/null @@ -1,36 +0,0 @@ -using System; -using System.Threading; -using System.Threading.Tasks; -using Artemis.UI.Screens.Workshop.Categories; -using Artemis.UI.Screens.Workshop.Parameters; -using Artemis.UI.Shared; -using Artemis.UI.Shared.Routing; -using Artemis.WebClient.Workshop; - -namespace Artemis.UI.Screens.Workshop.Layout; - -public class LayoutListViewModel : RoutableScreen, IWorkshopViewModel -{ - private int _page; - - public LayoutListViewModel(CategoriesViewModel categoriesViewModel) - { - CategoriesViewModel = categoriesViewModel; - } - - public CategoriesViewModel CategoriesViewModel { get; } - - public int Page - { - get => _page; - set => RaiseAndSetIfChanged(ref _page, value); - } - - public override Task OnNavigating(WorkshopListParameters parameters, NavigationArguments args, CancellationToken cancellationToken) - { - Page = Math.Max(1, parameters.Page); - return Task.CompletedTask; - } - - public EntryType? EntryType => WebClient.Workshop.EntryType.Layout; -} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailView.axaml b/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailView.axaml new file mode 100644 index 000000000..99365cb52 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailView.axaml @@ -0,0 +1,58 @@ + + + + + + + + + Management + + + + + + downloads + + + + + Created + + + + + + + + + + + + + View workshop page + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailView.axaml.cs new file mode 100644 index 000000000..729f02a2b --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Library; + +public partial class SubmissionDetailView : ReactiveUserControl +{ + public SubmissionDetailView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailViewModel.cs b/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailViewModel.cs new file mode 100644 index 000000000..7d94935c0 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/SubmissionDetailViewModel.cs @@ -0,0 +1,228 @@ +using System; +using System.Collections.Generic; +using System.Collections.Specialized; +using System.ComponentModel; +using System.IO; +using System.Linq; +using System.Reactive; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Screens.Workshop.Entries; +using Artemis.UI.Screens.Workshop.Parameters; +using Artemis.UI.Screens.Workshop.SubmissionWizard; +using Artemis.UI.Shared.Routing; +using Artemis.UI.Shared.Services; +using Artemis.UI.Shared.Utilities; +using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Exceptions; +using Artemis.WebClient.Workshop.Handlers.UploadHandlers; +using Artemis.WebClient.Workshop.Services; +using Avalonia.Media.Imaging; +using ReactiveUI; +using StrawberryShake; + +namespace Artemis.UI.Screens.Workshop.Library; + +public class SubmissionDetailViewModel : RoutableScreen +{ + private readonly IWorkshopClient _client; + private readonly Func _getGetSpecificationsVm; + private readonly IRouter _router; + private readonly IWindowService _windowService; + private readonly IWorkshopService _workshopService; + private IGetSubmittedEntryById_Entry? _entry; + private EntrySpecificationsViewModel? _entrySpecificationsViewModel; + private bool _hasChanges; + + public SubmissionDetailViewModel(IWorkshopClient client, IWindowService windowService, IWorkshopService workshopService, IRouter router, Func getSpecificationsVm) { + _client = client; + _windowService = windowService; + _workshopService = workshopService; + _router = router; + _getGetSpecificationsVm = getSpecificationsVm; + + CreateRelease = ReactiveCommand.CreateFromTask(ExecuteCreateRelease); + DeleteSubmission = ReactiveCommand.CreateFromTask(ExecuteDeleteSubmission); + ViewWorkshopPage = ReactiveCommand.CreateFromTask(ExecuteViewWorkshopPage); + DiscardChanges = ReactiveCommand.CreateFromTask(ExecuteDiscardChanges, this.WhenAnyValue(vm => vm.HasChanges)); + SaveChanges = ReactiveCommand.CreateFromTask(ExecuteSaveChanges, this.WhenAnyValue(vm => vm.HasChanges)); + } + + public ReactiveCommand CreateRelease { get; } + public ReactiveCommand DeleteSubmission { get; } + public ReactiveCommand ViewWorkshopPage { get; } + public ReactiveCommand SaveChanges { get; } + public ReactiveCommand DiscardChanges { get; } + + public EntrySpecificationsViewModel? EntrySpecificationsViewModel + { + get => _entrySpecificationsViewModel; + set => RaiseAndSetIfChanged(ref _entrySpecificationsViewModel, value); + } + + public IGetSubmittedEntryById_Entry? Entry + { + get => _entry; + set => RaiseAndSetIfChanged(ref _entry, value); + } + + public bool HasChanges + { + get => _hasChanges; + private set => RaiseAndSetIfChanged(ref _hasChanges, value); + } + + public override async Task OnNavigating(WorkshopDetailParameters parameters, NavigationArguments args, CancellationToken cancellationToken) + { + IOperationResult result = await _client.GetSubmittedEntryById.ExecuteAsync(parameters.EntryId, cancellationToken); + if (result.IsErrorResult()) + return; + + Entry = result.Data?.Entry; + await ApplyFromEntry(cancellationToken); + } + + public override async Task OnClosing(NavigationArguments args) + { + if (!HasChanges) + return; + + bool confirmed = await _windowService.ShowConfirmContentDialog("You have unsaved changes", "Do you want to discard your unsaved changes?"); + if (!confirmed) + args.Cancel(); + } + + private async Task ApplyFromEntry(CancellationToken cancellationToken) + { + if (Entry == null) + return; + + if (EntrySpecificationsViewModel != null) + { + EntrySpecificationsViewModel.PropertyChanged -= EntrySpecificationsViewModelOnPropertyChanged; + ((INotifyCollectionChanged) EntrySpecificationsViewModel.SelectedCategories).CollectionChanged -= SelectedCategoriesOnCollectionChanged; + EntrySpecificationsViewModel.Tags.CollectionChanged -= TagsOnCollectionChanged; + } + + EntrySpecificationsViewModel viewModel = _getGetSpecificationsVm(); + + viewModel.IconBitmap = await GetEntryIcon(cancellationToken); + viewModel.Name = Entry.Name; + viewModel.Summary = Entry.Summary; + viewModel.Description = Entry.Description; + viewModel.PreselectedCategories = Entry.Categories.Select(c => c.Id).ToList(); + + viewModel.Tags.Clear(); + foreach (string tag in Entry.Tags.Select(c => c.Name)) + viewModel.Tags.Add(tag); + + EntrySpecificationsViewModel = viewModel; + EntrySpecificationsViewModel.PropertyChanged += EntrySpecificationsViewModelOnPropertyChanged; + ((INotifyCollectionChanged) EntrySpecificationsViewModel.SelectedCategories).CollectionChanged += SelectedCategoriesOnCollectionChanged; + EntrySpecificationsViewModel.Tags.CollectionChanged += TagsOnCollectionChanged; + } + + private async Task GetEntryIcon(CancellationToken cancellationToken) + { + if (Entry == null) + return null; + + Stream? stream = await _workshopService.GetEntryIcon(Entry.Id, cancellationToken); + return stream != null ? new Bitmap(stream) : null; + } + + private void UpdateHasChanges() + { + if (EntrySpecificationsViewModel == null || Entry == null) + return; + + List categories = EntrySpecificationsViewModel.Categories.Where(c => c.IsSelected).Select(c => c.Id).OrderBy(c => c).ToList(); + List tags = EntrySpecificationsViewModel.Tags.OrderBy(t => t).ToList(); + + HasChanges = EntrySpecificationsViewModel.Name != Entry.Name || + EntrySpecificationsViewModel.Description != Entry.Description || + EntrySpecificationsViewModel.Summary != Entry.Summary || + EntrySpecificationsViewModel.IconChanged || + !tags.SequenceEqual(Entry.Tags.Select(t => t.Name).OrderBy(t => t)) || + !categories.SequenceEqual(Entry.Categories.Select(c => c.Id).OrderBy(c => c)); + } + + private async Task ExecuteDiscardChanges() + { + await ApplyFromEntry(CancellationToken.None); + } + + private async Task ExecuteSaveChanges(CancellationToken cancellationToken) + { + if (Entry == null || EntrySpecificationsViewModel == null || !EntrySpecificationsViewModel.ValidationContext.GetIsValid()) + return; + + UpdateEntryInput input = new() + { + Id = Entry.Id, + Name = EntrySpecificationsViewModel.Name, + Summary = EntrySpecificationsViewModel.Summary, + Description = EntrySpecificationsViewModel.Description, + Categories = EntrySpecificationsViewModel.SelectedCategories, + Tags = EntrySpecificationsViewModel.Tags + }; + + IOperationResult result = await _client.UpdateEntry.ExecuteAsync(input, cancellationToken); + result.EnsureNoErrors(); + + if (EntrySpecificationsViewModel.IconChanged && EntrySpecificationsViewModel.IconBitmap != null) + { + using MemoryStream stream = new(); + EntrySpecificationsViewModel.IconBitmap.Save(stream); + ImageUploadResult imageResult = await _workshopService.SetEntryIcon(Entry.Id, new Progress(), stream, cancellationToken); + if (!imageResult.IsSuccess) + throw new ArtemisWorkshopException("Failed to upload image. " + imageResult.Message); + } + + HasChanges = false; + } + + private async Task ExecuteCreateRelease(CancellationToken cancellationToken) + { + if (Entry != null) + await _windowService.ShowDialogAsync(Entry); + } + + private async Task ExecuteDeleteSubmission(CancellationToken cancellationToken) + { + if (Entry == null) + return; + + bool confirmed = await _windowService.ShowConfirmContentDialog( + "Delete submission?", + "You cannot undo this by yourself.\r\n" + + "Users that have already downloaded your submission will keep it."); + if (!confirmed) + return; + + IOperationResult result = await _client.RemoveEntry.ExecuteAsync(Entry.Id, cancellationToken); + result.EnsureNoErrors(); + await _router.Navigate("workshop/library/submissions"); + } + + private async Task ExecuteViewWorkshopPage() + { + if (Entry != null) + await _workshopService.NavigateToEntry(Entry.Id, Entry.EntryType); + } + + private void TagsOnCollectionChanged(object? sender, NotifyCollectionChangedEventArgs e) + { + UpdateHasChanges(); + } + + private void SelectedCategoriesOnCollectionChanged(object? sender, NotifyCollectionChangedEventArgs e) + { + UpdateHasChanges(); + } + + private void EntrySpecificationsViewModelOnPropertyChanged(object? sender, PropertyChangedEventArgs e) + { + UpdateHasChanges(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemView.axaml b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemView.axaml new file mode 100644 index 000000000..00e7eae8f --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemView.axaml @@ -0,0 +1,58 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemView.axaml.cs new file mode 100644 index 000000000..be6199b60 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemView.axaml.cs @@ -0,0 +1,13 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Markup.Xaml; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public partial class InstalledTabItemView : UserControl +{ + public InstalledTabItemView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemViewModel.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemViewModel.cs new file mode 100644 index 000000000..1b22e37aa --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabItemViewModel.cs @@ -0,0 +1,70 @@ +using System; +using System.Reactive; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Shared; +using Artemis.UI.Shared.Routing; +using Artemis.UI.Shared.Services; +using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Handlers.InstallationHandlers; +using Artemis.WebClient.Workshop.Services; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public class InstalledTabItemViewModel : ViewModelBase +{ + private readonly IWorkshopService _workshopService; + private readonly IRouter _router; + private readonly EntryInstallationHandlerFactory _factory; + private readonly IWindowService _windowService; + private bool _isRemoved; + + public InstalledTabItemViewModel(InstalledEntry installedEntry, IWorkshopService workshopService, IRouter router, EntryInstallationHandlerFactory factory, IWindowService windowService) + { + _workshopService = workshopService; + _router = router; + _factory = factory; + _windowService = windowService; + InstalledEntry = installedEntry; + + ViewWorkshopPage = ReactiveCommand.CreateFromTask(ExecuteViewWorkshopPage); + ViewLocal = ReactiveCommand.CreateFromTask(ExecuteViewLocal); + Uninstall = ReactiveCommand.CreateFromTask(ExecuteUninstall); + } + + public bool IsRemoved + { + get => _isRemoved; + private set => RaiseAndSetIfChanged(ref _isRemoved, value); + } + + public InstalledEntry InstalledEntry { get; } + public ReactiveCommand ViewWorkshopPage { get; } + public ReactiveCommand ViewLocal { get; } + public ReactiveCommand Uninstall { get; } + + private async Task ExecuteViewWorkshopPage() + { + await _workshopService.NavigateToEntry(InstalledEntry.EntryId, InstalledEntry.EntryType); + } + + private async Task ExecuteViewLocal(CancellationToken cancellationToken) + { + if (InstalledEntry.EntryType == EntryType.Profile && Guid.TryParse(InstalledEntry.LocalReference, out Guid profileId)) + { + await _router.Navigate($"profile-editor/{profileId}"); + } + } + + private async Task ExecuteUninstall(CancellationToken cancellationToken) + { + bool confirmed = await _windowService.ShowConfirmContentDialog("Do you want to uninstall this entry?", "Both the entry and its contents will be removed."); + if (!confirmed) + return; + + IEntryInstallationHandler handler = _factory.CreateHandler(InstalledEntry.EntryType); + await handler.UninstallAsync(InstalledEntry, cancellationToken); + IsRemoved = true; + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabView.axaml b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabView.axaml new file mode 100644 index 000000000..fe1fef9e2 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabView.axaml @@ -0,0 +1,39 @@ + + + + + + + + + + Not much here yet, huh! + + Any entries you download from the workshop you can later manage here + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabView.axaml.cs new file mode 100644 index 000000000..8a9a8b444 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabView.axaml.cs @@ -0,0 +1,14 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Markup.Xaml; +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public partial class InstalledTabView : ReactiveUserControl +{ + public InstalledTabView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabViewModel.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabViewModel.cs new file mode 100644 index 000000000..7267c6e25 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/InstalledTabViewModel.cs @@ -0,0 +1,59 @@ +using System; +using System.Collections.Generic; +using System.Collections.ObjectModel; +using System.Reactive; +using System.Reactive.Linq; +using Artemis.UI.Shared.Routing; +using Artemis.WebClient.Workshop.Services; +using DynamicData; +using DynamicData.Binding; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public class InstalledTabViewModel : RoutableScreen +{ + private string? _searchEntryInput; + + public InstalledTabViewModel(IWorkshopService workshopService, IRouter router, Func getInstalledTabItemViewModel) + { + SourceList installedEntries = new(); + IObservable> pluginFilter = this.WhenAnyValue(vm => vm.SearchEntryInput).Throttle(TimeSpan.FromMilliseconds(100)).Select(CreatePredicate); + + installedEntries.Connect() + .Filter(pluginFilter) + .Sort(SortExpressionComparer.Descending(p => p.InstalledAt)) + .Transform(getInstalledTabItemViewModel) + .AutoRefresh(vm => vm.IsRemoved) + .Filter(vm => !vm.IsRemoved) + .Bind(out ReadOnlyObservableCollection installedEntryViewModels) + .Subscribe(); + + List entries = workshopService.GetInstalledEntries(); + installedEntries.AddRange(entries); + + Empty = entries.Count == 0; + InstalledEntries = installedEntryViewModels; + + OpenWorkshop = ReactiveCommand.CreateFromTask(async () => await router.Navigate("workshop")); + } + + public bool Empty { get; } + public ReactiveCommand OpenWorkshop { get; } + + public ReadOnlyObservableCollection InstalledEntries { get; } + + public string? SearchEntryInput + { + get => _searchEntryInput; + set => RaiseAndSetIfChanged(ref _searchEntryInput, value); + } + + private Func CreatePredicate(string? text) + { + if (string.IsNullOrWhiteSpace(text)) + return _ => true; + + return data => data.Name.Contains(text, StringComparison.InvariantCultureIgnoreCase); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemView.axaml b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemView.axaml new file mode 100644 index 000000000..583f14c4f --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemView.axaml @@ -0,0 +1,76 @@ + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemView.axaml.cs new file mode 100644 index 000000000..8bf14a3bf --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemView.axaml.cs @@ -0,0 +1,13 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Markup.Xaml; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public partial class SubmissionsTabItemView : UserControl +{ + public SubmissionsTabItemView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemViewModel.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemViewModel.cs new file mode 100644 index 000000000..21c8098ee --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabItemViewModel.cs @@ -0,0 +1,30 @@ +using System.Reactive; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Shared; +using Artemis.UI.Shared.Routing; +using Artemis.WebClient.Workshop; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public class SubmissionsTabItemViewModel : ViewModelBase +{ + private readonly IRouter _router; + + public SubmissionsTabItemViewModel(IGetSubmittedEntries_SubmittedEntries entry, IRouter router) + { + _router = router; + Entry = entry; + + NavigateToEntry = ReactiveCommand.CreateFromTask(ExecuteNavigateToEntry); + } + + public IGetSubmittedEntries_SubmittedEntries Entry { get; } + public ReactiveCommand NavigateToEntry { get; } + + private async Task ExecuteNavigateToEntry(CancellationToken cancellationToken) + { + await _router.Navigate($"workshop/library/submissions/{Entry.Id}"); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabView.axaml b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabView.axaml new file mode 100644 index 000000000..c52644447 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabView.axaml @@ -0,0 +1,51 @@ + + + + + + + + + + + You are not logged in + + In order to manage your submissions you must be logged in. + + + + + + + + Oh boy, it's empty here 🤔 + + Any entries you submit to the workshop you can later manage here + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabView.axaml.cs new file mode 100644 index 000000000..cedd6574c --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabView.axaml.cs @@ -0,0 +1,14 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Markup.Xaml; +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public partial class SubmissionsTabView : ReactiveUserControl +{ + public SubmissionsTabView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabViewModel.cs b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabViewModel.cs new file mode 100644 index 000000000..76947ec5d --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/Tabs/SubmissionsTabViewModel.cs @@ -0,0 +1,106 @@ +using System; +using System.Collections.ObjectModel; +using System.Reactive; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Extensions; +using Artemis.UI.Screens.Workshop.CurrentUser; +using Artemis.UI.Screens.Workshop.SubmissionWizard; +using Artemis.UI.Shared.Routing; +using Artemis.UI.Shared.Services; +using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Services; +using DynamicData; +using ReactiveUI; +using StrawberryShake; + +namespace Artemis.UI.Screens.Workshop.Library.Tabs; + +public class SubmissionsTabViewModel : RoutableScreen +{ + private readonly IWorkshopClient _client; + private readonly SourceCache _entries; + private readonly IWindowService _windowService; + private bool _isLoading = true; + private bool _workshopReachable; + + public SubmissionsTabViewModel(IWorkshopClient client, + IAuthenticationService authenticationService, + IWindowService windowService, + IWorkshopService workshopService, + Func getSubmissionsTabItemViewModel) + { + _client = client; + _windowService = windowService; + _entries = new SourceCache(e => e.Id); + _entries.Connect() + .Transform(getSubmissionsTabItemViewModel) + .Bind(out ReadOnlyObservableCollection entries) + .Subscribe(); + + AddSubmission = ReactiveCommand.CreateFromTask(ExecuteAddSubmission, this.WhenAnyValue(vm => vm.WorkshopReachable)); + Login = ReactiveCommand.CreateFromTask(ExecuteLogin, this.WhenAnyValue(vm => vm.WorkshopReachable)); + + IsLoggedIn = authenticationService.IsLoggedIn; + Entries = entries; + + this.WhenActivatedAsync(async d => + { + WorkshopReachable = await workshopService.ValidateWorkshopStatus(d.AsCancellationToken()); + if (WorkshopReachable) + await GetEntries(d.AsCancellationToken()); + }); + } + + public ReactiveCommand Login { get; } + public ReactiveCommand AddSubmission { get; } + + public IObservable IsLoggedIn { get; } + public ReadOnlyObservableCollection Entries { get; } + + public bool WorkshopReachable + { + get => _workshopReachable; + set => RaiseAndSetIfChanged(ref _workshopReachable, value); + } + + public bool IsLoading + { + get => _isLoading; + set => RaiseAndSetIfChanged(ref _isLoading, value); + } + + private async Task ExecuteLogin(CancellationToken ct) + { + await _windowService.CreateContentDialog().WithViewModel(out WorkshopLoginViewModel _).WithTitle("Workshop login").ShowAsync(); + await GetEntries(ct); + } + + private async Task ExecuteAddSubmission(CancellationToken arg) + { + await _windowService.ShowDialogAsync(); + } + + private async Task GetEntries(CancellationToken ct) + { + IsLoading = true; + + try + { + IOperationResult result = await _client.GetSubmittedEntries.ExecuteAsync(null, ct); + + if (result.Data?.SubmittedEntries == null) + _entries.Clear(); + else + _entries.Edit(e => + { + e.Clear(); + e.AddOrUpdate(result.Data.SubmittedEntries); + }); + } + finally + { + IsLoading = false; + } + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryVIew.axaml b/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryVIew.axaml new file mode 100644 index 000000000..af8f455a4 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryVIew.axaml @@ -0,0 +1,32 @@ + + + + + + + + + + + + + + + diff --git a/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryVIew.axaml.cs b/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryVIew.axaml.cs new file mode 100644 index 000000000..7758b5729 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryVIew.axaml.cs @@ -0,0 +1,28 @@ +using System; +using System.Reactive.Disposables; +using Artemis.UI.Shared; +using Avalonia.ReactiveUI; +using Avalonia.Threading; +using FluentAvalonia.UI.Controls; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Library; + +public partial class WorkshopLibraryView : ReactiveUserControl +{ + public WorkshopLibraryView() + { + InitializeComponent(); + this.WhenActivated(d => { ViewModel.WhenAnyValue(vm => vm.Screen).WhereNotNull().Subscribe(Navigate).DisposeWith(d); }); + } + + private void Navigate(ViewModelBase viewModel) + { + Dispatcher.UIThread.Invoke(() => TabFrame.NavigateFromObject(viewModel)); + } + + private void NavigationView_OnBackRequested(object? sender, NavigationViewBackRequestedEventArgs e) + { + ViewModel?.GoBack(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryViewModel.cs b/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryViewModel.cs new file mode 100644 index 000000000..56abc8ffd --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Library/WorkshopLibraryViewModel.cs @@ -0,0 +1,65 @@ +using System.Collections.ObjectModel; +using System.Linq; +using System.Reactive.Disposables; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Routing; +using Artemis.UI.Shared.Routing; +using ReactiveUI; +using System; +using System.Reactive.Linq; + +namespace Artemis.UI.Screens.Workshop.Library; + +public class WorkshopLibraryViewModel : RoutableHostScreen +{ + private readonly IRouter _router; + private RouteViewModel? _selectedTab; + private ObservableAsPropertyHelper? _viewingDetails; + + /// + public WorkshopLibraryViewModel(IRouter router) + { + _router = router; + + Tabs = new ObservableCollection + { + new("Installed", "workshop/library/installed"), + new("Submissions", "workshop/library/submissions") + }; + + this.WhenActivated(d => + { + _viewingDetails = _router.CurrentPath.Select(p => p != null && p.StartsWith("workshop/library/submissions/")).ToProperty(this, vm => vm.ViewingDetails).DisposeWith(d); + // Navigate on tab change + this.WhenAnyValue(vm => vm.SelectedTab) + .WhereNotNull() + .Subscribe(s => router.Navigate(s.Path, new RouterNavigationOptions {IgnoreOnPartialMatch = true})) + .DisposeWith(d); + }); + } + + public bool ViewingDetails => _viewingDetails?.Value ?? false; + public ObservableCollection Tabs { get; } + + public RouteViewModel? SelectedTab + { + get => _selectedTab; + set => RaiseAndSetIfChanged(ref _selectedTab, value); + } + + public override async Task OnNavigating(NavigationArguments args, CancellationToken cancellationToken) + { + SelectedTab = Tabs.FirstOrDefault(t => t.Matches(args.Path)); + if (SelectedTab == null) + await args.Router.Navigate(Tabs.First().Path); + } + + public void GoBack() + { + if (ViewingDetails) + _router.GoBack(); + else + _router.Navigate("workshop"); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Parameters/WorkshopDetailParameters.cs b/src/Artemis.UI/Screens/Workshop/Parameters/WorkshopDetailParameters.cs index dfd37805f..86503f0d4 100644 --- a/src/Artemis.UI/Screens/Workshop/Parameters/WorkshopDetailParameters.cs +++ b/src/Artemis.UI/Screens/Workshop/Parameters/WorkshopDetailParameters.cs @@ -1,8 +1,6 @@ -using System; - namespace Artemis.UI.Screens.Workshop.Parameters; public class WorkshopDetailParameters { - public Guid EntryId { get; set; } + public long EntryId { get; set; } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml b/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml index 12d0a8227..e8dbea39a 100644 --- a/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml +++ b/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml @@ -3,23 +3,131 @@ xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" xmlns:profile="clr-namespace:Artemis.UI.Screens.Workshop.Profile" - mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450" + xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" + xmlns:il="clr-namespace:AsyncImageLoader;assembly=AsyncImageLoader.Avalonia" + xmlns:converters="clr-namespace:Artemis.UI.Converters" + xmlns:mdxaml="https://github.com/whistyun/Markdown.Avalonia.Tight" + xmlns:mdsvg="https://github.com/whistyun/Markdown.Avalonia.Svg" + xmlns:ui="clr-namespace:Artemis.UI" + xmlns:converters1="clr-namespace:Artemis.UI.Shared.Converters;assembly=Artemis.UI.Shared" + mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="800" x:Class="Artemis.UI.Screens.Workshop.Profile.ProfileDetailsView" x:DataType="profile:ProfileDetailsViewModel"> - - - - - - - - - Side panel + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + downloads + + + + + Created + + + + + Updated + + + - - Profile details panel + + + Latest release + + + - - + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml.cs index 25ca88e9f..1150bd94c 100644 --- a/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/Profile/ProfileDetailsView.axaml.cs @@ -1,4 +1,3 @@ -using Avalonia.Markup.Xaml; using Avalonia.ReactiveUI; namespace Artemis.UI.Screens.Workshop.Profile; @@ -9,9 +8,4 @@ public partial class ProfileDetailsView : ReactiveUserControl, IWorkshopViewModel +public class ProfileDetailsViewModel : RoutableScreen { private readonly IWorkshopClient _client; + private readonly ProfileEntryInstallationHandler _installationHandler; + private readonly INotificationService _notificationService; + private readonly IWindowService _windowService; + private readonly ObservableAsPropertyHelper _updatedAt; private IGetEntryById_Entry? _entry; - public ProfileDetailsViewModel(IWorkshopClient client) + public ProfileDetailsViewModel(IWorkshopClient client, ProfileEntryInstallationHandler installationHandler, INotificationService notificationService, IWindowService windowService) { _client = client; + _installationHandler = installationHandler; + _notificationService = notificationService; + _windowService = windowService; + _updatedAt = this.WhenAnyValue(vm => vm.Entry).Select(e => e?.LatestRelease?.CreatedAt ?? e?.CreatedAt).ToProperty(this, vm => vm.UpdatedAt); + + DownloadLatestRelease = ReactiveCommand.CreateFromTask(ExecuteDownloadLatestRelease); } - public EntryType? EntryType => null; + public ReactiveCommand DownloadLatestRelease { get; } + + public DateTimeOffset? UpdatedAt => _updatedAt.Value; public IGetEntryById_Entry? Entry { get => _entry; - set => RaiseAndSetIfChanged(ref _entry, value); + private set => RaiseAndSetIfChanged(ref _entry, value); } public override async Task OnNavigating(WorkshopDetailParameters parameters, NavigationArguments args, CancellationToken cancellationToken) @@ -32,7 +51,7 @@ public class ProfileDetailsViewModel : RoutableScreen result = await _client.GetEntryById.ExecuteAsync(entryId, cancellationToken); if (result.IsErrorResult()) @@ -40,4 +59,20 @@ public class ProfileDetailsViewModel : RoutableScreen(), cancellationToken); + if (result.IsSuccess) + _notificationService.CreateNotification().WithTitle("Profile installed").WithSeverity(NotificationSeverity.Success).Show(); + else + _notificationService.CreateNotification().WithTitle("Failed to install profile").WithMessage(result.Message).WithSeverity(NotificationSeverity.Error).Show(); + } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfileListView.axaml b/src/Artemis.UI/Screens/Workshop/Profile/ProfileListView.axaml deleted file mode 100644 index 3ae086124..000000000 --- a/src/Artemis.UI/Screens/Workshop/Profile/ProfileListView.axaml +++ /dev/null @@ -1,41 +0,0 @@ - - - - - - Categories - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfileListViewModel.cs b/src/Artemis.UI/Screens/Workshop/Profile/ProfileListViewModel.cs deleted file mode 100644 index 98e45f00b..000000000 --- a/src/Artemis.UI/Screens/Workshop/Profile/ProfileListViewModel.cs +++ /dev/null @@ -1,146 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using System.Reactive.Disposables; -using System.Reactive.Linq; -using System.Threading; -using System.Threading.Tasks; -using Artemis.UI.Screens.Workshop.Categories; -using Artemis.UI.Screens.Workshop.Entries; -using Artemis.UI.Screens.Workshop.Parameters; -using Artemis.UI.Shared; -using Artemis.UI.Shared.Routing; -using Artemis.UI.Shared.Services; -using Artemis.UI.Shared.Services.Builders; -using Artemis.WebClient.Workshop; -using ReactiveUI; -using StrawberryShake; - -namespace Artemis.UI.Screens.Workshop.Profile; - -public class ProfileListViewModel : RoutableScreen, IWorkshopViewModel -{ - private readonly IRouter _router; - private readonly INotificationService _notificationService; - private readonly IWorkshopClient _workshopClient; - private readonly ObservableAsPropertyHelper _showPagination; - private readonly ObservableAsPropertyHelper _isLoading; - private List? _entries; - private int _page; - private int _loadedPage = -1; - private int _totalPages = 1; - private int _entriesPerPage = 10; - - public ProfileListViewModel(IWorkshopClient workshopClient, IRouter router, CategoriesViewModel categoriesViewModel, INotificationService notificationService) - { - _workshopClient = workshopClient; - _router = router; - _notificationService = notificationService; - _showPagination = this.WhenAnyValue(vm => vm.TotalPages).Select(t => t > 1).ToProperty(this, vm => vm.ShowPagination); - _isLoading = this.WhenAnyValue(vm => vm.Page, vm => vm.LoadedPage, (p, c) => p != c).ToProperty(this, vm => vm.IsLoading); - - CategoriesViewModel = categoriesViewModel; - - // Respond to page changes - this.WhenAnyValue(vm => vm.Page).Skip(1).Subscribe(p => Task.Run(() => _router.Navigate($"workshop/profiles/{p}"))); - // Respond to filter changes - this.WhenActivated(d => CategoriesViewModel.WhenAnyValue(vm => vm.CategoryFilters).Skip(1).Subscribe(_ => - { - // Reset to page one, will trigger a query - if (Page != 1) - Page = 1; - // If already at page one, force a query - else - Task.Run(() => Query(CancellationToken.None)); - }).DisposeWith(d)); - } - - public bool ShowPagination => _showPagination.Value; - public bool IsLoading => _isLoading.Value; - - public CategoriesViewModel CategoriesViewModel { get; } - - public List? Entries - { - get => _entries; - set => RaiseAndSetIfChanged(ref _entries, value); - } - - public int Page - { - get => _page; - set => RaiseAndSetIfChanged(ref _page, value); - } - - public int LoadedPage - { - get => _loadedPage; - set => RaiseAndSetIfChanged(ref _loadedPage, value); - } - - public int TotalPages - { - get => _totalPages; - set => RaiseAndSetIfChanged(ref _totalPages, value); - } - - public int EntriesPerPage - { - get => _entriesPerPage; - set => RaiseAndSetIfChanged(ref _entriesPerPage, value); - } - - public override async Task OnNavigating(WorkshopListParameters parameters, NavigationArguments args, CancellationToken cancellationToken) - { - Page = Math.Max(1, parameters.Page); - - // Throttle page changes - await Task.Delay(200, cancellationToken); - - if (!cancellationToken.IsCancellationRequested) - await Query(cancellationToken); - } - - private async Task Query(CancellationToken cancellationToken) - { - try - { - EntryFilterInput filter = GetFilter(); - IOperationResult entries = await _workshopClient.GetEntries.ExecuteAsync(filter, EntriesPerPage * (Page - 1), EntriesPerPage, cancellationToken); - entries.EnsureNoErrors(); - - if (entries.Data?.Entries?.Items != null) - { - Entries = entries.Data.Entries.Items.Select(n => new EntryListViewModel(n, _router)).ToList(); - TotalPages = (int) Math.Ceiling(entries.Data.Entries.TotalCount / (double) EntriesPerPage); - } - else - TotalPages = 1; - } - catch (Exception e) - { - _notificationService.CreateNotification() - .WithTitle("Failed to load entries") - .WithMessage(e.Message) - .WithSeverity(NotificationSeverity.Error) - .Show(); - } - finally - { - LoadedPage = Page; - } - } - - private EntryFilterInput GetFilter() - { - EntryFilterInput filter = new() - { - EntryType = new EntryTypeOperationFilterInput {Eq = WebClient.Workshop.EntryType.Profile}, - And = CategoriesViewModel.CategoryFilters - }; - - return filter; - } - - public EntryType? EntryType => null; -} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewView.axaml b/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewView.axaml new file mode 100644 index 000000000..0b30e7d07 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewView.axaml @@ -0,0 +1,58 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewView.axaml.cs new file mode 100644 index 000000000..b956e6ecf --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewView.axaml.cs @@ -0,0 +1,79 @@ +using System; +using System.Linq; +using System.Reactive.Linq; +using Avalonia; +using Avalonia.Controls.PanAndZoom; +using Avalonia.Input; +using Avalonia.Media; +using Avalonia.ReactiveUI; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Profile; + +public partial class ProfilePreviewView : ReactiveUserControl +{ + private bool _movedByUser; + + public ProfilePreviewView() + { + InitializeComponent(); + + ZoomBorder.PropertyChanged += ZoomBorderOnPropertyChanged; + ZoomBorder.PointerMoved += ZoomBorderOnPointerMoved; + ZoomBorder.PointerWheelChanged += ZoomBorderOnPointerWheelChanged; + UpdateZoomBorderBackground(); + + this.WhenAnyValue(v => v.Bounds).Where(_ => !_movedByUser).Subscribe(_ => AutoFit()); + } + + private void ZoomBorderOnPointerWheelChanged(object? sender, PointerWheelEventArgs e) + { + _movedByUser = true; + } + + private void ZoomBorderOnPointerMoved(object? sender, PointerEventArgs e) + { + if (e.GetCurrentPoint(ZoomBorder).Properties.IsMiddleButtonPressed) + _movedByUser = true; + } + + private void ZoomBorderOnPropertyChanged(object? sender, AvaloniaPropertyChangedEventArgs e) + { + if (e.Property.Name == nameof(ZoomBorder.Background)) + UpdateZoomBorderBackground(); + } + + private void UpdateZoomBorderBackground() + { + if (ZoomBorder.Background is VisualBrush visualBrush) + visualBrush.DestinationRect = new RelativeRect(ZoomBorder.OffsetX * -1, ZoomBorder.OffsetY * -1, 20, 20, RelativeUnit.Absolute); + } + + private void ZoomBorder_OnZoomChanged(object sender, ZoomChangedEventArgs e) + { + UpdateZoomBorderBackground(); + } + + private void AutoFit() + { + if (ViewModel == null || !ViewModel.Devices.Any()) + return; + + double left = ViewModel.Devices.Select(d => d.Rectangle.Left).Min(); + double top = ViewModel.Devices.Select(d => d.Rectangle.Top).Min(); + double bottom = ViewModel.Devices.Select(d => d.Rectangle.Bottom).Max(); + double right = ViewModel.Devices.Select(d => d.Rectangle.Right).Max(); + + // Add a 10 pixel margin around the rect + Rect scriptRect = new(new Point(left - 10, top - 10), new Point(right + 10, bottom + 10)); + + // The scale depends on the available space + double scale = Math.Min(3, Math.Min(Bounds.Width / scriptRect.Width, Bounds.Height / scriptRect.Height)); + + // Pan and zoom to make the script fit + ZoomBorder.Zoom(scale, 0, 0, true); + ZoomBorder.Pan(Bounds.Center.X - scriptRect.Center.X * scale, Bounds.Center.Y - scriptRect.Center.Y * scale, true); + + _movedByUser = false; + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewViewModel.cs b/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewViewModel.cs new file mode 100644 index 000000000..a874ff268 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Profile/ProfilePreviewViewModel.cs @@ -0,0 +1,55 @@ +using System; +using System.Collections.ObjectModel; +using System.Linq; +using System.Reactive.Disposables; +using Artemis.Core; +using Artemis.Core.Services; +using Artemis.UI.Shared; +using Artemis.UI.Shared.Services; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Profile; + +public class ProfilePreviewViewModel : ActivatableViewModelBase +{ + private readonly IProfileService _profileService; + private readonly IWindowService _windowService; + private ProfileConfiguration? _profileConfiguration; + + public ProfilePreviewViewModel(IProfileService profileService, IRgbService rgbService, IWindowService windowService) + { + _profileService = profileService; + _windowService = windowService; + + Devices = new ObservableCollection(rgbService.EnabledDevices.OrderBy(d => d.ZIndex)); + + this.WhenAnyValue(vm => vm.ProfileConfiguration).Subscribe(_ => Update()); + this.WhenActivated(d => Disposable.Create(() => PreviewProfile(null)).DisposeWith(d)); + } + + public ObservableCollection Devices { get; } + + public ProfileConfiguration? ProfileConfiguration + { + get => _profileConfiguration; + set => RaiseAndSetIfChanged(ref _profileConfiguration, value); + } + + private void Update() + { + try + { + PreviewProfile(ProfileConfiguration); + } + catch (Exception e) + { + _windowService.ShowExceptionDialog("Failed to load preview", e); + } + } + + private void PreviewProfile(ProfileConfiguration? profileConfiguration) + { + _profileService.FocusProfile = profileConfiguration; + _profileService.UpdateFocusProfile = profileConfiguration != null; + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Search/SearchResultView.axaml b/src/Artemis.UI/Screens/Workshop/Search/SearchResultView.axaml new file mode 100644 index 000000000..4fa117ff3 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Search/SearchResultView.axaml @@ -0,0 +1,62 @@ + + + + + + + + + + + + + + + by + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Search/SearchResultView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Search/SearchResultView.axaml.cs new file mode 100644 index 000000000..8490b69b2 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Search/SearchResultView.axaml.cs @@ -0,0 +1,14 @@ +using Avalonia; +using Avalonia.Controls; +using Avalonia.Markup.Xaml; +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.Search; + +public partial class SearchResultView : ReactiveUserControl +{ + public SearchResultView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Search/SearchResultViewModel.cs b/src/Artemis.UI/Screens/Workshop/Search/SearchResultViewModel.cs new file mode 100644 index 000000000..b4c5c0810 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/Search/SearchResultViewModel.cs @@ -0,0 +1,14 @@ +using Artemis.UI.Shared; +using Artemis.WebClient.Workshop; + +namespace Artemis.UI.Screens.Workshop.Search; + +public class SearchResultViewModel : ActivatableViewModelBase +{ + public SearchResultViewModel(ISearchEntries_SearchEntries entry) + { + Entry = entry; + } + + public ISearchEntries_SearchEntries Entry { get; } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml b/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml index 171d4c400..c66d738c1 100644 --- a/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml +++ b/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml @@ -5,53 +5,42 @@ xmlns:search="clr-namespace:Artemis.UI.Screens.Workshop.Search" xmlns:workshop="clr-namespace:Artemis.WebClient.Workshop;assembly=Artemis.WebClient.Workshop" xmlns:windowing="clr-namespace:FluentAvalonia.UI.Windowing;assembly=FluentAvalonia" + xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450" x:Class="Artemis.UI.Screens.Workshop.Search.SearchView" x:DataType="search:SearchViewModel"> - + - - - - - - - - - - - - - - - - - - - - - - + + - + windowing:AppWindow.AllowInteractionInTitleBar="True" /> + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml.cs b/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml.cs index e9da03bdb..c819fce33 100644 --- a/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/Search/SearchView.axaml.cs @@ -1,6 +1,4 @@ -using System; using Avalonia.Controls; -using Avalonia.Markup.Xaml; namespace Artemis.UI.Screens.Workshop.Search; @@ -10,9 +8,4 @@ public partial class SearchView : UserControl { InitializeComponent(); } - - private void InitializeComponent() - { - AvaloniaXamlLoader.Load(this); - } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Search/SearchViewModel.cs b/src/Artemis.UI/Screens/Workshop/Search/SearchViewModel.cs index b3370ee8b..000f0a330 100644 --- a/src/Artemis.UI/Screens/Workshop/Search/SearchViewModel.cs +++ b/src/Artemis.UI/Screens/Workshop/Search/SearchViewModel.cs @@ -4,64 +4,84 @@ using System.Linq; using System.Threading; using System.Threading.Tasks; using Artemis.UI.Screens.Workshop.CurrentUser; +using Artemis.UI.Services.Interfaces; using Artemis.UI.Shared; -using Artemis.UI.Shared.Routing; using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Services; using ReactiveUI; +using Serilog; using StrawberryShake; namespace Artemis.UI.Screens.Workshop.Search; public class SearchViewModel : ViewModelBase { - public CurrentUserViewModel CurrentUserViewModel { get; } - private readonly IRouter _router; + private readonly ILogger _logger; + private readonly IWorkshopService _workshopService; + private readonly IDebugService _debugService; private readonly IWorkshopClient _workshopClient; - private EntryType? _entryType; - private ISearchEntries_SearchEntries? _selectedEntry; + private bool _isLoading; + private SearchResultViewModel? _selectedEntry; - public SearchViewModel(IWorkshopClient workshopClient, IRouter router, CurrentUserViewModel currentUserViewModel) + public SearchViewModel(ILogger logger, IWorkshopClient workshopClient, IWorkshopService workshopService, CurrentUserViewModel currentUserViewModel, IDebugService debugService) { - CurrentUserViewModel = currentUserViewModel; + _logger = logger; _workshopClient = workshopClient; - _router = router; + _workshopService = workshopService; + _debugService = debugService; + CurrentUserViewModel = currentUserViewModel; SearchAsync = ExecuteSearchAsync; this.WhenAnyValue(vm => vm.SelectedEntry).WhereNotNull().Subscribe(NavigateToEntry); } + public CurrentUserViewModel CurrentUserViewModel { get; } + public Func>> SearchAsync { get; } - public ISearchEntries_SearchEntries? SelectedEntry + public SearchResultViewModel? SelectedEntry { get => _selectedEntry; set => RaiseAndSetIfChanged(ref _selectedEntry, value); } - public EntryType? EntryType + public bool IsLoading { - get => _entryType; - set => RaiseAndSetIfChanged(ref _entryType, value); + get => _isLoading; + set => RaiseAndSetIfChanged(ref _isLoading, value); } - private void NavigateToEntry(ISearchEntries_SearchEntries entry) + public void ShowDebugger() { - string? url = null; - if (entry.EntryType == WebClient.Workshop.EntryType.Profile) - url = $"workshop/profiles/{entry.Id}"; - if (entry.EntryType == WebClient.Workshop.EntryType.Layout) - url = $"workshop/layouts/{entry.Id}"; - - if (url != null) - Task.Run(() => _router.Navigate(url)); + _debugService.ShowDebugger(); + } + + private void NavigateToEntry(SearchResultViewModel searchResult) + { + _workshopService.NavigateToEntry(searchResult.Entry.Id, searchResult.Entry.EntryType); } private async Task> ExecuteSearchAsync(string? input, CancellationToken cancellationToken) { - if (string.IsNullOrWhiteSpace(input)) - return new List(); - - IOperationResult results = await _workshopClient.SearchEntries.ExecuteAsync(input, EntryType, cancellationToken); - return results.Data?.SearchEntries.Cast() ?? new List(); + try + { + if (string.IsNullOrWhiteSpace(input) || input.Length < 2) + return new List(); + + IsLoading = true; + IOperationResult results = await _workshopClient.SearchEntries.ExecuteAsync(input, null, cancellationToken); + return results.Data?.SearchEntries.Select(e => new SearchResultViewModel(e) as object) ?? new List(); + } + catch (Exception e) + { + if (e is not TaskCanceledException) + _logger.Error(e, "Failed to execute search"); + } + finally + { + IsLoading = false; + } + + return new List(); } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Search/SearchViewStyles.axaml b/src/Artemis.UI/Screens/Workshop/Search/SearchViewStyles.axaml index 584a83924..c489f0406 100644 --- a/src/Artemis.UI/Screens/Workshop/Search/SearchViewStyles.axaml +++ b/src/Artemis.UI/Screens/Workshop/Search/SearchViewStyles.axaml @@ -19,7 +19,7 @@ - + + + + Submission type + + + Please select the type of content you want to submit to the workshop. + + + + + Profile + A profile which others can install to enjoy new lighting and interactions. + + + + + + + Layout + A layout providing users with a visual representation of a device. + + + + + diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/EntryTypeStepView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/EntryTypeStepView.axaml.cs new file mode 100644 index 000000000..087084ca5 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/EntryTypeStepView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public partial class EntryTypeStepView : ReactiveUserControl +{ + public EntryTypeStepView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/EntryTypeStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/EntryTypeStepViewModel.cs new file mode 100644 index 000000000..e378e27a3 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/EntryTypeStepViewModel.cs @@ -0,0 +1,38 @@ +using System.Reactive.Linq; +using Artemis.WebClient.Workshop; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public class EntryTypeStepViewModel : SubmissionViewModel +{ + private EntryType? _selectedEntryType; + + /// + public EntryTypeStepViewModel() + { + GoBack = ReactiveCommand.Create(() => State.ChangeScreen()); + Continue = ReactiveCommand.Create(ExecuteContinue, this.WhenAnyValue(vm => vm.SelectedEntryType).Select(e => e != null)); + } + +#if DEBUG + public bool ShowLayouts => true; +# else + public bool ShowLayouts => false; +#endif + + public EntryType? SelectedEntryType + { + get => _selectedEntryType; + set => RaiseAndSetIfChanged(ref _selectedEntryType, value); + } + + private void ExecuteContinue() + { + if (SelectedEntryType == null) + return; + + State.EntryType = SelectedEntryType.Value; + State.StartForCurrentEntry(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepView.axaml new file mode 100644 index 000000000..cfb532af8 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepView.axaml @@ -0,0 +1,33 @@ + + + + + + + + First things first + + In order to submit anything to the workshop you must be logged in. + + Creating an account is free and we'll not bother you with a newsletter or crap like that. + + + + + + Click Log In below to (create an account) and log in. + + You'll also be able to log in with Google or Discord. + + + diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepView.axaml.cs new file mode 100644 index 000000000..3b4ec4b2b --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public partial class LoginStepView : ReactiveUserControl +{ + public LoginStepView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepViewModel.cs new file mode 100644 index 000000000..5b6cf680b --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/LoginStepViewModel.cs @@ -0,0 +1,42 @@ +using System.Linq; +using System.Security.Claims; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Screens.Workshop.CurrentUser; +using Artemis.UI.Shared.Services; +using Artemis.WebClient.Workshop.Services; +using FluentAvalonia.UI.Controls; +using IdentityModel; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public class LoginStepViewModel : SubmissionViewModel +{ + private readonly IAuthenticationService _authenticationService; + private readonly IWindowService _windowService; + + public LoginStepViewModel(IAuthenticationService authenticationService, IWindowService windowService) + { + _authenticationService = authenticationService; + _windowService = windowService; + + Continue = ReactiveCommand.CreateFromTask(ExecuteLogin); + ShowGoBack = false; + ShowHeader = false; + ContinueText = "Log In"; + } + + private async Task ExecuteLogin(CancellationToken ct) + { + ContentDialogResult result = await _windowService.CreateContentDialog().WithViewModel(out WorkshopLoginViewModel _).WithTitle("Workshop login").ShowAsync(); + if (result != ContentDialogResult.Primary) + return; + + Claim? emailVerified = _authenticationService.Claims.FirstOrDefault(c => c.Type == JwtClaimTypes.EmailVerified); + if (emailVerified?.Value == "true") + State.ChangeScreen(); + else + State.ChangeScreen(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsLayerViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsLayerViewModel.cs new file mode 100644 index 000000000..48b741698 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsLayerViewModel.cs @@ -0,0 +1,50 @@ +using System.Reactive; +using System.Reactive.Linq; +using System.Threading.Tasks; +using Artemis.Core; +using Artemis.Core.Services; +using Artemis.UI.Screens.ProfileEditor.ProfileTree.Dialogs; +using Artemis.UI.Shared; +using Artemis.UI.Shared.Services; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.Profile; + +public class ProfileAdaptionHintsLayerViewModel : ViewModelBase +{ + private readonly ObservableAsPropertyHelper _adaptionHintText; + private readonly IProfileService _profileService; + private readonly IWindowService _windowService; + private int _adaptionHintCount; + + public ProfileAdaptionHintsLayerViewModel(Layer layer, IWindowService windowService, IProfileService profileService) + { + _windowService = windowService; + _profileService = profileService; + _adaptionHintText = this.WhenAnyValue(vm => vm.AdaptionHintCount).Select(c => c == 1 ? "1 adaption hint" : $"{c} adaption hints").ToProperty(this, vm => vm.AdaptionHintText); + + Layer = layer; + EditAdaptionHints = ReactiveCommand.CreateFromTask(ExecuteEditAdaptionHints); + AdaptionHintCount = layer.Adapter.AdaptionHints.Count; + } + + public Layer Layer { get; } + + public ReactiveCommand EditAdaptionHints { get; } + + public int AdaptionHintCount + { + get => _adaptionHintCount; + private set => RaiseAndSetIfChanged(ref _adaptionHintCount, value); + } + + public string AdaptionHintText => _adaptionHintText.Value; + + private async Task ExecuteEditAdaptionHints() + { + await _windowService.ShowDialogAsync(Layer); + _profileService.SaveProfile(Layer.Profile, true); + + AdaptionHintCount = Layer.Adapter.AdaptionHints.Count; + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepView.axaml new file mode 100644 index 000000000..b85ffdb0f --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepView.axaml @@ -0,0 +1,67 @@ + + + + + + + + + Set up profile adaption hints + + + Add hints below to help decide where to place this each layer when the profile is imported by another user. + + + + + Learn more about adaption hints + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepView.axaml.cs new file mode 100644 index 000000000..a5f7815d7 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.Profile; + +public partial class ProfileAdaptionHintsStepView : ReactiveUserControl +{ + public ProfileAdaptionHintsStepView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepViewModel.cs new file mode 100644 index 000000000..fc55b987a --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileAdaptionHintsStepViewModel.cs @@ -0,0 +1,65 @@ +using System; +using System.Collections.ObjectModel; +using System.Linq; +using System.Reactive; +using System.Reactive.Disposables; +using System.Threading.Tasks; +using Artemis.Core; +using Artemis.Core.Services; +using Artemis.UI.Screens.ProfileEditor.ProfileTree.Dialogs; +using Artemis.UI.Shared.Services; +using DynamicData; +using DynamicData.Aggregation; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.Profile; + +public class ProfileAdaptionHintsStepViewModel : SubmissionViewModel +{ + private readonly SourceList _layers; + private readonly IProfileService _profileService; + private readonly IWindowService _windowService; + + public ProfileAdaptionHintsStepViewModel(IWindowService windowService, IProfileService profileService, Func getLayerViewModel) + { + _windowService = windowService; + _profileService = profileService; + _layers = new SourceList(); + _layers.Connect().Bind(out ReadOnlyObservableCollection layers).Subscribe(); + + GoBack = ReactiveCommand.Create(() => State.ChangeScreen()); + Continue = ReactiveCommand.Create(ExecuteContinue, _layers.Connect().AutoRefresh(l => l.AdaptionHintCount).Filter(l => l.AdaptionHintCount == 0).IsEmpty()); + EditAdaptionHints = ReactiveCommand.CreateFromTask(ExecuteEditAdaptionHints); + Layers = layers; + + this.WhenActivated((CompositeDisposable _) => + { + if (State.EntrySource is ProfileConfiguration profileConfiguration && profileConfiguration.Profile != null) + _layers.Edit(l => + { + l.Clear(); + l.AddRange(profileConfiguration.Profile.GetAllLayers().Select(getLayerViewModel)); + }); + }); + } + + public ReactiveCommand EditAdaptionHints { get; } + public ReadOnlyObservableCollection Layers { get; } + + private async Task ExecuteEditAdaptionHints(Layer layer) + { + await _windowService.ShowDialogAsync(layer); + _profileService.SaveProfile(layer.Profile, true); + } + + private void ExecuteContinue() + { + if (Layers.Any(l => l.AdaptionHintCount == 0)) + return; + + if (State.EntryId == null) + State.ChangeScreen(); + else + State.ChangeScreen(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepView.axaml new file mode 100644 index 000000000..8fd0eb216 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepView.axaml @@ -0,0 +1,56 @@ + + + + + + + + + + Profile selection + + + Please select the profile you want to share, a preview will be shown below. + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepView.axaml.cs new file mode 100644 index 000000000..84cb5bf27 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.Profile; + +public partial class ProfileSelectionStepView : ReactiveUserControl +{ + public ProfileSelectionStepView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepViewModel.cs new file mode 100644 index 000000000..adad736d7 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/Profile/ProfileSelectionStepViewModel.cs @@ -0,0 +1,84 @@ +using System; +using System.Collections.ObjectModel; +using System.Linq; +using System.Reactive.Disposables; +using System.Reactive.Linq; +using Artemis.Core; +using Artemis.Core.Services; +using Artemis.UI.Extensions; +using Artemis.UI.Screens.Workshop.Profile; +using Material.Icons; +using ReactiveUI; +using SkiaSharp; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.Profile; + +public class ProfileSelectionStepViewModel : SubmissionViewModel +{ + private readonly IProfileService _profileService; + private ProfileConfiguration? _selectedProfile; + + /// + public ProfileSelectionStepViewModel(IProfileService profileService, ProfilePreviewViewModel profilePreviewViewModel) + { + _profileService = profileService; + + // Use copies of the profiles, the originals are used by the core and could be disposed + Profiles = new ObservableCollection(_profileService.ProfileConfigurations.Select(_profileService.CloneProfileConfiguration)); + foreach (ProfileConfiguration profileConfiguration in Profiles) + _profileService.LoadProfileConfigurationIcon(profileConfiguration); + + ProfilePreview = profilePreviewViewModel; + + GoBack = ReactiveCommand.Create(() => State.ChangeScreen()); + Continue = ReactiveCommand.Create(ExecuteContinue, this.WhenAnyValue(vm => vm.SelectedProfile).Select(p => p != null)); + + this.WhenAnyValue(vm => vm.SelectedProfile).Subscribe(p => Update(p)); + this.WhenActivated((CompositeDisposable _) => + { + ShowGoBack = State.EntryId == null; + if (State.EntrySource is ProfileConfiguration profileConfiguration) + SelectedProfile = Profiles.FirstOrDefault(p => p.ProfileId == profileConfiguration.ProfileId); + }); + } + + public ObservableCollection Profiles { get; } + public ProfilePreviewViewModel ProfilePreview { get; } + + public ProfileConfiguration? SelectedProfile + { + get => _selectedProfile; + set => RaiseAndSetIfChanged(ref _selectedProfile, value); + } + + private void Update(ProfileConfiguration? profileConfiguration) + { + ProfilePreview.ProfileConfiguration = null; + + foreach (ProfileConfiguration configuration in Profiles) + { + if (configuration == profileConfiguration) + _profileService.ActivateProfile(configuration); + else + _profileService.DeactivateProfile(configuration); + } + + ProfilePreview.ProfileConfiguration = profileConfiguration; + } + + private void ExecuteContinue() + { + if (SelectedProfile == null) + return; + + State.EntrySource = SelectedProfile; + State.Name = SelectedProfile.Name; + State.Icon = SelectedProfile.Icon.GetIconStream(); + + // Render the material icon of the profile + if (State.Icon == null && SelectedProfile.Icon.IconName != null) + State.Icon = Enum.Parse(SelectedProfile.Icon.IconName).EncodeToBitmap(128, 14, SKColors.White); + + State.ChangeScreen(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepView.axaml new file mode 100644 index 000000000..18e53f90a --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepView.axaml @@ -0,0 +1,26 @@ + + + + + + + + + + + Provide some general information on your submission below. + + + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepView.axaml.cs new file mode 100644 index 000000000..6d05e97a1 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepView.axaml.cs @@ -0,0 +1,17 @@ +using Avalonia.Markup.Xaml; +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public partial class SpecificationsStepView : ReactiveUserControl +{ + public SpecificationsStepView() + { + InitializeComponent(); + } + + private void InitializeComponent() + { + AvaloniaXamlLoader.Load(this); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepViewModel.cs new file mode 100644 index 000000000..4e8a17c20 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SpecificationsStepViewModel.cs @@ -0,0 +1,119 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Reactive.Disposables; +using Artemis.UI.Extensions; +using Artemis.UI.Screens.Workshop.Entries; +using Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.Profile; +using Artemis.WebClient.Workshop; +using DynamicData; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public class SpecificationsStepViewModel : SubmissionViewModel +{ + private readonly Func _getEntrySpecificationsViewModel; + private EntrySpecificationsViewModel? _entrySpecificationsViewModel; + + public SpecificationsStepViewModel(Func getEntrySpecificationsViewModel) + { + _getEntrySpecificationsViewModel = getEntrySpecificationsViewModel; + + GoBack = ReactiveCommand.Create(ExecuteGoBack); + this.WhenActivated((CompositeDisposable d) => + { + DisplayName = $"{State.EntryType} Information"; + ApplyFromState(); + }); + } + + public EntrySpecificationsViewModel? EntrySpecificationsViewModel + { + get => _entrySpecificationsViewModel; + set => RaiseAndSetIfChanged(ref _entrySpecificationsViewModel, value); + } + + private void ExecuteGoBack() + { + // Apply what's there so far + ApplyToState(); + + switch (State.EntryType) + { + case EntryType.Layout: + break; + case EntryType.Plugin: + break; + case EntryType.Profile: + State.ChangeScreen(); + break; + default: + throw new ArgumentOutOfRangeException(); + } + } + + private void ExecuteContinue() + { + if (EntrySpecificationsViewModel == null || !EntrySpecificationsViewModel.ValidationContext.GetIsValid()) + return; + + ApplyToState(); + State.ChangeScreen(); + } + + private void ApplyFromState() + { + EntrySpecificationsViewModel viewModel = _getEntrySpecificationsViewModel(); + + // Basic fields + viewModel.Name = State.Name; + viewModel.Summary = State.Summary; + viewModel.Description = State.Description; + + // Tags + viewModel.Tags.Clear(); + viewModel.Tags.AddRange(State.Tags); + + // Categories + viewModel.PreselectedCategories = State.Categories; + + // Icon + if (State.Icon != null) + { + State.Icon.Seek(0, SeekOrigin.Begin); + viewModel.IconBitmap = BitmapExtensions.LoadAndResize(State.Icon, 128); + } + + EntrySpecificationsViewModel = viewModel; + Continue = ReactiveCommand.Create(ExecuteContinue, EntrySpecificationsViewModel.ValidationContext.Valid); + } + + private void ApplyToState() + { + if (EntrySpecificationsViewModel == null) + return; + + // Basic fields + State.Name = EntrySpecificationsViewModel.Name; + State.Summary = EntrySpecificationsViewModel.Summary; + State.Description = EntrySpecificationsViewModel.Description; + + // Categories and tasks + State.Categories = EntrySpecificationsViewModel.Categories.Where(c => c.IsSelected).Select(c => c.Id).ToList(); + State.Tags = new List(EntrySpecificationsViewModel.Tags); + + // Icon + State.Icon?.Dispose(); + if (EntrySpecificationsViewModel.IconBitmap != null) + { + State.Icon = new MemoryStream(); + EntrySpecificationsViewModel.IconBitmap.Save(State.Icon); + } + else + { + State.Icon = null; + } + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepView.axaml new file mode 100644 index 000000000..381f0ffb9 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepView.axaml @@ -0,0 +1,71 @@ + + + + + + + + + Ready to submit? + + + We have all the information we need, are you ready to submit the following to the workshop? + + + + + + + + + + + + + + by + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Artemis.UI/Screens/Workshop/Layout/LayoutListView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepView.axaml.cs similarity index 53% rename from src/Artemis.UI/Screens/Workshop/Layout/LayoutListView.axaml.cs rename to src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepView.axaml.cs index 9d604f4f6..ba64d1d7d 100644 --- a/src/Artemis.UI/Screens/Workshop/Layout/LayoutListView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepView.axaml.cs @@ -1,11 +1,11 @@ using Avalonia.Markup.Xaml; using Avalonia.ReactiveUI; -namespace Artemis.UI.Screens.Workshop.Layout; +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; -public partial class LayoutListView : ReactiveUserControl +public partial class SubmitStepView : ReactiveUserControl { - public LayoutListView() + public SubmitStepView() { InitializeComponent(); } diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepViewModel.cs new file mode 100644 index 000000000..8544c30ef --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/SubmitStepViewModel.cs @@ -0,0 +1,67 @@ +using System; +using System.Collections.ObjectModel; +using System.IO; +using System.Linq; +using System.Reactive.Disposables; +using System.Reactive.Linq; +using Artemis.UI.Screens.Workshop.Categories; +using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Services; +using Avalonia.Media.Imaging; +using IdentityModel; +using ReactiveUI; +using StrawberryShake; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public class SubmitStepViewModel : SubmissionViewModel +{ + private ReadOnlyObservableCollection? _categories; + private Bitmap? _iconBitmap; + + /// + public SubmitStepViewModel(IAuthenticationService authenticationService, IWorkshopClient workshopClient) + { + CurrentUser = authenticationService.Claims.FirstOrDefault(c => c.Type == JwtClaimTypes.Name)?.Value; + GoBack = ReactiveCommand.Create(() => State.ChangeScreen()); + Continue = ReactiveCommand.Create(() => State.ChangeScreen()); + + ContinueText = "Submit"; + + this.WhenActivated(d => + { + if (State.Icon != null) + { + State.Icon.Seek(0, SeekOrigin.Begin); + IconBitmap = new Bitmap(State.Icon); + IconBitmap.DisposeWith(d); + } + + Observable.FromAsync(workshopClient.GetCategories.ExecuteAsync).Subscribe(PopulateCategories).DisposeWith(d); + }); + } + + public Bitmap? IconBitmap + { + get => _iconBitmap; + set => RaiseAndSetIfChanged(ref _iconBitmap, value); + } + + public string? CurrentUser { get; } + + public ReadOnlyObservableCollection? Categories + { + get => _categories; + set => RaiseAndSetIfChanged(ref _categories, value); + } + + private void PopulateCategories(IOperationResult result) + { + if (result.Data == null) + Categories = null; + else + Categories = new ReadOnlyObservableCollection( + new ObservableCollection(result.Data.Categories.Where(c => State.Categories.Contains(c.Id)).Select(c => new CategoryViewModel(c))) + ); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepView.axaml new file mode 100644 index 000000000..e3b9e23c3 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepView.axaml @@ -0,0 +1,45 @@ + + + + + + + + + + Uploading your submission... + + + + + + + + All done! Hit finish to view your submission. + + + + 😢 + + Unfortunately something went wrong while uploading your submission. + + Hit finish to view your submission, from there you can try to upload a new release. + If this keeps occuring, hit us up on Discord + + + + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/Profile/ProfileListView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepView.axaml.cs similarity index 53% rename from src/Artemis.UI/Screens/Workshop/Profile/ProfileListView.axaml.cs rename to src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepView.axaml.cs index fb6c372c7..21d6bb439 100644 --- a/src/Artemis.UI/Screens/Workshop/Profile/ProfileListView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepView.axaml.cs @@ -1,11 +1,11 @@ using Avalonia.Markup.Xaml; using Avalonia.ReactiveUI; -namespace Artemis.UI.Screens.Workshop.Profile; +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; -public partial class ProfileListView : ReactiveUserControl +public partial class UploadStepView : ReactiveUserControl { - public ProfileListView() + public UploadStepView() { InitializeComponent(); } diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepViewModel.cs new file mode 100644 index 000000000..ab59d122f --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/UploadStepViewModel.cs @@ -0,0 +1,184 @@ +using System; +using System.Reactive.Disposables; +using System.Reactive.Linq; +using System.Threading; +using System.Threading.Tasks; +using Artemis.UI.Shared.Routing; +using Artemis.UI.Shared.Services; +using Artemis.UI.Shared.Utilities; +using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Exceptions; +using Artemis.WebClient.Workshop.Handlers.UploadHandlers; +using Artemis.WebClient.Workshop.Services; +using ReactiveUI; +using Serilog; +using StrawberryShake; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public class UploadStepViewModel : SubmissionViewModel +{ + private readonly ILogger _logger; + private readonly EntryUploadHandlerFactory _entryUploadHandlerFactory; + private readonly Progress _progress = new(); + private readonly ObservableAsPropertyHelper _progressIndeterminate; + private readonly ObservableAsPropertyHelper _progressPercentage; + private readonly IRouter _router; + private readonly IWindowService _windowService; + private readonly IWorkshopClient _workshopClient; + private readonly IWorkshopService _workshopService; + + private long? _entryId; + private bool _failed; + private bool _finished; + private bool _succeeded; + + /// + public UploadStepViewModel(ILogger logger, + IWorkshopClient workshopClient, + IWorkshopService workshopService, + EntryUploadHandlerFactory entryUploadHandlerFactory, + IWindowService windowService, + IRouter router) + { + _logger = logger; + _workshopClient = workshopClient; + _workshopService = workshopService; + _entryUploadHandlerFactory = entryUploadHandlerFactory; + _windowService = windowService; + _router = router; + + ShowGoBack = false; + ContinueText = "Finish"; + Continue = ReactiveCommand.CreateFromTask(ExecuteContinue, this.WhenAnyValue(vm => vm.Finished)); + + _progressPercentage = Observable.FromEventPattern(x => _progress.ProgressChanged += x, x => _progress.ProgressChanged -= x) + .Select(e => e.EventArgs.ProgressPercentage) + .ToProperty(this, vm => vm.ProgressPercentage); + _progressIndeterminate = Observable.FromEventPattern(x => _progress.ProgressChanged += x, x => _progress.ProgressChanged -= x) + .Select(e => e.EventArgs.ProgressPercentage == 0) + .ToProperty(this, vm => vm.ProgressIndeterminate); + + this.WhenActivated(d => Observable.FromAsync(ExecuteUpload).Subscribe().DisposeWith(d)); + } + + public int ProgressPercentage => _progressPercentage.Value; + public bool ProgressIndeterminate => _progressIndeterminate.Value; + + public bool Finished + { + get => _finished; + set => RaiseAndSetIfChanged(ref _finished, value); + } + + public bool Succeeded + { + get => _succeeded; + set => RaiseAndSetIfChanged(ref _succeeded, value); + } + + public bool Failed + { + get => _failed; + set => RaiseAndSetIfChanged(ref _failed, value); + } + + private async Task ExecuteUpload(CancellationToken cancellationToken) + { + // Use the existing entry or create a new one + _entryId = State.EntryId ?? await CreateEntry(cancellationToken); + + // If a new entry had to be created but that failed, stop here, CreateEntry will send the user back + if (_entryId == null) + return; + + try + { + IEntryUploadHandler uploadHandler = _entryUploadHandlerFactory.CreateHandler(State.EntryType); + EntryUploadResult uploadResult = await uploadHandler.CreateReleaseAsync(_entryId.Value, State.EntrySource!, _progress, cancellationToken); + if (!uploadResult.IsSuccess) + { + string? message = uploadResult.Message; + if (message != null) + message += "\r\n\r\n"; + else + message = ""; + message += "Your submission has still been saved, you may try to upload a new release"; + await _windowService.ShowConfirmContentDialog("Failed to upload workshop entry", message, "Close", null); + } + + Succeeded = true; + } + catch (Exception e) + { + _logger.Error(e, "Failed to upload submission for entry {EntryId}", _entryId); + + // Something went wrong when creating a release :c + // We'll keep the workshop entry so that the user can make changes and try again + Failed = true; + } + finally + { + Finished = true; + } + } + + private async Task CreateEntry(CancellationToken cancellationToken) + { + IOperationResult result = await _workshopClient.AddEntry.ExecuteAsync(new CreateEntryInput + { + EntryType = State.EntryType, + Name = State.Name, + Summary = State.Summary, + Description = State.Description, + Categories = State.Categories, + Tags = State.Tags + }, cancellationToken); + + long? entryId = result.Data?.AddEntry?.Id; + if (result.IsErrorResult() || entryId == null) + { + await _windowService.ShowConfirmContentDialog("Failed to create workshop entry", result.Errors.ToString() ?? "Not even an error message", "Close", null); + State.ChangeScreen(); + return null; + } + + if (cancellationToken.IsCancellationRequested) + { + State.ChangeScreen(); + return null; + } + + + if (State.Icon == null) + return entryId; + + // Upload image + try + { + ImageUploadResult imageUploadResult = await _workshopService.SetEntryIcon(entryId.Value, _progress, State.Icon, cancellationToken); + if (!imageUploadResult.IsSuccess) + throw new ArtemisWorkshopException(imageUploadResult.Message); + } + catch (Exception e) + { + // It's not critical if this fails + await _windowService.ShowConfirmContentDialog( + "Failed to upload icon", + "Your submission will continue, you can try upload a new image afterwards\r\n" + e.Message, + "Continue", + null + ); + } + + return entryId; + } + + private async Task ExecuteContinue() + { + State.Close(); + + if (_entryId != null) + await _router.Navigate($"workshop/library/submissions/{_entryId.Value}"); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepView.axaml new file mode 100644 index 000000000..17745e6c1 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepView.axaml @@ -0,0 +1,50 @@ + + + + + + + + Confirm email address + + Before you can continue, please confirm your email address. () + + You'll find the confirmation mail in your inbox. + + + Don't see an email? Check your spam box! + + + + + + + + + + + + PS: We take this step to avoid the workshop getting spammed with low quality content. + + + diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepView.axaml.cs new file mode 100644 index 000000000..a5899731e --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepView.axaml.cs @@ -0,0 +1,11 @@ +using Avalonia.ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public partial class ValidateEmailStepView : ReactiveUserControl +{ + public ValidateEmailStepView() + { + InitializeComponent(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepViewModel.cs new file mode 100644 index 000000000..a2a840421 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/ValidateEmailStepViewModel.cs @@ -0,0 +1,78 @@ +using System; +using System.Diagnostics; +using System.Linq; +using System.Reactive; +using System.Reactive.Disposables; +using System.Security.Claims; +using System.Threading; +using System.Threading.Tasks; +using Artemis.Core; +using Artemis.WebClient.Workshop; +using Artemis.WebClient.Workshop.Services; +using IdentityModel; +using ReactiveUI; +using Timer = System.Timers.Timer; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; + +public class ValidateEmailStepViewModel : SubmissionViewModel +{ + private readonly IAuthenticationService _authenticationService; + private ObservableAsPropertyHelper? _email; + + public ValidateEmailStepViewModel(IAuthenticationService authenticationService) + { + _authenticationService = authenticationService; + + Continue = ReactiveCommand.Create(ExecuteContinue); + Refresh = ReactiveCommand.CreateFromTask(ExecuteRefresh); + Resend = ReactiveCommand.Create(() => Utilities.OpenUrl(WorkshopConstants.AUTHORITY_URL + "/account/confirm/resend")); + + ShowGoBack = false; + ShowHeader = false; + + this.WhenActivated(d => + { + _email = authenticationService.GetClaim(JwtClaimTypes.Email).ToProperty(this, vm => vm.Email).DisposeWith(d); + + Timer updateTimer = new(TimeSpan.FromSeconds(15)); + updateTimer.Elapsed += (_, _) => Task.Run(Update); + updateTimer.Start(); + + updateTimer.DisposeWith(d); + }); + } + + public ReactiveCommand Refresh { get; } + public ReactiveCommand Resend { get; } + + public Claim? Email => _email?.Value; + + private async Task Update() + { + try + { + // Use the refresh token to login again, updating claims + await _authenticationService.AutoLogin(true); + + Claim? emailVerified = _authenticationService.Claims.FirstOrDefault(c => c.Type == JwtClaimTypes.EmailVerified); + if (emailVerified?.Value == "true") + ExecuteContinue(); + } + catch (Exception) + { + // ignored, meh + } + } + + private void ExecuteContinue() + { + State.ChangeScreen(); + } + + private async Task ExecuteRefresh(CancellationToken ct) + { + await Update(); + await Task.Delay(1000, ct); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml index 7341a2307..9f9538c38 100644 --- a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml @@ -2,10 +2,18 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" - mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450" - x:Class="Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.WelcomeStepView"> - - Welcome to the Workshop Submission Wizard 🧙 - Here we'll take you, step by step, through the process of uploading your submission to the workshop. + xmlns:steps="clr-namespace:Artemis.UI.Screens.Workshop.SubmissionWizard.Steps" + mc:Ignorable="d" d:DesignWidth="970" d:DesignHeight="900" + x:Class="Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.WelcomeStepView" + x:DataType="steps:WelcomeStepViewModel"> + + + Welcome to the Workshop Submission Wizard 🧙 + + + Here we'll take you, step by step, through the process of uploading your submission to the workshop. + + + - + \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml.cs index 9de30c623..cfeea65f8 100644 --- a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepView.axaml.cs @@ -1,4 +1,3 @@ -using Avalonia.Markup.Xaml; using Avalonia.ReactiveUI; namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; @@ -9,9 +8,4 @@ public partial class WelcomeStepView : ReactiveUserControl { InitializeComponent(); } - - private void InitializeComponent() - { - AvaloniaXamlLoader.Load(this); - } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepViewModel.cs index cf70f6410..87c96fb7d 100644 --- a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepViewModel.cs +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/Steps/WelcomeStepViewModel.cs @@ -1,22 +1,38 @@ -using System.Reactive; +using System.Linq; +using System.Threading.Tasks; +using Artemis.WebClient.Workshop.Services; +using IdentityModel; using ReactiveUI; namespace Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; public class WelcomeStepViewModel : SubmissionViewModel { - #region Overrides of SubmissionViewModel + private readonly IAuthenticationService _authenticationService; - /// - public override ReactiveCommand Continue { get; } - - /// - public override ReactiveCommand GoBack { get; } = null!; - - public WelcomeStepViewModel() + public WelcomeStepViewModel(IAuthenticationService authenticationService) { + _authenticationService = authenticationService; + + Continue = ReactiveCommand.CreateFromTask(ExecuteContinue); + ShowHeader = false; ShowGoBack = false; } - #endregion + private async Task ExecuteContinue() + { + bool loggedIn = await _authenticationService.AutoLogin(true); + + if (!loggedIn) + { + State.ChangeScreen(); + } + else + { + if (_authenticationService.Claims.Any(c => c.Type == JwtClaimTypes.EmailVerified && c.Value == "true")) + State.ChangeScreen(); + else + State.ChangeScreen(); + } + } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionViewModel.cs new file mode 100644 index 000000000..282b34590 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionViewModel.cs @@ -0,0 +1,53 @@ +using System.Reactive; +using Artemis.UI.Shared; +using ReactiveUI; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard; + +public abstract class SubmissionViewModel : ValidatableViewModelBase +{ + private ReactiveCommand? _continue; + private ReactiveCommand? _goBack; + private string _continueText = "Continue"; + private bool _showFinish; + private bool _showGoBack = true; + private bool _showHeader = true; + + public SubmissionWizardState State { get; set; } = null!; + + public ReactiveCommand? Continue + { + get => _continue; + set => RaiseAndSetIfChanged(ref _continue, value); + } + + public ReactiveCommand? GoBack + { + get => _goBack; + set => RaiseAndSetIfChanged(ref _goBack, value); + } + + public bool ShowHeader + { + get => _showHeader; + set => RaiseAndSetIfChanged(ref _showHeader, value); + } + + public bool ShowGoBack + { + get => _showGoBack; + set => RaiseAndSetIfChanged(ref _showGoBack, value); + } + + public bool ShowFinish + { + get => _showFinish; + set => RaiseAndSetIfChanged(ref _showFinish, value); + } + + public string ContinueText + { + get => _continueText; + set => RaiseAndSetIfChanged(ref _continueText, value); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardState.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardState.cs new file mode 100644 index 000000000..893896621 --- /dev/null +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardState.cs @@ -0,0 +1,62 @@ +using System; +using System.Collections.Generic; +using System.IO; +using Artemis.UI.Screens.Workshop.SubmissionWizard.Steps.Profile; +using Artemis.UI.Shared.Services; +using Artemis.WebClient.Workshop; +using DryIoc; + +namespace Artemis.UI.Screens.Workshop.SubmissionWizard; + +public class SubmissionWizardState +{ + private readonly IContainer _container; + private readonly IWindowService _windowService; + private readonly IWorkshopWizardViewModel _wizardViewModel; + + public SubmissionWizardState(IWorkshopWizardViewModel wizardViewModel, IContainer container, IWindowService windowService) + { + _wizardViewModel = wizardViewModel; + _container = container; + _windowService = windowService; + } + + public EntryType EntryType { get; set; } + public long? EntryId { get; set; } + + public string Name { get; set; } = string.Empty; + public Stream? Icon { get; set; } + public string Summary { get; set; } = string.Empty; + public string Description { get; set; } = string.Empty; + + public List Categories { get; set; } = new(); + public List Tags { get; set; } = new(); + public List Images { get; set; } = new(); + + public object? EntrySource { get; set; } + + public void ChangeScreen() where TSubmissionViewModel : SubmissionViewModel + { + try + { + _wizardViewModel.Screen = _container.Resolve(); + } + catch (Exception e) + { + _windowService.ShowExceptionDialog("Wizard screen failed to activate", e); + } + } + + public void Close() + { + _wizardViewModel.ShouldClose = true; + } + + public void StartForCurrentEntry() + { + if (EntryType == EntryType.Profile) + ChangeScreen(); + else + throw new NotImplementedException(); + } +} \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml index 18315e3fc..d64ed0199 100644 --- a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml @@ -12,12 +12,13 @@ Icon="/Assets/Images/Logo/application.ico" Title="Artemis | Workshop submission wizard" Width="1000" - Height="735" + Height="950" WindowStartupLocation="CenterOwner"> - + + @@ -44,9 +45,7 @@ - + diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml.cs index 58dade5ef..6c7ece92f 100644 --- a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml.cs +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardView.axaml.cs @@ -1,5 +1,6 @@ using System; using System.Reactive.Disposables; +using System.Reactive.Linq; using Artemis.UI.Shared; using Avalonia; using Avalonia.Threading; @@ -17,6 +18,7 @@ public partial class SubmissionWizardView : ReactiveAppWindow ViewModel.WhenAnyValue(vm => vm.Screen).Subscribe(Navigate).DisposeWith(d)); + this.WhenActivated(d => ViewModel.WhenAnyValue(vm => vm.ShouldClose).Where(c => c).Subscribe(_ => Close()).DisposeWith(d)); } private void Navigate(SubmissionViewModel viewModel) @@ -25,9 +27,9 @@ public partial class SubmissionWizardView : ReactiveAppWindow Frame.NavigateFromObject(viewModel)); } - catch (Exception) + catch (Exception e) { - // ignored + ViewModel?.WindowService.ShowExceptionDialog("Wizard screen failed to activate", e); } } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardViewModel.cs b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardViewModel.cs index fed2c3241..a18c63e71 100644 --- a/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardViewModel.cs +++ b/src/Artemis.UI/Screens/Workshop/SubmissionWizard/SubmissionWizardViewModel.cs @@ -1,47 +1,47 @@ -using System.Reactive; -using Artemis.UI.Screens.Workshop.CurrentUser; +using Artemis.UI.Screens.Workshop.CurrentUser; using Artemis.UI.Screens.Workshop.SubmissionWizard.Steps; using Artemis.UI.Shared; -using ReactiveUI; +using Artemis.UI.Shared.Services; +using DryIoc; namespace Artemis.UI.Screens.Workshop.SubmissionWizard; -public class SubmissionWizardViewModel : DialogViewModelBase +public class SubmissionWizardViewModel : ActivatableViewModelBase, IWorkshopWizardViewModel { + private readonly SubmissionWizardState _state; private SubmissionViewModel _screen; + private bool _shouldClose; - public SubmissionWizardViewModel(CurrentUserViewModel currentUserViewModel) + public SubmissionWizardViewModel(IContainer container, + IWindowService windowService, + CurrentUserViewModel currentUserViewModel, + WelcomeStepViewModel welcomeStepViewModel) { - _screen = new WelcomeStepViewModel(); + _state = new SubmissionWizardState(this, container, windowService); + _screen = welcomeStepViewModel; + _screen.State = _state; + + WindowService = windowService; CurrentUserViewModel = currentUserViewModel; + CurrentUserViewModel.AllowLogout = false; } + public IWindowService WindowService { get; } public CurrentUserViewModel CurrentUserViewModel { get; } public SubmissionViewModel Screen { get => _screen; - set => RaiseAndSetIfChanged(ref _screen, value); - } -} - -public abstract class SubmissionViewModel : ActivatableViewModelBase -{ - private bool _showFinish; - private bool _showGoBack = true; - - public abstract ReactiveCommand Continue { get; } - public abstract ReactiveCommand GoBack { get; } - - public bool ShowGoBack - { - get => _showGoBack; - set => RaiseAndSetIfChanged(ref _showGoBack, value); + set + { + value.State = _state; + RaiseAndSetIfChanged(ref _screen, value); + } } - public bool ShowFinish + public bool ShouldClose { - get => _showFinish; - set => RaiseAndSetIfChanged(ref _showFinish, value); + get => _shouldClose; + set => RaiseAndSetIfChanged(ref _shouldClose, value); } } \ No newline at end of file diff --git a/src/Artemis.UI/Screens/Workshop/WorkshopViewModel.cs b/src/Artemis.UI/Screens/Workshop/WorkshopViewModel.cs index f6b53a013..aa6fc9eb2 100644 --- a/src/Artemis.UI/Screens/Workshop/WorkshopViewModel.cs +++ b/src/Artemis.UI/Screens/Workshop/WorkshopViewModel.cs @@ -1,38 +1,18 @@ -using System; -using System.Threading; -using System.Threading.Tasks; -using Artemis.UI.Screens.Workshop.Home; +using Artemis.UI.Screens.Workshop.Home; using Artemis.UI.Screens.Workshop.Search; using Artemis.UI.Shared; using Artemis.UI.Shared.Routing; -using Artemis.WebClient.Workshop; namespace Artemis.UI.Screens.Workshop; -public class WorkshopViewModel : RoutableScreen, IMainScreenViewModel +public class WorkshopViewModel : RoutableHostScreen, IMainScreenViewModel { - private readonly SearchViewModel _searchViewModel; - public WorkshopViewModel(SearchViewModel searchViewModel, WorkshopHomeViewModel homeViewModel) { - _searchViewModel = searchViewModel; - TitleBarViewModel = searchViewModel; HomeViewModel = homeViewModel; } public ViewModelBase TitleBarViewModel { get; } public WorkshopHomeViewModel HomeViewModel { get; } - - /// - public override Task OnNavigating(NavigationArguments args, CancellationToken cancellationToken) - { - _searchViewModel.EntryType = Screen?.EntryType; - return Task.CompletedTask; - } -} - -public interface IWorkshopViewModel -{ - public EntryType? EntryType { get; } } \ No newline at end of file diff --git a/src/Artemis.UI/Services/DebugService.cs b/src/Artemis.UI/Services/DebugService.cs index 138bfa994..62b644f29 100644 --- a/src/Artemis.UI/Services/DebugService.cs +++ b/src/Artemis.UI/Services/DebugService.cs @@ -22,7 +22,8 @@ public class DebugService : IDebugService private void CreateDebugger() { - _debugViewModel = _windowService.ShowWindow(); + _windowService.ShowWindow(out DebugViewModel debugViewModel); + _debugViewModel = debugViewModel; } public void ClearDebugger() diff --git a/src/Artemis.UI/Services/Updating/ReleaseInstaller.cs b/src/Artemis.UI/Services/Updating/ReleaseInstaller.cs index 04ca4f74f..91a9a2c9c 100644 --- a/src/Artemis.UI/Services/Updating/ReleaseInstaller.cs +++ b/src/Artemis.UI/Services/Updating/ReleaseInstaller.cs @@ -9,6 +9,8 @@ using System.Threading.Tasks; using Artemis.Core; using Artemis.UI.Exceptions; using Artemis.UI.Extensions; +using Artemis.UI.Shared.Extensions; +using Artemis.UI.Shared.Utilities; using Artemis.WebClient.Updating; using Octodiff.Core; using Octodiff.Diagnostics; @@ -32,7 +34,7 @@ public class ReleaseInstaller : CorePropertyChanged private IGetReleaseById_PublishedRelease _release = null!; private IGetReleaseById_PublishedRelease_Artifacts _artifact = null!; - private Progress _stepProgress = new(); + private Progress _stepProgress = new(); private string _status = string.Empty; private float _floatProgress; @@ -69,9 +71,7 @@ public class ReleaseInstaller : CorePropertyChanged public async Task InstallAsync(CancellationToken cancellationToken) { - _stepProgress = new Progress(); - - ((IProgress) _progress).Report(0); + _stepProgress = new Progress(); Status = "Retrieving details"; _logger.Information("Retrieving details for release {ReleaseId}", _releaseId); @@ -99,7 +99,7 @@ public class ReleaseInstaller : CorePropertyChanged { // 10 - 50% _stepProgress.ProgressChanged += StepProgressOnProgressChanged; - void StepProgressOnProgressChanged(object? sender, float e) => ((IProgress) _progress).Report(10f + e * 0.4f); + void StepProgressOnProgressChanged(object? sender, StreamProgress e) => ((IProgress) _progress).Report(10f + e.ProgressPercentage * 0.4f); Status = "Downloading..."; await using MemoryStream stream = new(); @@ -113,7 +113,7 @@ public class ReleaseInstaller : CorePropertyChanged { // 50 - 60% _stepProgress.ProgressChanged += StepProgressOnProgressChanged; - void StepProgressOnProgressChanged(object? sender, float e) => ((IProgress) _progress).Report(50f + e * 0.1f); + void StepProgressOnProgressChanged(object? sender, StreamProgress e) => ((IProgress) _progress).Report(50f + e.ProgressPercentage * 0.1f); Status = "Patching..."; await using FileStream newFileStream = new(Path.Combine(Constants.UpdatingFolder, $"{_release.Version}.zip"), FileMode.Create, FileAccess.ReadWrite, FileShare.Read); @@ -139,7 +139,7 @@ public class ReleaseInstaller : CorePropertyChanged { // 10 - 60% _stepProgress.ProgressChanged += StepProgressOnProgressChanged; - void StepProgressOnProgressChanged(object? sender, float e) => ((IProgress) _progress).Report(10f + e * 0.5f); + void StepProgressOnProgressChanged(object? sender, StreamProgress e) => ((IProgress) _progress).Report(10f + e.ProgressPercentage * 0.5f); Status = "Downloading..."; await using FileStream stream = new(Path.Combine(Constants.UpdatingFolder, $"{_release.Version}.zip"), FileMode.Create, FileAccess.ReadWrite, FileShare.Read); @@ -155,7 +155,7 @@ public class ReleaseInstaller : CorePropertyChanged { // 60 - 100% _stepProgress.ProgressChanged += StepProgressOnProgressChanged; - void StepProgressOnProgressChanged(object? sender, float e) => ((IProgress) _progress).Report(60f + e * 0.4f); + void StepProgressOnProgressChanged(object? sender, StreamProgress e) => ((IProgress) _progress).Report(60f + e.ProgressPercentage * 0.4f); Status = "Extracting..."; // Ensure the directory is empty diff --git a/src/Artemis.UI/Styles/Artemis.axaml b/src/Artemis.UI/Styles/Artemis.axaml index 3b700ecdc..55109b2d3 100644 --- a/src/Artemis.UI/Styles/Artemis.axaml +++ b/src/Artemis.UI/Styles/Artemis.axaml @@ -1,16 +1,32 @@  + xmlns:avalonia="clr-namespace:Material.Icons.Avalonia;assembly=Material.Icons.Avalonia" + xmlns:aedit="using:AvaloniaEdit" + xmlns:aedit2="using:AvaloniaEdit.Editing"> + + + + + + + + + avares://Artemis.UI/Assets/Fonts#Roboto Mono diff --git a/src/Artemis.UI/Styles/Markdown.axaml b/src/Artemis.UI/Styles/Markdown.axaml index 5374e71b0..6f03f8291 100644 --- a/src/Artemis.UI/Styles/Markdown.axaml +++ b/src/Artemis.UI/Styles/Markdown.axaml @@ -2,6 +2,9 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:avalonia="clr-namespace:Markdown.Avalonia;assembly=Markdown.Avalonia" xmlns:ctxt="clr-namespace:ColorTextBlock.Avalonia;assembly=ColorTextBlock.Avalonia"> + diff --git a/src/Artemis.VisualScripting/Artemis.VisualScripting.csproj b/src/Artemis.VisualScripting/Artemis.VisualScripting.csproj index 913205f9b..379f1ad87 100644 --- a/src/Artemis.VisualScripting/Artemis.VisualScripting.csproj +++ b/src/Artemis.VisualScripting/Artemis.VisualScripting.csproj @@ -23,12 +23,4 @@ - - - - HotkeyPressNodeCustomView.axaml - Code - - - diff --git a/src/Artemis.VisualScripting/Nodes/Static/Screens/DisplayValueNodeCustomView.axaml b/src/Artemis.VisualScripting/Nodes/Static/Screens/DisplayValueNodeCustomView.axaml index 8ddaf5f77..8831dd1e9 100644 --- a/src/Artemis.VisualScripting/Nodes/Static/Screens/DisplayValueNodeCustomView.axaml +++ b/src/Artemis.VisualScripting/Nodes/Static/Screens/DisplayValueNodeCustomView.axaml @@ -25,7 +25,7 @@ Text="{CompiledBinding Converter={StaticResource SKColorToStringConverter}}" VerticalAlignment="Center" HorizontalAlignment="Stretch" - FontFamily="Consolas"/> + FontFamily="{StaticResource RobotoMono}"/> - + diff --git a/src/Artemis.WebClient.Workshop/Artemis.WebClient.Workshop.csproj b/src/Artemis.WebClient.Workshop/Artemis.WebClient.Workshop.csproj index e09fefe42..391d5c4b5 100644 --- a/src/Artemis.WebClient.Workshop/Artemis.WebClient.Workshop.csproj +++ b/src/Artemis.WebClient.Workshop/Artemis.WebClient.Workshop.csproj @@ -25,6 +25,7 @@ + @@ -34,5 +35,21 @@ MSBuild:GenerateGraphQLCode + + MSBuild:GenerateGraphQLCode + + + MSBuild:GenerateGraphQLCode + + + MSBuild:GenerateGraphQLCode + + + MSBuild:GenerateGraphQLCode + + + + + diff --git a/src/Artemis.WebClient.Workshop/DryIoc/ContainerExtensions.cs b/src/Artemis.WebClient.Workshop/DryIoc/ContainerExtensions.cs index bdcdbede6..ddadc110e 100644 --- a/src/Artemis.WebClient.Workshop/DryIoc/ContainerExtensions.cs +++ b/src/Artemis.WebClient.Workshop/DryIoc/ContainerExtensions.cs @@ -1,5 +1,10 @@ +using System.Reflection; +using Artemis.WebClient.Workshop.Extensions; +using Artemis.WebClient.Workshop.Handlers.InstallationHandlers; +using Artemis.WebClient.Workshop.Handlers.UploadHandlers; using Artemis.WebClient.Workshop.Repositories; using Artemis.WebClient.Workshop.Services; +using Artemis.WebClient.Workshop.State; using DryIoc; using DryIoc.Microsoft.DependencyInjection; using IdentityModel.Client; @@ -18,11 +23,17 @@ public static class ContainerExtensions /// The builder building the current container public static void RegisterWorkshopClient(this IContainer container) { + Assembly[] workshopAssembly = {typeof(WorkshopConstants).Assembly}; + ServiceCollection serviceCollection = new(); serviceCollection .AddHttpClient() .AddWorkshopClient() + .AddHttpMessageHandler() .ConfigureHttpClient(client => client.BaseAddress = new Uri(WorkshopConstants.WORKSHOP_URL + "/graphql")); + serviceCollection.AddHttpClient(WorkshopConstants.WORKSHOP_CLIENT_NAME) + .AddHttpMessageHandler() + .ConfigureHttpClient(client => client.BaseAddress = new Uri(WorkshopConstants.WORKSHOP_URL)); serviceCollection.AddSingleton(r => { @@ -34,5 +45,10 @@ public static class ContainerExtensions container.Register(Reuse.Singleton); container.Register(Reuse.Singleton); + container.Register(Reuse.Singleton); + + container.Register(Reuse.Transient); + container.RegisterMany(workshopAssembly, type => type.IsAssignableTo(), Reuse.Transient); + container.RegisterMany(workshopAssembly, type => type.IsAssignableTo(), Reuse.Transient); } } \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Entities/Release.cs b/src/Artemis.WebClient.Workshop/Entities/Release.cs new file mode 100644 index 000000000..ace9fb21f --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Entities/Release.cs @@ -0,0 +1,22 @@ +using System.ComponentModel.DataAnnotations; + +namespace Artemis.WebClient.Workshop.Entities; + +public class Release +{ + public long Id { get; set; } + + [MaxLength(64)] + public string Version { get; set; } = string.Empty; + + public DateTimeOffset CreatedAt { get; set; } + + public long Downloads { get; set; } + + public long DownloadSize { get; set; } + + [MaxLength(32)] + public string? Md5Hash { get; set; } + + public long EntryId { get; set; } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Exceptions/ArtemisWebClientException.cs b/src/Artemis.WebClient.Workshop/Exceptions/ArtemisWebClientException.cs index 1bd2af386..a5400594c 100644 --- a/src/Artemis.WebClient.Workshop/Exceptions/ArtemisWebClientException.cs +++ b/src/Artemis.WebClient.Workshop/Exceptions/ArtemisWebClientException.cs @@ -1,6 +1,4 @@ -using System; - -namespace Artemis.Core; +namespace Artemis.WebClient.Workshop.Exceptions; /// /// An exception thrown when a web client related error occurs diff --git a/src/Artemis.WebClient.Workshop/Exceptions/ArtemisWorkshopException.cs b/src/Artemis.WebClient.Workshop/Exceptions/ArtemisWorkshopException.cs new file mode 100644 index 000000000..968526086 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Exceptions/ArtemisWorkshopException.cs @@ -0,0 +1,22 @@ +namespace Artemis.WebClient.Workshop.Exceptions; + +/// +/// An exception thrown when a workshop related error occurs +/// +public class ArtemisWorkshopException : Exception +{ + /// + public ArtemisWorkshopException() + { + } + + /// + public ArtemisWorkshopException(string? message) : base(message) + { + } + + /// + public ArtemisWorkshopException(string? message, Exception? innerException) : base(message, innerException) + { + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Extensions/ClientBuilderExtensions.cs b/src/Artemis.WebClient.Workshop/Extensions/ClientBuilderExtensions.cs new file mode 100644 index 000000000..6db06795f --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Extensions/ClientBuilderExtensions.cs @@ -0,0 +1,18 @@ +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Http; +using StrawberryShake; + +namespace Artemis.WebClient.Workshop.Extensions; + +public static class ClientBuilderExtensions +{ + public static IClientBuilder AddHttpMessageHandler(this IClientBuilder builder) where THandler : DelegatingHandler where T : IStoreAccessor + { + builder.Services.Configure( + builder.ClientName, + options => options.HttpMessageHandlerBuilderActions.Add(b => b.AdditionalHandlers.Add(b.Services.GetRequiredService())) + ); + + return builder; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryInstallResult.cs b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryInstallResult.cs new file mode 100644 index 000000000..aaf56859e --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryInstallResult.cs @@ -0,0 +1,26 @@ +namespace Artemis.WebClient.Workshop.Handlers.InstallationHandlers; + +public class EntryInstallResult +{ + public bool IsSuccess { get; set; } + public string? Message { get; set; } + public object? Result { get; set; } + + public static EntryInstallResult FromFailure(string? message) + { + return new EntryInstallResult + { + IsSuccess = false, + Message = message + }; + } + + public static EntryInstallResult FromSuccess(object installationResult) + { + return new EntryInstallResult + { + IsSuccess = true, + Result = installationResult + }; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryInstallationHandlerFactory.cs b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryInstallationHandlerFactory.cs new file mode 100644 index 000000000..ec9c63566 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryInstallationHandlerFactory.cs @@ -0,0 +1,23 @@ +using Artemis.WebClient.Workshop.Handlers.InstallationHandlers.Implementations; +using DryIoc; + +namespace Artemis.WebClient.Workshop.Handlers.InstallationHandlers; + +public class EntryInstallationHandlerFactory +{ + private readonly IContainer _container; + + public EntryInstallationHandlerFactory(IContainer container) + { + _container = container; + } + + public IEntryInstallationHandler CreateHandler(EntryType entryType) + { + return entryType switch + { + EntryType.Profile => _container.Resolve(), + _ => throw new NotSupportedException($"EntryType '{entryType}' is not supported.") + }; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryUninstallResult.cs b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryUninstallResult.cs new file mode 100644 index 000000000..6d28edf9a --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/EntryUninstallResult.cs @@ -0,0 +1,24 @@ +namespace Artemis.WebClient.Workshop.Handlers.InstallationHandlers; + +public class EntryUninstallResult +{ + public bool IsSuccess { get; set; } + public string? Message { get; set; } + + public static EntryUninstallResult FromFailure(string? message) + { + return new EntryUninstallResult + { + IsSuccess = false, + Message = message + }; + } + + public static EntryUninstallResult FromSuccess() + { + return new EntryUninstallResult + { + IsSuccess = true + }; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/IEntryInstallationHandler.cs b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/IEntryInstallationHandler.cs new file mode 100644 index 000000000..d8e7ba951 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/IEntryInstallationHandler.cs @@ -0,0 +1,10 @@ +using Artemis.UI.Shared.Utilities; +using Artemis.WebClient.Workshop.Services; + +namespace Artemis.WebClient.Workshop.Handlers.InstallationHandlers; + +public interface IEntryInstallationHandler +{ + Task InstallAsync(IGetEntryById_Entry entry, long releaseId, Progress progress, CancellationToken cancellationToken); + Task UninstallAsync(InstalledEntry installedEntry, CancellationToken cancellationToken); +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/Implementations/ProfileEntryInstallationHandler.cs b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/Implementations/ProfileEntryInstallationHandler.cs new file mode 100644 index 000000000..f3473f66a --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/InstallationHandlers/Implementations/ProfileEntryInstallationHandler.cs @@ -0,0 +1,99 @@ +using Artemis.Core; +using Artemis.Core.Services; +using Artemis.UI.Shared.Extensions; +using Artemis.UI.Shared.Utilities; +using Artemis.WebClient.Workshop.Services; + +namespace Artemis.WebClient.Workshop.Handlers.InstallationHandlers.Implementations; + +public class ProfileEntryInstallationHandler : IEntryInstallationHandler +{ + private readonly IHttpClientFactory _httpClientFactory; + private readonly IProfileService _profileService; + private readonly IWorkshopService _workshopService; + + public ProfileEntryInstallationHandler(IHttpClientFactory httpClientFactory, IProfileService profileService, IWorkshopService workshopService) + { + _httpClientFactory = httpClientFactory; + _profileService = profileService; + _workshopService = workshopService; + } + + public async Task InstallAsync(IGetEntryById_Entry entry, long releaseId, Progress progress, CancellationToken cancellationToken) + { + using MemoryStream stream = new(); + + // Download the provided release + try + { + HttpClient client = _httpClientFactory.CreateClient(WorkshopConstants.WORKSHOP_CLIENT_NAME); + await client.DownloadDataAsync($"releases/download/{releaseId}", stream, progress, cancellationToken); + } + catch (Exception e) + { + return EntryInstallResult.FromFailure(e.Message); + } + + // Find existing installation to potentially replace the profile + InstalledEntry? installedEntry = _workshopService.GetInstalledEntry(entry); + if (installedEntry != null && Guid.TryParse(installedEntry.LocalReference, out Guid profileId)) + { + ProfileConfiguration? existing = _profileService.ProfileCategories.SelectMany(c => c.ProfileConfigurations).FirstOrDefault(c => c.ProfileId == profileId); + if (existing != null) + { + ProfileConfiguration overwritten = await _profileService.OverwriteProfile(stream, existing); + installedEntry.LocalReference = overwritten.ProfileId.ToString(); + + // Update the release and return the profile configuration + UpdateRelease(releaseId, installedEntry); + return EntryInstallResult.FromSuccess(overwritten); + } + } + + // Ensure there is an installed entry + installedEntry ??= _workshopService.CreateInstalledEntry(entry); + + // Add the profile as a fresh import + ProfileCategory category = _profileService.ProfileCategories.FirstOrDefault(c => c.Name == "Workshop") ?? _profileService.CreateProfileCategory("Workshop", true); + ProfileConfiguration imported = await _profileService.ImportProfile(stream, category, true, true, null); + installedEntry.LocalReference = imported.ProfileId.ToString(); + + // Update the release and return the profile configuration + UpdateRelease(releaseId, installedEntry); + return EntryInstallResult.FromSuccess(imported); + } + + public async Task UninstallAsync(InstalledEntry installedEntry, CancellationToken cancellationToken) + { + if (!Guid.TryParse(installedEntry.LocalReference, out Guid profileId)) + return EntryUninstallResult.FromFailure("Local reference does not contain a GUID"); + + return await Task.Run(() => + { + try + { + // Find the profile if still there + ProfileConfiguration? profile = _profileService.ProfileConfigurations.FirstOrDefault(c => c.ProfileId == profileId); + if (profile != null) + _profileService.DeleteProfile(profile); + + // Remove the release + _workshopService.RemoveInstalledEntry(installedEntry); + } + catch (Exception e) + { + return EntryUninstallResult.FromFailure(e.Message); + } + + return EntryUninstallResult.FromSuccess(); + }, cancellationToken); + } + + private void UpdateRelease(long releaseId, InstalledEntry installedEntry) + { + installedEntry.ReleaseId = releaseId; + installedEntry.ReleaseVersion = "TODO"; + installedEntry.InstalledAt = DateTimeOffset.UtcNow; + _workshopService.SaveInstalledEntry(installedEntry); + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/EntryUploadHandlerFactory.cs b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/EntryUploadHandlerFactory.cs new file mode 100644 index 000000000..17dc5245c --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/EntryUploadHandlerFactory.cs @@ -0,0 +1,24 @@ +using Artemis.WebClient.Workshop.Handlers.UploadHandlers.Implementations; +using DryIoc; + +namespace Artemis.WebClient.Workshop.Handlers.UploadHandlers; + +public class EntryUploadHandlerFactory +{ + private readonly IContainer _container; + + public EntryUploadHandlerFactory(IContainer container) + { + _container = container; + } + + public IEntryUploadHandler CreateHandler(EntryType entryType) + { + return entryType switch + { + EntryType.Profile => _container.Resolve(), + EntryType.Layout => _container.Resolve(), + _ => throw new NotSupportedException($"EntryType '{entryType}' is not supported.") + }; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/EntryUploadResult.cs b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/EntryUploadResult.cs new file mode 100644 index 000000000..f55a3dd77 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/EntryUploadResult.cs @@ -0,0 +1,28 @@ +using Artemis.WebClient.Workshop.Entities; + +namespace Artemis.WebClient.Workshop.Handlers.UploadHandlers; + +public class EntryUploadResult +{ + public bool IsSuccess { get; set; } + public string? Message { get; set; } + public Release? Release { get; set; } + + public static EntryUploadResult FromFailure(string? message) + { + return new EntryUploadResult + { + IsSuccess = false, + Message = message + }; + } + + public static EntryUploadResult FromSuccess(Release release) + { + return new EntryUploadResult + { + IsSuccess = true, + Release = release + }; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/IEntryUploadHandler.cs b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/IEntryUploadHandler.cs new file mode 100644 index 000000000..88ff68896 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/IEntryUploadHandler.cs @@ -0,0 +1,8 @@ +using Artemis.UI.Shared.Utilities; + +namespace Artemis.WebClient.Workshop.Handlers.UploadHandlers; + +public interface IEntryUploadHandler +{ + Task CreateReleaseAsync(long entryId, object file, Progress progress, CancellationToken cancellationToken); +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/ImageUploadResult.cs b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/ImageUploadResult.cs new file mode 100644 index 000000000..7ebc08d2f --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/ImageUploadResult.cs @@ -0,0 +1,21 @@ +namespace Artemis.WebClient.Workshop.Handlers.UploadHandlers; + +public class ImageUploadResult +{ + public bool IsSuccess { get; set; } + public string? Message { get; set; } + + public static ImageUploadResult FromFailure(string? message) + { + return new ImageUploadResult + { + IsSuccess = false, + Message = message + }; + } + + public static ImageUploadResult FromSuccess() + { + return new ImageUploadResult {IsSuccess = true}; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/Implementations/LayoutEntryUploadHandler.cs b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/Implementations/LayoutEntryUploadHandler.cs new file mode 100644 index 000000000..0b16fae3d --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/Implementations/LayoutEntryUploadHandler.cs @@ -0,0 +1,12 @@ +using Artemis.UI.Shared.Utilities; + +namespace Artemis.WebClient.Workshop.Handlers.UploadHandlers.Implementations; + +public class LayoutEntryUploadHandler : IEntryUploadHandler +{ + /// + public async Task CreateReleaseAsync(long entryId, object file, Progress progress, CancellationToken cancellationToken) + { + throw new NotImplementedException(); + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/Implementations/ProfileEntryUploadHandler.cs b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/Implementations/ProfileEntryUploadHandler.cs new file mode 100644 index 000000000..4ec607d85 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Handlers/UploadHandlers/Implementations/ProfileEntryUploadHandler.cs @@ -0,0 +1,46 @@ +using System.Net.Http.Headers; +using Artemis.Core; +using Artemis.Core.Services; +using Artemis.UI.Shared.Utilities; +using Artemis.WebClient.Workshop.Entities; +using Newtonsoft.Json; + +namespace Artemis.WebClient.Workshop.Handlers.UploadHandlers.Implementations; + +public class ProfileEntryUploadHandler : IEntryUploadHandler +{ + private readonly IHttpClientFactory _httpClientFactory; + private readonly IProfileService _profileService; + + public ProfileEntryUploadHandler(IHttpClientFactory httpClientFactory, IProfileService profileService) + { + _httpClientFactory = httpClientFactory; + _profileService = profileService; + } + + /// + public async Task CreateReleaseAsync(long entryId, object file, Progress progress, CancellationToken cancellationToken) + { + if (file is not ProfileConfiguration profileConfiguration) + throw new InvalidOperationException("Can only create releases for profile configurations"); + + await using Stream archiveStream = await _profileService.ExportProfile(profileConfiguration); + + // Submit the archive + HttpClient client = _httpClientFactory.CreateClient(WorkshopConstants.WORKSHOP_CLIENT_NAME); + + // Construct the request + MultipartFormDataContent content = new(); + ProgressableStreamContent streamContent = new(archiveStream, progress); + streamContent.Headers.ContentType = new MediaTypeHeaderValue("application/zip"); + content.Add(streamContent, "file", "file.zip"); + + // Submit + HttpResponseMessage response = await client.PostAsync("releases/upload/" + entryId, content, cancellationToken); + if (!response.IsSuccessStatusCode) + return EntryUploadResult.FromFailure($"{response.StatusCode} - {await response.Content.ReadAsStringAsync(cancellationToken)}"); + + Release? release = JsonConvert.DeserializeObject(await response.Content.ReadAsStringAsync(cancellationToken)); + return release != null ? EntryUploadResult.FromSuccess(release) : EntryUploadResult.FromFailure("Failed to deserialize response"); + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Queries/CreateEntry.graphql b/src/Artemis.WebClient.Workshop/Queries/CreateEntry.graphql new file mode 100644 index 000000000..7191bc704 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Queries/CreateEntry.graphql @@ -0,0 +1,5 @@ +mutation AddEntry ($input: CreateEntryInput!) { + addEntry(input: $input) { + id + } +} diff --git a/src/Artemis.WebClient.Workshop/Queries/GetCategories.graphql b/src/Artemis.WebClient.Workshop/Queries/GetCategories.graphql index d39be92f2..864caa217 100644 --- a/src/Artemis.WebClient.Workshop/Queries/GetCategories.graphql +++ b/src/Artemis.WebClient.Workshop/Queries/GetCategories.graphql @@ -1,5 +1,5 @@ query GetCategories { - categories { + categories(order: {name: ASC}) { id name icon diff --git a/src/Artemis.WebClient.Workshop/Queries/GetEntries.graphql b/src/Artemis.WebClient.Workshop/Queries/GetEntries.graphql index c7e639c31..e5e0648be 100644 --- a/src/Artemis.WebClient.Workshop/Queries/GetEntries.graphql +++ b/src/Artemis.WebClient.Workshop/Queries/GetEntries.graphql @@ -1,5 +1,5 @@ query GetEntries($filter: EntryFilterInput $skip: Int $take: Int) { - entries(where: $filter skip: $skip take: $take) { + entries(where: $filter skip: $skip take: $take, order: {createdAt: DESC}) { totalCount items { id @@ -8,7 +8,7 @@ query GetEntries($filter: EntryFilterInput $skip: Int $take: Int) { summary entryType downloads - createdAt + createdAt categories { name icon diff --git a/src/Artemis.WebClient.Workshop/Queries/GetEntryById.graphql b/src/Artemis.WebClient.Workshop/Queries/GetEntryById.graphql index e5ba4a96d..932ccef30 100644 --- a/src/Artemis.WebClient.Workshop/Queries/GetEntryById.graphql +++ b/src/Artemis.WebClient.Workshop/Queries/GetEntryById.graphql @@ -1,7 +1,23 @@ -query GetEntryById($id: UUID!) { +query GetEntryById($id: Long!) { entry(id: $id) { + id author name + summary entryType + downloads + createdAt + description + categories { + name + icon + } + latestRelease { + id + version + downloadSize + md5Hash + createdAt + } } } \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Queries/GetSubmittedEntries.graphql b/src/Artemis.WebClient.Workshop/Queries/GetSubmittedEntries.graphql new file mode 100644 index 000000000..d4199e074 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Queries/GetSubmittedEntries.graphql @@ -0,0 +1,14 @@ +query GetSubmittedEntries($filter: EntryFilterInput) { + submittedEntries(where: $filter order: {createdAt: DESC}) { + id + name + summary + entryType + downloads + createdAt + categories { + name + icon + } + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Queries/GetSubmittedEntryById.graphql b/src/Artemis.WebClient.Workshop/Queries/GetSubmittedEntryById.graphql new file mode 100644 index 000000000..ea7b0bab3 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Queries/GetSubmittedEntryById.graphql @@ -0,0 +1,17 @@ +query GetSubmittedEntryById($id: Long!) { + entry(id: $id) { + id + name + summary + entryType + downloads + createdAt + description + categories { + id + } + tags { + name + } + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Queries/RemoveEntry.graphql b/src/Artemis.WebClient.Workshop/Queries/RemoveEntry.graphql new file mode 100644 index 000000000..983fbe0d9 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Queries/RemoveEntry.graphql @@ -0,0 +1,5 @@ +mutation RemoveEntry ($id: Long!) { + removeEntry(id: $id) { + id + } +} diff --git a/src/Artemis.WebClient.Workshop/Queries/SearchEntries.graphql b/src/Artemis.WebClient.Workshop/Queries/SearchEntries.graphql index 620aeb70d..27e4eabbd 100644 --- a/src/Artemis.WebClient.Workshop/Queries/SearchEntries.graphql +++ b/src/Artemis.WebClient.Workshop/Queries/SearchEntries.graphql @@ -4,6 +4,7 @@ query SearchEntries($input: String! $type: EntryType) { name summary entryType + author categories { id name diff --git a/src/Artemis.WebClient.Workshop/Queries/UpdateEntry.graphql b/src/Artemis.WebClient.Workshop/Queries/UpdateEntry.graphql new file mode 100644 index 000000000..fe53ec3e5 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Queries/UpdateEntry.graphql @@ -0,0 +1,5 @@ +mutation UpdateEntry ($input: UpdateEntryInput!) { + updateEntry(input: $input) { + id + } +} diff --git a/src/Artemis.WebClient.Workshop/Services/AccessToken.cs b/src/Artemis.WebClient.Workshop/Services/AccessToken.cs index f67dd9395..7f4f7ba04 100644 --- a/src/Artemis.WebClient.Workshop/Services/AccessToken.cs +++ b/src/Artemis.WebClient.Workshop/Services/AccessToken.cs @@ -1,4 +1,5 @@ using Artemis.Core; +using Artemis.WebClient.Workshop.Exceptions; using IdentityModel.Client; namespace Artemis.WebClient.Workshop.Services; diff --git a/src/Artemis.WebClient.Workshop/Services/AuthenticationDelegatingHandler.cs b/src/Artemis.WebClient.Workshop/Services/AuthenticationDelegatingHandler.cs new file mode 100644 index 000000000..3adda3145 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Services/AuthenticationDelegatingHandler.cs @@ -0,0 +1,22 @@ +using System.Net.Http.Headers; + +namespace Artemis.WebClient.Workshop.Services; + +public class AuthenticationDelegatingHandler : DelegatingHandler +{ + private readonly IAuthenticationService _authenticationService; + + /// + public AuthenticationDelegatingHandler(IAuthenticationService authenticationService) + { + _authenticationService = authenticationService; + } + + protected override async Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) + { + string? token = await _authenticationService.GetBearer(); + if (token != null) + request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token); + return await base.SendAsync(request, cancellationToken); + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Services/AuthenticationService.cs b/src/Artemis.WebClient.Workshop/Services/AuthenticationService.cs new file mode 100644 index 000000000..5f47d2567 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Services/AuthenticationService.cs @@ -0,0 +1,296 @@ +using System.Collections.ObjectModel; +using System.IdentityModel.Tokens.Jwt; +using System.Net; +using System.Reactive.Linq; +using System.Reactive.Subjects; +using System.Security.Claims; +using System.Security.Cryptography; +using System.Text; +using Artemis.Core; +using Artemis.WebClient.Workshop.Exceptions; +using Artemis.WebClient.Workshop.Repositories; +using DynamicData; +using IdentityModel; +using IdentityModel.Client; +using Serilog; + +namespace Artemis.WebClient.Workshop.Services; + +internal class AuthenticationService : CorePropertyChanged, IAuthenticationService +{ + private const string CLIENT_ID = "artemis.desktop"; + private readonly IAuthenticationRepository _authenticationRepository; + private readonly SemaphoreSlim _authLock = new(1); + private readonly SourceList _claims; + + private readonly IDiscoveryCache _discoveryCache; + private readonly ILogger _logger; + private readonly IHttpClientFactory _httpClientFactory; + private readonly BehaviorSubject _isLoggedInSubject = new(false); + + private AuthenticationToken? _token; + private bool _noStoredRefreshToken; + + public AuthenticationService(ILogger logger, IHttpClientFactory httpClientFactory, IDiscoveryCache discoveryCache, IAuthenticationRepository authenticationRepository) + { + _logger = logger; + _httpClientFactory = httpClientFactory; + _discoveryCache = discoveryCache; + _authenticationRepository = authenticationRepository; + + _claims = new SourceList(); + _claims.Connect().Bind(out ReadOnlyObservableCollection claims).Subscribe(); + Claims = claims; + } + + private async Task GetDiscovery() + { + DiscoveryDocumentResponse disco = await _discoveryCache.GetAsync(); + if (disco.IsError) + throw new ArtemisWebClientException("Failed to retrieve discovery document: " + disco.Error); + + return disco; + } + + private void SetCurrentUser(TokenResponse response) + { + _token = new AuthenticationToken(response); + SetStoredRefreshToken(_token.RefreshToken); + + JwtSecurityTokenHandler handler = new(); + JwtSecurityToken? token = handler.ReadJwtToken(response.IdentityToken); + if (token == null) + throw new ArtemisWebClientException("Failed to read JWT token"); + + _claims.Edit(c => + { + c.Clear(); + c.AddRange(token.Claims); + }); + + _isLoggedInSubject.OnNext(true); + } + + private async Task UseRefreshToken(string refreshToken) + { + try + { + DiscoveryDocumentResponse disco = await GetDiscovery(); + HttpClient client = _httpClientFactory.CreateClient(); + TokenResponse response = await client.RequestRefreshTokenAsync(new RefreshTokenRequest + { + Address = disco.TokenEndpoint, + ClientId = CLIENT_ID, + RefreshToken = refreshToken + }); + + if (response.IsError) + { + if (response.Error is OidcConstants.TokenErrors.ExpiredToken or OidcConstants.TokenErrors.InvalidGrant) + { + SetStoredRefreshToken(null); + return false; + } + + throw new ArtemisWebClientException("Failed to request refresh token: " + response.Error); + } + + SetCurrentUser(response); + return true; + } + catch (Exception e) + { + _logger.Error(e, "Failed to use refresh token"); + SetStoredRefreshToken(null); + return false; + } + } + + private static byte[] HashSha256(string inputString) + { + using SHA256 sha256 = SHA256.Create(); + return sha256.ComputeHash(Encoding.UTF8.GetBytes(inputString)); + } + + /// + public IObservable IsLoggedIn => _isLoggedInSubject.AsObservable(); + + /// + public ReadOnlyObservableCollection Claims { get; } + + /// + public IObservable GetClaim(string type) + { + return _claims.Connect() + .Filter(c => c.Type == JwtClaimTypes.Email) + .ToCollection() + .Select(f => f.FirstOrDefault()); + } + + public async Task GetBearer() + { + await _authLock.WaitAsync(); + + try + { + // If not logged in, attempt to auto login first + if (!_isLoggedInSubject.Value) + await InternalAutoLogin(); + + // If there is no token, even after an auto-login, there's no bearer to add + if (_token == null) + return null; + + // If the token is expiring, refresh it + if (_token.Expired && !await UseRefreshToken(_token.RefreshToken)) + return null; + + return _token.AccessToken; + } + catch (Exception e) + { + _logger.Error(e, "Failed to retrieve bearer token"); + return null; + } + finally + { + _authLock.Release(); + } + } + + /// + public async Task AutoLogin(bool force = false) + { + await _authLock.WaitAsync(); + + try + { + return await InternalAutoLogin(force); + } + finally + { + _authLock.Release(); + } + } + + + /// + public async Task Login(CancellationToken cancellationToken) + { + await _authLock.WaitAsync(cancellationToken); + + try + { + if (_isLoggedInSubject.Value) + return; + + // Start a HTTP listener, this port could be in use but chances are very slim + string redirectUri = "http://localhost:56789"; + using HttpListener listener = new(); + listener.Prefixes.Add(redirectUri + "/"); + listener.Start(); + + // Discover the Identity endpoint + DiscoveryDocumentResponse disco = await GetDiscovery(); + if (disco.AuthorizeEndpoint == null) + throw new ArtemisWebClientException("Could not determine the authorize endpoint"); + + // Generate the PKCE code verifier and code challenge + string codeVerifier = CryptoRandom.CreateUniqueId(); + string codeChallenge = Base64Url.Encode(HashSha256(codeVerifier)); + string state = Guid.NewGuid().ToString("N"); + + // Open the web browser for the user to log in and authorize the app + RequestUrl authRequestUrl = new(disco.AuthorizeEndpoint); + string url = authRequestUrl.CreateAuthorizeUrl( + CLIENT_ID, + "code", + "openid profile email offline_access api", + redirectUri, + state, + codeChallenge: codeChallenge, + codeChallengeMethod: OidcConstants.CodeChallengeMethods.Sha256); + Utilities.OpenUrl(url); + + // Wait for the callback with the code + HttpListenerContext context = await listener.GetContextAsync().WaitAsync(cancellationToken); + string? code = context.Request.QueryString.Get("code"); + string? returnState = context.Request.QueryString.Get("state"); + + // Validate that a code was given and that our state matches, ensuring we don't respond to a request we did not initialize + if (code == null || returnState != state) + throw new ArtemisWebClientException("Did not get the expected response on the callback"); + + // Redirect the browser to a page hosted by Identity indicating success + context.Response.StatusCode = (int) HttpStatusCode.Redirect; + context.Response.AddHeader("Location", $"{WorkshopConstants.AUTHORITY_URL}/account/login/success"); + context.Response.Close(); + + // Request auth tokens + HttpClient client = _httpClientFactory.CreateClient(); + TokenResponse response = await client.RequestAuthorizationCodeTokenAsync(new AuthorizationCodeTokenRequest + { + Address = disco.TokenEndpoint, + ClientId = CLIENT_ID, + Code = code, + CodeVerifier = codeVerifier, + RedirectUri = redirectUri + }, cancellationToken); + + if (response.IsError) + throw new ArtemisWebClientException("Failed to request device authorization: " + response.Error); + + // Set the current user using the new tokens + SetCurrentUser(response); + + listener.Stop(); + listener.Close(); + } + catch (HttpListenerException e) + { + throw new ArtemisWebClientException($"HTTP listener for login callback failed with error code {e.ErrorCode}", e); + } + finally + { + _authLock.Release(); + } + } + + /// + public void Logout() + { + _token = null; + _claims.Clear(); + SetStoredRefreshToken(null); + + _isLoggedInSubject.OnNext(false); + } + + private async Task InternalAutoLogin(bool force = false) + { + if (!force && _isLoggedInSubject.Value) + return true; + + if (_noStoredRefreshToken) + return false; + + string? refreshToken = GetStoredRefreshToken(); + if (refreshToken == null) + return false; + + return await UseRefreshToken(refreshToken); + } + + private string? GetStoredRefreshToken() + { + string? token = _authenticationRepository.GetRefreshToken(); + _noStoredRefreshToken = token == null; + return token; + } + + private void SetStoredRefreshToken(string? token) + { + _authenticationRepository.SetRefreshToken(token); + _noStoredRefreshToken = token == null; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Services/IAuthenticationService.cs b/src/Artemis.WebClient.Workshop/Services/IAuthenticationService.cs deleted file mode 100644 index eed47ecff..000000000 --- a/src/Artemis.WebClient.Workshop/Services/IAuthenticationService.cs +++ /dev/null @@ -1,214 +0,0 @@ -using System.Collections.ObjectModel; -using System.Diagnostics; -using System.IdentityModel.Tokens.Jwt; -using System.Security.Claims; -using Artemis.Core; -using Artemis.Core.Services; -using Artemis.WebClient.Workshop.Repositories; -using IdentityModel; -using IdentityModel.Client; - -namespace Artemis.WebClient.Workshop.Services; - -public interface IAuthenticationService : IProtectedArtemisService -{ - bool IsLoggedIn { get; } - string? UserCode { get; } - ReadOnlyObservableCollection Claims { get; } - - Task GetBearer(); - Task AutoLogin(); - Task Login(); - void Logout(); -} - -internal class AuthenticationService : CorePropertyChanged, IAuthenticationService -{ - private const string CLIENT_ID = "artemis.desktop"; - - private readonly IDiscoveryCache _discoveryCache; - private readonly IAuthenticationRepository _authenticationRepository; - private readonly IHttpClientFactory _httpClientFactory; - private readonly ObservableCollection _claims = new(); - private readonly SemaphoreSlim _bearerLock = new(1); - - private AuthenticationToken? _token; - private string? _userCode; - private bool _isLoggedIn; - - public AuthenticationService(IHttpClientFactory httpClientFactory, IDiscoveryCache discoveryCache, IAuthenticationRepository authenticationRepository) - { - _httpClientFactory = httpClientFactory; - _discoveryCache = discoveryCache; - _authenticationRepository = authenticationRepository; - - Claims = new ReadOnlyObservableCollection(_claims); - } - - /// - public bool IsLoggedIn - { - get => _isLoggedIn; - private set => SetAndNotify(ref _isLoggedIn, value); - } - - /// - public string? UserCode - { - get => _userCode; - private set => SetAndNotify(ref _userCode, value); - } - - /// - public ReadOnlyObservableCollection Claims { get; } - - public async Task GetBearer() - { - await _bearerLock.WaitAsync(); - - try - { - // If not logged in, attempt to auto login first - if (!IsLoggedIn) - await AutoLogin(); - - if (_token == null) - return null; - - // If the token is expiring, refresh it - if (_token.Expired && !await UseRefreshToken(_token.RefreshToken)) - return null; - - return _token.AccessToken; - } - finally - { - _bearerLock.Release(); - } - } - - /// - public async Task AutoLogin() - { - if (IsLoggedIn) - return true; - - string? refreshToken = _authenticationRepository.GetRefreshToken(); - if (refreshToken == null) - return false; - - return await UseRefreshToken(refreshToken); - } - - /// - public async Task Login() - { - DiscoveryDocumentResponse disco = await GetDiscovery(); - HttpClient client = _httpClientFactory.CreateClient(); - DeviceAuthorizationResponse response = await client.RequestDeviceAuthorizationAsync(new DeviceAuthorizationRequest - { - Address = disco.DeviceAuthorizationEndpoint, - ClientId = CLIENT_ID, - Scope = "openid profile email offline_access api" - }); - if (response.IsError) - throw new ArtemisWebClientException("Failed to request device authorization: " + response.Error); - if (response.DeviceCode == null) - throw new ArtemisWebClientException("Failed to request device authorization: Got no device code"); - - DateTimeOffset timeout = DateTimeOffset.UtcNow.AddSeconds(response.ExpiresIn ?? 1800); - - Process.Start(new ProcessStartInfo {FileName = response.VerificationUriComplete, UseShellExecute = true}); - await Task.Delay(TimeSpan.FromSeconds(response.Interval)); - while (DateTimeOffset.UtcNow < timeout) - { - bool success = await AttemptRequestDeviceToken(client, response.DeviceCode); - if (success) - return true; - await Task.Delay(TimeSpan.FromSeconds(response.Interval)); - } - - return false; - } - - /// - public void Logout() - { - _token = null; - _claims.Clear(); - _authenticationRepository.SetRefreshToken(null); - - IsLoggedIn = false; - } - - private async Task GetDiscovery() - { - DiscoveryDocumentResponse disco = await _discoveryCache.GetAsync(); - if (disco.IsError) - throw new ArtemisWebClientException("Failed to retrieve discovery document: " + disco.Error); - - return disco; - } - - private async Task AttemptRequestDeviceToken(HttpClient client, string deviceCode) - { - DiscoveryDocumentResponse disco = await GetDiscovery(); - TokenResponse response = await client.RequestDeviceTokenAsync(new DeviceTokenRequest - { - Address = disco.TokenEndpoint, - ClientId = CLIENT_ID, - DeviceCode = deviceCode - }); - - if (response.IsError) - { - if (response.Error == OidcConstants.TokenErrors.AuthorizationPending || response.Error == OidcConstants.TokenErrors.SlowDown) - return false; - - throw new ArtemisWebClientException("Failed to request device token: " + response.Error); - } - - SetCurrentUser(response); - return true; - } - - private void SetCurrentUser(TokenResponse response) - { - _token = new AuthenticationToken(response); - _authenticationRepository.SetRefreshToken(_token.RefreshToken); - - JwtSecurityTokenHandler handler = new(); - JwtSecurityToken? token = handler.ReadJwtToken(response.IdentityToken); - if (token == null) - throw new ArtemisWebClientException("Failed to read JWT token"); - - _claims.Clear(); - foreach (Claim responseClaim in token.Claims) - _claims.Add(responseClaim); - - IsLoggedIn = true; - } - - private async Task UseRefreshToken(string refreshToken) - { - DiscoveryDocumentResponse disco = await GetDiscovery(); - HttpClient client = _httpClientFactory.CreateClient(); - TokenResponse response = await client.RequestRefreshTokenAsync(new RefreshTokenRequest() - { - Address = disco.TokenEndpoint, - ClientId = CLIENT_ID, - RefreshToken = refreshToken - }); - - if (response.IsError) - { - if (response.Error == OidcConstants.TokenErrors.ExpiredToken) - return false; - - throw new ArtemisWebClientException("Failed to request refresh token: " + response.Error); - } - - SetCurrentUser(response); - return false; - } -} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Services/InstalledEntry.cs b/src/Artemis.WebClient.Workshop/Services/InstalledEntry.cs new file mode 100644 index 000000000..8d5cebd18 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Services/InstalledEntry.cs @@ -0,0 +1,68 @@ +using Artemis.Storage.Entities.Workshop; + +namespace Artemis.WebClient.Workshop.Services; + +public class InstalledEntry +{ + internal InstalledEntry(EntryEntity entity) + { + Entity = entity; + + Load(); + } + + public InstalledEntry(IGetEntryById_Entry entry) + { + Entity = new EntryEntity(); + + EntryId = entry.Id; + EntryType = entry.EntryType; + + Author = entry.Author; + Name = entry.Name; + } + + public long EntryId { get; set; } + public EntryType EntryType { get; set; } + + public string Author { get; set; } = string.Empty; + public string Name { get; set; } = string.Empty; + + public long ReleaseId { get; set; } + public string ReleaseVersion { get; set; } = string.Empty; + public DateTimeOffset InstalledAt { get; set; } + + public string? LocalReference { get; set; } + + internal EntryEntity Entity { get; } + + internal void Load() + { + EntryId = Entity.EntryId; + EntryType = (EntryType) Entity.EntryType; + + Author = Entity.Author; + Name = Entity.Name; + + ReleaseId = Entity.ReleaseId; + ReleaseVersion = Entity.ReleaseVersion; + InstalledAt = Entity.InstalledAt; + + LocalReference = Entity.LocalReference; + } + + internal void Save() + { + Entity.EntryId = EntryId; + Entity.EntryType = (int) EntryType; + + Entity.Author = Author; + Entity.Name = Name; + + Entity.ReleaseId = ReleaseId; + Entity.ReleaseVersion = ReleaseVersion; + Entity.InstalledAt = InstalledAt; + + Entity.LocalReference = LocalReference; + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Services/Interfaces/IAuthenticationService.cs b/src/Artemis.WebClient.Workshop/Services/Interfaces/IAuthenticationService.cs new file mode 100644 index 000000000..c3a3686ba --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Services/Interfaces/IAuthenticationService.cs @@ -0,0 +1,17 @@ +using System.Collections.ObjectModel; +using System.Security.Claims; +using Artemis.Core.Services; + +namespace Artemis.WebClient.Workshop.Services; + +public interface IAuthenticationService : IProtectedArtemisService +{ + IObservable IsLoggedIn { get; } + ReadOnlyObservableCollection Claims { get; } + + IObservable GetClaim(string type); + Task GetBearer(); + Task AutoLogin(bool force = false); + Task Login(CancellationToken cancellationToken); + void Logout(); +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Services/Interfaces/IWorkshopService.cs b/src/Artemis.WebClient.Workshop/Services/Interfaces/IWorkshopService.cs new file mode 100644 index 000000000..3f2303c00 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Services/Interfaces/IWorkshopService.cs @@ -0,0 +1,22 @@ +using Artemis.UI.Shared.Utilities; +using Artemis.WebClient.Workshop.Handlers.UploadHandlers; + +namespace Artemis.WebClient.Workshop.Services; + +public interface IWorkshopService +{ + Task GetEntryIcon(long entryId, CancellationToken cancellationToken); + Task SetEntryIcon(long entryId, Progress progress, Stream icon, CancellationToken cancellationToken); + Task GetWorkshopStatus(CancellationToken cancellationToken); + Task ValidateWorkshopStatus(CancellationToken cancellationToken); + Task NavigateToEntry(long entryId, EntryType entryType); + + List GetInstalledEntries(); + InstalledEntry? GetInstalledEntry(IGetEntryById_Entry entry); + InstalledEntry CreateInstalledEntry(IGetEntryById_Entry entry); + void RemoveInstalledEntry(InstalledEntry installedEntry); + void SaveInstalledEntry(InstalledEntry entry); + + + public record WorkshopStatus(bool IsReachable, string Message); +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/Services/WorkshopService.cs b/src/Artemis.WebClient.Workshop/Services/WorkshopService.cs new file mode 100644 index 000000000..f40a15375 --- /dev/null +++ b/src/Artemis.WebClient.Workshop/Services/WorkshopService.cs @@ -0,0 +1,139 @@ +using System.Net.Http.Headers; +using Artemis.Storage.Entities.Workshop; +using Artemis.Storage.Repositories.Interfaces; +using Artemis.UI.Shared.Routing; +using Artemis.UI.Shared.Utilities; +using Artemis.WebClient.Workshop.Handlers.UploadHandlers; + +namespace Artemis.WebClient.Workshop.Services; + +public class WorkshopService : IWorkshopService +{ + private readonly IHttpClientFactory _httpClientFactory; + private readonly IRouter _router; + private readonly IEntryRepository _entryRepository; + + public WorkshopService(IHttpClientFactory httpClientFactory, IRouter router, IEntryRepository entryRepository) + { + _httpClientFactory = httpClientFactory; + _router = router; + _entryRepository = entryRepository; + } + + public async Task GetEntryIcon(long entryId, CancellationToken cancellationToken) + { + HttpClient client = _httpClientFactory.CreateClient(WorkshopConstants.WORKSHOP_CLIENT_NAME); + try + { + HttpResponseMessage response = await client.GetAsync($"entries/{entryId}/icon", cancellationToken); + response.EnsureSuccessStatusCode(); + return await response.Content.ReadAsStreamAsync(cancellationToken); + } + catch (HttpRequestException) + { + // ignored + return null; + } + } + + public async Task SetEntryIcon(long entryId, Progress progress, Stream icon, CancellationToken cancellationToken) + { + icon.Seek(0, SeekOrigin.Begin); + + // Submit the archive + HttpClient client = _httpClientFactory.CreateClient(WorkshopConstants.WORKSHOP_CLIENT_NAME); + + // Construct the request + MultipartFormDataContent content = new(); + ProgressableStreamContent streamContent = new(icon, progress); + streamContent.Headers.ContentType = new MediaTypeHeaderValue("image/png"); + content.Add(streamContent, "file", "file.png"); + + // Submit + HttpResponseMessage response = await client.PostAsync($"entries/{entryId}/icon", content, cancellationToken); + if (!response.IsSuccessStatusCode) + return ImageUploadResult.FromFailure($"{response.StatusCode} - {await response.Content.ReadAsStringAsync(cancellationToken)}"); + return ImageUploadResult.FromSuccess(); + } + + /// + public async Task GetWorkshopStatus(CancellationToken cancellationToken) + { + try + { + // Don't use the workshop client which adds auth headers + HttpClient client = _httpClientFactory.CreateClient(); + HttpResponseMessage response = await client.SendAsync(new HttpRequestMessage(HttpMethod.Head, WorkshopConstants.WORKSHOP_URL + "/status"), cancellationToken); + return new IWorkshopService.WorkshopStatus(response.IsSuccessStatusCode, response.StatusCode.ToString()); + } + catch (OperationCanceledException e) + { + return new IWorkshopService.WorkshopStatus(false, e.Message); + } + catch (HttpRequestException e) + { + return new IWorkshopService.WorkshopStatus(false, e.Message); + } + } + + /// + public async Task ValidateWorkshopStatus(CancellationToken cancellationToken) + { + IWorkshopService.WorkshopStatus status = await GetWorkshopStatus(cancellationToken); + if (!status.IsReachable && !cancellationToken.IsCancellationRequested) + await _router.Navigate($"workshop/offline/{status.Message}"); + return status.IsReachable; + } + + public async Task NavigateToEntry(long entryId, EntryType entryType) + { + switch (entryType) + { + case EntryType.Profile: + await _router.Navigate($"workshop/entries/profiles/details/{entryId}"); + break; + case EntryType.Layout: + await _router.Navigate($"workshop/entries/layouts/details/{entryId}"); + break; + case EntryType.Plugin: + await _router.Navigate($"workshop/entries/plugins/details/{entryId}"); + break; + default: + throw new ArgumentOutOfRangeException(nameof(entryType)); + } + } + + public List GetInstalledEntries() + { + return _entryRepository.GetAll().Select(e => new InstalledEntry(e)).ToList(); + } + + /// + public InstalledEntry? GetInstalledEntry(IGetEntryById_Entry entry) + { + EntryEntity? entity = _entryRepository.GetByEntryId(entry.Id); + if (entity == null) + return null; + + return new InstalledEntry(entity); + } + + /// + public InstalledEntry CreateInstalledEntry(IGetEntryById_Entry entry) + { + return new InstalledEntry(entry); + } + + /// + public void RemoveInstalledEntry(InstalledEntry installedEntry) + { + _entryRepository.Remove(installedEntry.Entity); + } + + /// + public void SaveInstalledEntry(InstalledEntry entry) + { + entry.Save(); + _entryRepository.Save(entry.Entity); + } +} \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/WorkshopConstants.cs b/src/Artemis.WebClient.Workshop/WorkshopConstants.cs index b514ae862..192a92b56 100644 --- a/src/Artemis.WebClient.Workshop/WorkshopConstants.cs +++ b/src/Artemis.WebClient.Workshop/WorkshopConstants.cs @@ -4,4 +4,5 @@ public static class WorkshopConstants { public const string AUTHORITY_URL = "https://identity.artemis-rgb.com"; public const string WORKSHOP_URL = "https://workshop.artemis-rgb.com"; + public const string WORKSHOP_CLIENT_NAME = "WorkshopApiClient"; } \ No newline at end of file diff --git a/src/Artemis.WebClient.Workshop/graphql.config.yml b/src/Artemis.WebClient.Workshop/graphql.config.yml index 9662a514f..a8ba99703 100644 --- a/src/Artemis.WebClient.Workshop/graphql.config.yml +++ b/src/Artemis.WebClient.Workshop/graphql.config.yml @@ -2,7 +2,7 @@ schema: schema.graphql extensions: endpoints: Default GraphQL Endpoint: - url: https://workshop.artemis-rgb.com/graphql + url: https://localhost:7281/graphql headers: user-agent: JS GraphQL introspect: true diff --git a/src/Artemis.WebClient.Workshop/schema.graphql b/src/Artemis.WebClient.Workshop/schema.graphql index 59efce290..c889bca85 100644 --- a/src/Artemis.WebClient.Workshop/schema.graphql +++ b/src/Artemis.WebClient.Workshop/schema.graphql @@ -7,7 +7,7 @@ schema { type Category { icon: String! - id: Int! + id: Long! name: String! } @@ -37,8 +37,11 @@ type Entry { downloads: Long! entryType: EntryType! icon: Image - id: UUID! + iconId: UUID + id: Long! images: [Image!]! + latestRelease: Release + latestReleaseId: Long name: String! releases: [Release!]! summary: String! @@ -52,14 +55,16 @@ type Image { type Mutation { addEntry(input: CreateEntryInput!): Entry + removeEntry(id: Long!): Entry updateEntry(input: UpdateEntryInput!): Entry } type Query { categories(order: [CategorySortInput!], where: CategoryFilterInput): [Category!]! entries(order: [EntrySortInput!], skip: Int, take: Int, where: EntryFilterInput): EntriesCollectionSegment - entry(id: UUID!): Entry + entry(id: Long!): Entry searchEntries(input: String!, order: [EntrySortInput!], type: EntryType, where: EntryFilterInput): [Entry!]! + submittedEntries(order: [EntrySortInput!], where: EntryFilterInput): [Entry!]! } type Release { @@ -67,14 +72,14 @@ type Release { downloadSize: Long! downloads: Long! entry: Entry! - entryId: UUID! - id: UUID! + entryId: Long! + id: Long! md5Hash: String version: String! } type Tag { - id: Int! + id: Long! name: String! } @@ -106,7 +111,7 @@ scalar UUID input CategoryFilterInput { and: [CategoryFilterInput!] icon: StringOperationFilterInput - id: IntOperationFilterInput + id: LongOperationFilterInput name: StringOperationFilterInput or: [CategoryFilterInput!] } @@ -118,7 +123,7 @@ input CategorySortInput { } input CreateEntryInput { - categories: [Int!]! + categories: [Long!]! description: String! entryType: EntryType! name: String! @@ -151,8 +156,11 @@ input EntryFilterInput { downloads: LongOperationFilterInput entryType: EntryTypeOperationFilterInput icon: ImageFilterInput - id: UuidOperationFilterInput + iconId: UuidOperationFilterInput + id: LongOperationFilterInput images: ListFilterInputTypeOfImageFilterInput + latestRelease: ReleaseFilterInput + latestReleaseId: LongOperationFilterInput name: StringOperationFilterInput or: [EntryFilterInput!] releases: ListFilterInputTypeOfReleaseFilterInput @@ -168,7 +176,10 @@ input EntrySortInput { downloads: SortEnumType entryType: SortEnumType icon: ImageSortInput + iconId: SortEnumType id: SortEnumType + latestRelease: ReleaseSortInput + latestReleaseId: SortEnumType name: SortEnumType summary: SortEnumType } @@ -192,21 +203,6 @@ input ImageSortInput { mimeType: SortEnumType } -input IntOperationFilterInput { - eq: Int - gt: Int - gte: Int - in: [Int] - lt: Int - lte: Int - neq: Int - ngt: Int - ngte: Int - nin: [Int] - nlt: Int - nlte: Int -} - input ListFilterInputTypeOfCategoryFilterInput { all: CategoryFilterInput any: Boolean @@ -256,13 +252,24 @@ input ReleaseFilterInput { downloadSize: LongOperationFilterInput downloads: LongOperationFilterInput entry: EntryFilterInput - entryId: UuidOperationFilterInput - id: UuidOperationFilterInput + entryId: LongOperationFilterInput + id: LongOperationFilterInput md5Hash: StringOperationFilterInput or: [ReleaseFilterInput!] version: StringOperationFilterInput } +input ReleaseSortInput { + createdAt: SortEnumType + downloadSize: SortEnumType + downloads: SortEnumType + entry: EntrySortInput + entryId: SortEnumType + id: SortEnumType + md5Hash: SortEnumType + version: SortEnumType +} + input StringOperationFilterInput { and: [StringOperationFilterInput!] contains: String @@ -280,15 +287,15 @@ input StringOperationFilterInput { input TagFilterInput { and: [TagFilterInput!] - id: IntOperationFilterInput + id: LongOperationFilterInput name: StringOperationFilterInput or: [TagFilterInput!] } input UpdateEntryInput { - categories: [Int!]! + categories: [Long!]! description: String! - id: UUID! + id: Long! name: String! summary: String! tags: [String!]!