diff --git a/.github/workflows/admin-sample.cd.yml b/.github/workflows/admin-sample.cd.yml index 0e67144451..cd06d839b4 100644 --- a/.github/workflows/admin-sample.cd.yml +++ b/.github/workflows/admin-sample.cd.yml @@ -84,7 +84,7 @@ jobs: - name: Upload server web artifact uses: actions/upload-artifact@v5 with: - name: AdminPanelWeb + name: AdminPanel path: server-web include-hidden-files: true # Required for wwwroot/.well-known folder @@ -293,7 +293,7 @@ jobs: - uses: maxim-lobanov/setup-xcode@v1.6.0 with: - xcode-version: '26.1' + xcode-version: '26.2' - uses: actions/setup-node@v6 with: diff --git a/.github/workflows/bit.ci.BlazorUI.yml b/.github/workflows/bit.ci.BlazorUI.yml index 06901ec1de..85d5ce4dd5 100644 --- a/.github/workflows/bit.ci.BlazorUI.yml +++ b/.github/workflows/bit.ci.BlazorUI.yml @@ -50,4 +50,4 @@ jobs: run: dotnet build src/BlazorUI/Bit.BlazorUI.slnx - name: Test - run: cd src/BlazorUI/Bit.BlazorUI.Tests && dotnet test --no-build --verbosity normal + run: cd src/BlazorUI/Tests/Bit.BlazorUI.Tests && dotnet test --no-build --verbosity normal diff --git a/.github/workflows/bit.full.ci.yml b/.github/workflows/bit.full.ci.yml index aba7fb9a01..74d2f116df 100644 --- a/.github/workflows/bit.full.ci.yml +++ b/.github/workflows/bit.full.ci.yml @@ -70,7 +70,10 @@ jobs: run: cd src && dotnet workload install wasm-tools - name: Create project from template with PostgreSQL - run: dotnet new bit-bp --name TestPostgreSQL --database PostgreSQL --module Sales --signalR --aspire + run: dotnet new bit-bp --name TestPostgreSQL --database PostgreSQL --module Sales --signalR --aspire --redis + + - name: Trust dotnet cert + run: dotnet dev-certs https --trust # Necessary for aspire related resources - name: Create appsettings.json for Client.Web run: | @@ -229,13 +232,13 @@ jobs: - name: Build sample configuration 1 run: | - dotnet new bit-bp --name TestProject --database SqlServer --filesStorage AzureBlobStorage --api Integrated --captcha reCaptcha --pipeline Azure --module Admin --offlineDb --appInsights --sentry --signalR --notification --cloudflare --ads --aspire + dotnet new bit-bp --name TestProject --database SqlServer --filesStorage AzureBlobStorage --api Integrated --captcha reCaptcha --pipeline Azure --module Admin --offlineDb --appInsights --sentry --signalR --notification --cloudflare --ads --aspire --redis dotnet build TestProject/TestProject.sln -p:InvariantGlobalization=false -p:Environment=Staging rm -r "TestProject" - name: Build sample configuration 2 run: | - dotnet new bit-bp --name TestProject2 --database Other --filesStorage S3 --api Standalone --captcha None --pipeline None --module None --offlineDb false --appInsights false --sentry false --signalR false --notification false --cloudflare false --ads false --aspire true + dotnet new bit-bp --name TestProject2 --database Other --filesStorage S3 --api Standalone --captcha None --pipeline None --module None --offlineDb false --appInsights false --sentry false --signalR false --notification false --cloudflare false --ads false --aspire true --redis false dotnet build TestProject2/TestProject2.slnx -p:InvariantGlobalization=true -p:Environment=Development rm -r "TestProject2" diff --git a/.github/workflows/blazorui.demo.cd.yml b/.github/workflows/blazorui.demo.cd.yml index 7a73e548a1..bdef14165a 100644 --- a/.github/workflows/blazorui.demo.cd.yml +++ b/.github/workflows/blazorui.demo.cd.yml @@ -164,7 +164,7 @@ jobs: - uses: maxim-lobanov/setup-xcode@v1.6.0 with: - xcode-version: '26.1' + xcode-version: '26.2' - uses: actions/setup-node@v6 with: diff --git a/.github/workflows/todo-sample.cd.yml b/.github/workflows/todo-sample.cd.yml index bc510f9a98..f954080fe4 100644 --- a/.github/workflows/todo-sample.cd.yml +++ b/.github/workflows/todo-sample.cd.yml @@ -458,7 +458,7 @@ jobs: - uses: maxim-lobanov/setup-xcode@v1.6.0 with: - xcode-version: '26.1' + xcode-version: '26.2' - name: Create project from Boilerplate run: | diff --git a/src/Besql/Bit.Besql/wwwroot/bit-besql.js b/src/Besql/Bit.Besql/wwwroot/bit-besql.js index 85f55469cb..de69cd7353 100644 --- a/src/Besql/Bit.Besql/wwwroot/bit-besql.js +++ b/src/Besql/Bit.Besql/wwwroot/bit-besql.js @@ -1,5 +1,5 @@ var BitBesql = window.BitBesql || {}; -BitBesql.version = window['bit-besql version'] = '10.2.1'; +BitBesql.version = window['bit-besql version'] = '10.3.0'; BitBesql.persist = async function besqlPersist(fileName) { diff --git a/src/Bit.Build.props b/src/Bit.Build.props index bf006ebf7b..ecd9faa2df 100644 --- a/src/Bit.Build.props +++ b/src/Bit.Build.props @@ -27,7 +27,7 @@ https://github.com/bitfoundation/bitplatform - 10.2.1 + 10.3.0 $(ReleaseVersion) https://github.com/bitfoundation/bitplatform/releases/tag/v-$(ReleaseVersion) $([System.String]::Copy($(ReleaseVersion)).Replace('-pre-', '.')) diff --git a/src/Bit.slnx b/src/Bit.slnx index edbadc6304..ded9933557 100644 --- a/src/Bit.slnx +++ b/src/Bit.slnx @@ -44,12 +44,12 @@ + - - + diff --git a/src/BlazorES2019/Bit.BlazorES2019/buildTransitive/Bit.BlazorES2019.targets b/src/BlazorES2019/Bit.BlazorES2019/buildTransitive/Bit.BlazorES2019.targets index c2397b923f..91d8390f56 100644 --- a/src/BlazorES2019/Bit.BlazorES2019/buildTransitive/Bit.BlazorES2019.targets +++ b/src/BlazorES2019/Bit.BlazorES2019/buildTransitive/Bit.BlazorES2019.targets @@ -1,13 +1,13 @@ - - + + PreserveNewest PreserveNewest wwwroot\_framework\%(Filename)%(Extension) - - + + PreserveNewest PreserveNewest wwwroot\_framework\%(Filename)%(Extension) diff --git a/src/BlazorUI/Bit.BlazorUI.Assets/package-lock.json b/src/BlazorUI/Bit.BlazorUI.Assets/package-lock.json index 56f60e57e1..29f6ecf8e2 100644 --- a/src/BlazorUI/Bit.BlazorUI.Assets/package-lock.json +++ b/src/BlazorUI/Bit.BlazorUI.Assets/package-lock.json @@ -5,7 +5,7 @@ "packages": { "": { "devDependencies": { - "sass": "1.97.0" + "sass": "1.97.1" } }, "node_modules/@parcel/watcher": { @@ -448,9 +448,9 @@ } }, "node_modules/sass": { - "version": "1.97.0", - "resolved": "https://registry.npmjs.org/sass/-/sass-1.97.0.tgz", - "integrity": "sha512-KR0igP1z4avUJetEuIeOdDlwaUDvkH8wSx7FdSjyYBS3dpyX3TzHfAMO0G1Q4/3cdjcmi3r7idh+KCmKqS+KeQ==", + "version": "1.97.1", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.97.1.tgz", + "integrity": "sha512-uf6HoO8fy6ClsrShvMgaKUn14f2EHQLQRtpsZZLeU/Mv0Q1K5P0+x2uvH6Cub39TVVbWNSrraUhDAoFph6vh0A==", "dev": true, "license": "MIT", "dependencies": { diff --git a/src/BlazorUI/Bit.BlazorUI.Assets/package.json b/src/BlazorUI/Bit.BlazorUI.Assets/package.json index 723ab56604..46229d079a 100644 --- a/src/BlazorUI/Bit.BlazorUI.Assets/package.json +++ b/src/BlazorUI/Bit.BlazorUI.Assets/package.json @@ -1,5 +1,5 @@ { "devDependencies": { - "sass": "1.97.0" + "sass": "1.97.1" } } diff --git a/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor b/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor index ba49ac62ff..102db220cd 100644 --- a/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor +++ b/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor @@ -1,9 +1,11 @@ -@namespace Bit.BlazorUI +@namespace Bit.BlazorUI @inherits BitComponentBase
@@ -14,7 +16,7 @@
- + @ChildContent @@ -22,4 +24,4 @@
-
\ No newline at end of file + diff --git a/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor.cs b/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor.cs index d5c1190f6d..585b7304f7 100644 --- a/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor.cs +++ b/src/BlazorUI/Bit.BlazorUI.Extras/Components/AppShell/BitAppShell.razor.cs @@ -12,8 +12,9 @@ public partial class BitAppShell : BitComponentBase public const string Container = "BitAppShell.Container"; + private bool _locationChanged; - private ElementReference _containerRef = default!; + private ElementReference? _containerRef; @@ -28,17 +29,12 @@ public partial class BitAppShell : BitComponentBase [Parameter] public bool AutoGoToTop { get; set; } /// - /// The cascading values to be provided for the children of the layout. - /// - [Parameter] public IEnumerable? CascadingValues { get; set; } - - /// - /// The content of the layout. + /// The content of the app shell. /// [Parameter] public RenderFragment? ChildContent { get; set; } /// - /// Custom CSS classes for different parts of the layout. + /// Custom CSS classes for different parts of the app shell. /// [Parameter] public BitAppShellClassStyles? Classes { get; set; } @@ -48,10 +44,20 @@ public partial class BitAppShell : BitComponentBase [Parameter] public bool PersistScroll { get; set; } /// - /// Custom CSS styles for different parts of the layout. + /// Custom CSS styles for different parts of the app shell. /// [Parameter] public BitAppShellClassStyles? Styles { get; set; } + /// + /// The cascading value list to be provided for the children of the app shell. + /// + [Parameter] public BitCascadingValueList? ValueList { get; set; } + + /// + /// The cascading values to be provided for the children of the app shell. + /// + [Parameter] public IEnumerable? Values { get; set; } + /// @@ -59,13 +65,15 @@ public partial class BitAppShell : BitComponentBase /// public async Task GoToTop(BitScrollBehavior? behavior = null) { - await _js.BitExtrasGoToTop(_containerRef, behavior); + if (_containerRef.HasValue is false) return; + + await _js.BitExtrasGoToTop(_containerRef.Value, behavior); } /// - /// The element reference to the main container of the ap shell. + /// The element reference to the main container of the app shell. /// - public ElementReference ContainerRef => _containerRef; + public ElementReference? ContainerRef => _containerRef; @@ -95,7 +103,9 @@ protected override async Task OnAfterRenderAsync(bool firstRender) { if (firstRender && PersistScroll) { - await _js.BitAppShellInitScroll(_containerRef, _navManager.Uri); + if (_containerRef.HasValue is false) return; + + await _js.BitAppShellInitScroll(_containerRef.Value, _navManager.Uri); } if (_locationChanged && firstRender is false) diff --git a/src/BlazorUI/Bit.BlazorUI.Extras/Components/NavPanel/BitNavPanel.razor.cs b/src/BlazorUI/Bit.BlazorUI.Extras/Components/NavPanel/BitNavPanel.razor.cs index 0232ea7c51..04ef91b48f 100644 --- a/src/BlazorUI/Bit.BlazorUI.Extras/Components/NavPanel/BitNavPanel.razor.cs +++ b/src/BlazorUI/Bit.BlazorUI.Extras/Components/NavPanel/BitNavPanel.razor.cs @@ -391,7 +391,7 @@ private async Task HandleOnSwipeTrigger(BitSwipeTrapTriggerArgs args) if (IsOpen is false) return $"{StyleBuilder.Value};{(isToggled ? Styles?.Toggled : string.Empty)}".Trim(';'); var translate = ((Dir != BitDir.Rtl && diffXPanel < 0) || (Dir == BitDir.Rtl && diffXPanel > 0)) - ? $"transform: translateX({diffXPanel}px)" + ? FormattableString.Invariant($"transform: translateX({diffXPanel}px)") : string.Empty; return $"{translate};{StyleBuilder.Value};{(isToggled ? Styles?.Toggled : string.Empty)}".Trim(';'); } diff --git a/src/BlazorUI/Bit.BlazorUI.Extras/package-lock.json b/src/BlazorUI/Bit.BlazorUI.Extras/package-lock.json index 2cf455c4ce..82cf55908c 100644 --- a/src/BlazorUI/Bit.BlazorUI.Extras/package-lock.json +++ b/src/BlazorUI/Bit.BlazorUI.Extras/package-lock.json @@ -6,7 +6,7 @@ "": { "devDependencies": { "esbuild": "0.27.2", - "sass": "1.97.0", + "sass": "1.97.1", "typescript": "5.9.3" } }, @@ -934,9 +934,9 @@ } }, "node_modules/sass": { - "version": "1.97.0", - "resolved": "https://registry.npmjs.org/sass/-/sass-1.97.0.tgz", - "integrity": "sha512-KR0igP1z4avUJetEuIeOdDlwaUDvkH8wSx7FdSjyYBS3dpyX3TzHfAMO0G1Q4/3cdjcmi3r7idh+KCmKqS+KeQ==", + "version": "1.97.1", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.97.1.tgz", + "integrity": "sha512-uf6HoO8fy6ClsrShvMgaKUn14f2EHQLQRtpsZZLeU/Mv0Q1K5P0+x2uvH6Cub39TVVbWNSrraUhDAoFph6vh0A==", "dev": true, "license": "MIT", "dependencies": { diff --git a/src/BlazorUI/Bit.BlazorUI.Extras/package.json b/src/BlazorUI/Bit.BlazorUI.Extras/package.json index aaccc035b3..fd3a150745 100644 --- a/src/BlazorUI/Bit.BlazorUI.Extras/package.json +++ b/src/BlazorUI/Bit.BlazorUI.Extras/package.json @@ -1,7 +1,7 @@ { "devDependencies": { "esbuild": "0.27.2", - "sass": "1.97.0", + "sass": "1.97.1", "typescript": "5.9.3" } } diff --git a/src/BlazorUI/Bit.BlazorUI.Icons/package-lock.json b/src/BlazorUI/Bit.BlazorUI.Icons/package-lock.json index 607c92e468..142fd366ce 100644 --- a/src/BlazorUI/Bit.BlazorUI.Icons/package-lock.json +++ b/src/BlazorUI/Bit.BlazorUI.Icons/package-lock.json @@ -5,7 +5,7 @@ "packages": { "": { "devDependencies": { - "sass": "1.97.0" + "sass": "1.97.1" } }, "node_modules/@parcel/watcher": { @@ -448,9 +448,9 @@ } }, "node_modules/sass": { - "version": "1.97.0", - "resolved": "https://registry.npmjs.org/sass/-/sass-1.97.0.tgz", - "integrity": "sha512-KR0igP1z4avUJetEuIeOdDlwaUDvkH8wSx7FdSjyYBS3dpyX3TzHfAMO0G1Q4/3cdjcmi3r7idh+KCmKqS+KeQ==", + "version": "1.97.1", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.97.1.tgz", + "integrity": "sha512-uf6HoO8fy6ClsrShvMgaKUn14f2EHQLQRtpsZZLeU/Mv0Q1K5P0+x2uvH6Cub39TVVbWNSrraUhDAoFph6vh0A==", "dev": true, "license": "MIT", "dependencies": { diff --git a/src/BlazorUI/Bit.BlazorUI.Icons/package.json b/src/BlazorUI/Bit.BlazorUI.Icons/package.json index 723ab56604..46229d079a 100644 --- a/src/BlazorUI/Bit.BlazorUI.Icons/package.json +++ b/src/BlazorUI/Bit.BlazorUI.Icons/package.json @@ -1,5 +1,5 @@ { "devDependencies": { - "sass": "1.97.0" + "sass": "1.97.1" } } diff --git a/src/BlazorUI/Bit.BlazorUI.Tests/Components/Buttons/BitActionButtonTests.cs b/src/BlazorUI/Bit.BlazorUI.Tests/Components/Buttons/BitActionButtonTests.cs deleted file mode 100644 index 02b4a4daaf..0000000000 --- a/src/BlazorUI/Bit.BlazorUI.Tests/Components/Buttons/BitActionButtonTests.cs +++ /dev/null @@ -1,722 +0,0 @@ -using System; -using System.Diagnostics; -using Bunit; -using Microsoft.AspNetCore.Components; -using Microsoft.AspNetCore.Components.Forms; -using Microsoft.VisualStudio.TestTools.UnitTesting; - -namespace Bit.BlazorUI.Tests.Components.Buttons; - -[TestClass] -public class BitActionButtonTests : BunitTestContext -{ - [TestMethod, - DataRow(true), - DataRow(false) - ] - public void BitActionButtonIsEnabledTest(bool isEnabled) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IsEnabled, isEnabled); - }); - - var button = component.Find(".bit-acb"); - - if (isEnabled) - { - Assert.IsFalse(button.ClassList.Contains("bit-dis")); - } - else - { - Assert.IsTrue(button.ClassList.Contains("bit-dis")); - } - } - - [TestMethod, - DataRow("Icon1"), - DataRow("Icon2") - ] - public void BitActionButtonIconTest(string iconName) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IconName, iconName); - }); - - var icon = component.Find(".bit-icon"); - - Assert.IsTrue(icon.ClassList.Contains($"bit-icon--{iconName}")); - } - - [TestMethod, - DataRow("title1"), - DataRow("title2") - ] - public void BitActionButtonTitleTest(string title) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Title, title); - }); - - var button = component.Find(".bit-acb"); - - var expectedTitle = title; - var actualTitle = button.GetAttribute("title"); - - Assert.AreEqual(expectedTitle, actualTitle); - } - - [TestMethod, - DataRow(true), - DataRow(false) - ] - public void BitActionButtonOnClickTest(bool isEnabled) - { - var clicked = false; - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IsEnabled, isEnabled); - parameters.Add(p => p.OnClick, () => clicked = true); - }); - - var button = component.Find(".bit-acb"); - - if (isEnabled) - { - Assert.IsFalse(button.ClassList.Contains("bit-dis")); - } - else - { - Assert.IsTrue(button.ClassList.Contains("bit-dis")); - } - - button.Click(); - - var expected = isEnabled; - var actual = clicked; - - Assert.AreEqual(expected, actual); - } - - [TestMethod, - DataRow(true, false, false), - DataRow(true, true, false), - DataRow(false, false, true), - DataRow(false, true, false), - ] - public void BitActionButtonDisabledFocusTest(bool isEnabled, bool allowDisabledFocus, bool expectedResult) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IsEnabled, isEnabled); - parameters.Add(p => p.AllowDisabledFocus, allowDisabledFocus); - }); - - var button = component.Find(".bit-acb"); - - var hasTabIndexAttr = button.HasAttribute("tabindex"); - - Assert.AreEqual(expectedResult, hasTabIndexAttr); - - if (hasTabIndexAttr) - { - Assert.IsTrue(button?.GetAttribute("tabindex")?.Equals("-1")); - } - - } - - [TestMethod, - DataRow("description1"), - DataRow("description2") - ] - public void BitActionButtonAriaDescriptionTest(string ariaDescription) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.AriaDescription, ariaDescription); - }); - - var button = component.Find(".bit-acb"); - - Assert.IsTrue(button?.GetAttribute("aria-describedby")?.Contains(ariaDescription)); - } - - [TestMethod, - DataRow("label1"), - DataRow("label2") - ] - public void BitActionButtonAriaLabelTest(string ariaLabel) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.AriaLabel, ariaLabel); - }); - - var button = component.Find(".bit-acb"); - - Assert.IsTrue(button?.GetAttribute("aria-label")?.Contains(ariaLabel)); - } - - [TestMethod, - DataRow(true), - DataRow(false), - ] - public void BitActionButtonAriaHiddenTest(bool ariaHidden) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.AriaHidden, ariaHidden); - }); - - var button = component.Find(".bit-acb"); - - Assert.AreEqual(ariaHidden, button.HasAttribute("aria-hidden")); - } - - [TestMethod, - DataRow(null, true), - DataRow(null, false), - DataRow("", true), - DataRow("", false), - DataRow(" ", true), - DataRow(" ", false), - DataRow("href", true), - DataRow("href", false) - ] - public void BitActionButtonShouldRenderExpectedElementBasedOnHref(string href, bool isEnabled) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Href, href); - parameters.Add(p => p.IsEnabled, isEnabled); - }); - - var button = component.Find(".bit-acb"); - - var expectedTag = href.HasValue() ? "a" : "button"; - - var actualTag = button.TagName; - - Assert.AreEqual(expectedTag, actualTag, ignoreCase: true); - } - - [TestMethod, - DataRow(BitButtonType.Button), - DataRow(BitButtonType.Submit), - DataRow(BitButtonType.Reset) - ] - public void BitActionButtonTypeOfButtonTest(BitButtonType buttonType) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.ButtonType, buttonType); - }); - - var button = component.Find(".bit-acb"); - - var expectedType = buttonType switch - { - BitButtonType.Button => "button", - BitButtonType.Submit => "submit", - BitButtonType.Reset => "reset", - _ => throw new NotSupportedException(), - }; - - var actualType = button.GetAttribute("type"); - - Assert.AreEqual(expectedType, actualType); - } - - [TestMethod] - public void BitActionButtonSubmitStateInEditContextTest() - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.EditContext, new EditContext(this)); - }); - - var button = component.Find(".bit-acb"); - - var expectedType = "submit"; - var actualType = button.GetAttribute("type"); - - Assert.AreEqual(expectedType, actualType); - } - - [TestMethod] - public void BitActionButtonButtonStateNotOverriddenInEditContextTest() - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.EditContext, new EditContext(this)); - parameters.Add(p => p.ButtonType, BitButtonType.Button); - }); - - var button = component.Find(".bit-acb"); - - var expectedType = "button"; - var actualType = button.GetAttribute("type"); - - Assert.AreEqual(expectedType, actualType); - } - - [TestMethod] - public void BitActionButtonTabIndexIsRespectedWhenEnabled() - { - const string expectedTabIndex = "3"; - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.TabIndex, expectedTabIndex); - }); - - var button = component.Find(".bit-acb"); - - var actualTabIndex = button.GetAttribute("tabindex"); - - Assert.AreEqual(expectedTabIndex, actualTabIndex); - } - - [TestMethod, - DataRow(true, "href1"), - DataRow(false, "href2") - ] - public void BitActionButtonAnchorRespectsEnabledState(bool isEnabled, string href) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Href, href); - parameters.Add(p => p.IsEnabled, isEnabled); - }); - - var button = component.Find(".bit-acb"); - - var expectedHrefPresence = isEnabled; - - Assert.AreEqual(expectedHrefPresence, button.HasAttribute("href")); - Assert.AreEqual(isEnabled is false, button.HasAttribute("disabled")); - - if (isEnabled) - { - Assert.AreEqual(href, button.GetAttribute("href")); - } - } - - [TestMethod, - DataRow(BitColor.Primary), - DataRow(BitColor.Secondary), - DataRow(BitColor.Tertiary), - DataRow(BitColor.Info), - DataRow(BitColor.Success), - DataRow(BitColor.Warning), - DataRow(BitColor.SevereWarning), - DataRow(BitColor.Error), - DataRow(BitColor.PrimaryBackground), - DataRow(BitColor.SecondaryBackground), - DataRow(BitColor.TertiaryBackground), - DataRow(BitColor.PrimaryForeground), - DataRow(BitColor.SecondaryForeground), - DataRow(BitColor.TertiaryForeground), - DataRow(BitColor.PrimaryBorder), - DataRow(BitColor.SecondaryBorder), - DataRow(BitColor.TertiaryBorder), - DataRow(null) - ] - public void BitActionButtonColorClassTest(BitColor? color) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Color, color); - }); - - var button = component.Find(".bit-acb"); - - var expectedClass = GetColorClass(color); - - Assert.IsTrue(button.ClassList.Contains(expectedClass)); - } - - [TestMethod, - DataRow(BitSize.Small), - DataRow(BitSize.Medium), - DataRow(BitSize.Large), - DataRow(null) - ] - public void BitActionButtonSizeClassTest(BitSize? size) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Size, size); - }); - - var button = component.Find(".bit-acb"); - - var expectedClass = GetSizeClass(size); - - Assert.IsTrue(button.ClassList.Contains(expectedClass)); - } - - [TestMethod, - DataRow(true), - DataRow(false) - ] - public void BitActionButtonFullWidthClassTest(bool fullWidth) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.FullWidth, fullWidth); - }); - - var button = component.Find(".bit-acb"); - - var expectedClassPresence = fullWidth; - - Assert.AreEqual(expectedClassPresence, button.ClassList.Contains("bit-acb-fwi")); - } - - [TestMethod, - DataRow(BitIconPosition.Start), - DataRow(BitIconPosition.End), - DataRow(null) - ] - public void BitActionButtonIconPositionClassTest(BitIconPosition? iconPosition) - { - var com = RenderComponent(parameters => - { - if (iconPosition.HasValue) - { - parameters.Add(p => p.IconPosition, iconPosition.Value); - } - }); - - var bitButton = com.Find(".bit-acb"); - - var expectedClassPresence = iconPosition == BitIconPosition.End; - - Assert.AreEqual(expectedClassPresence, bitButton.ClassList.Contains("bit-acb-eni")); - } - - [TestMethod, - DataRow("https://bitplatform.dev", BitLinkRels.NoOpener | BitLinkRels.NoReferrer, "noopener noreferrer"), - DataRow("#section", BitLinkRels.NoOpener | BitLinkRels.NoReferrer, null) - ] - public void BitActionButtonRelAttributeShouldFollowHrefRules(string href, BitLinkRels rel, string? expectedRel) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Href, href); - parameters.Add(p => p.Rel, rel); - }); - - var button = component.Find(".bit-acb"); - - var hasRelAttribute = button.HasAttribute("rel"); - - Assert.AreEqual(string.IsNullOrEmpty(expectedRel) is false, hasRelAttribute); - - if (expectedRel is not null) - { - Assert.AreEqual(expectedRel, button.GetAttribute("rel")); - } - } - - [TestMethod] - public void BitActionButtonShouldNotRenderIconWhenIconNameIsNull() - { - var component = RenderComponent(); - - Assert.IsEmpty(component.FindAll(".bit-acb-ico")); - } - - [TestMethod] - public void BitActionButtonShouldHideContentWhenIconOnlyIsTrue() - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IconOnly, true); - parameters.Add(p => p.IconName, "Emoji2"); - parameters.AddChildContent("content"); - }); - - var icon = component.Find(".bit-acb-ico"); - - Assert.IsNotNull(icon); - Assert.IsEmpty(component.FindAll(".bit-acb-con")); - } - - [TestMethod] - public void BitActionButtonShouldApplyCustomClasses() - { - var rootClass = "root-class"; - var iconClass = "icon-class"; - var contentClass = "content-class"; - - var classes = new BitActionButtonClassStyles - { - Root = rootClass, - Icon = iconClass, - Content = contentClass - }; - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IconName, "Add"); - parameters.Add(p => p.Classes, classes); - parameters.AddChildContent("bit"); - }); - - var button = component.Find(".bit-acb"); - var icon = component.Find(".bit-acb-ico"); - var content = component.Find(".bit-acb-con"); - - Assert.IsTrue(button.ClassList.Contains(rootClass)); - Assert.IsTrue(icon.ClassList.Contains(iconClass)); - Assert.IsTrue(content.ClassList.Contains(contentClass)); - } - - [TestMethod] - public void BitActionButtonShouldApplyCustomStyles() - { - var rootStyle = "color: red;"; - var iconStyle = "margin: 4px;"; - var contentStyle = "padding: 8px;"; - - var styles = new BitActionButtonClassStyles - { - Root = rootStyle, - Icon = iconStyle, - Content = contentStyle - }; - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IconName, "Share"); - parameters.Add(p => p.Styles, styles); - parameters.AddChildContent("bit"); - }); - - var button = component.Find(".bit-acb"); - var icon = component.Find(".bit-acb-ico"); - var content = component.Find(".bit-acb-con"); - - Assert.IsTrue(button.GetAttribute("style")?.Contains(rootStyle)); - Assert.AreEqual(iconStyle, icon.GetAttribute("style")); - Assert.AreEqual(contentStyle, content.GetAttribute("style")); - } - - [TestMethod] - public void BitActionButtonShouldRenderChildContentWhenIconOnlyIsFalse() - { - const string content = "Action content"; - - var component = RenderComponent(parameters => - { - parameters.AddChildContent(content); - parameters.Add(p => p.IconOnly, false); - }); - - var button = component.Find(".bit-acb"); - - Assert.Contains(content, button.TextContent); - } - - [TestMethod, - DataRow("data-value", "ID-123"), - DataRow("aria-test", "this is test") - ] - public void BitActionButtonShouldRespectArbitraryHtmlAttributes(string name, string value) - { - var component = RenderComponent((name, value)); - - var button = component.Find(".bit-acb"); - - Assert.AreEqual(value, button.GetAttribute(name)); - } - - [TestMethod] - public void BitActionButtonShouldRespectCascadingDir() - { - var component = RenderComponent>(parameters => - { - parameters.Add(p => p.Value, BitDir.Rtl); - parameters.AddChildContent(builder => - { - builder.OpenComponent(0); - builder.CloseComponent(); - }); - }); - - var button = component.Find(".bit-acb"); - - Assert.AreEqual("rtl", button.GetAttribute("dir")); - } - - [TestMethod] - public void BitActionButtonInsideEditFormShouldSubmitForm() - { - var submitted = false; - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Model, new object()); - parameters.Add(p => p.OnValidSubmit, _ => submitted = true); - parameters.Add(p => p.ChildContent, (RenderFragment)((ec) => builder => - { - builder.OpenComponent(0); - builder.CloseComponent(); - })); - }); - - var form = component.Find("form"); - - form.Submit(); - - Assert.IsTrue(submitted); - } - - [TestMethod] - public void BitActionButtonTargetAttributeShouldRenderWhenHrefProvided() - { - const string href = "https://bitplatform.dev"; - const string target = "_blank"; - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Href, href); - parameters.Add(p => p.Target, target); - }); - - var anchor = component.Find(".bit-acb"); - - Assert.AreEqual(target, anchor.GetAttribute("target")); - } - - [TestMethod] - public void BitActionButtonRenderPerformanceSmokeTest() - { - const int renderCount = 200; - - var stopwatch = Stopwatch.StartNew(); - - for (var i = 0; i < renderCount; i++) - { - RenderComponent(parameters => - { - parameters.Add(p => p.Title, $"title-{i}"); - parameters.Add(p => p.IconName, "Add"); - parameters.Add(p => p.IsEnabled, i % 3 != 0); - parameters.Add(p => p.FullWidth, i % 2 == 0); - parameters.Add(p => p.Href, i % 5 == 0 ? "https://bitplatform.dev" : null); - }); - } - - stopwatch.Stop(); - - Assert.IsTrue(stopwatch.Elapsed < TimeSpan.FromSeconds(5), - $"Rendering {renderCount} BitActionButton instances took {stopwatch.Elapsed}."); - } - - [TestMethod] - public void BitActionButtonDynamicParameterUpdateShouldRefreshMarkup() - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IsEnabled, true); - parameters.Add(p => p.IconName, "Add"); - }); - - var button = component.Find(".bit-acb"); - var icon = component.Find(".bit-acb-ico"); - - Assert.IsFalse(button.ClassList.Contains("bit-dis")); - Assert.IsTrue(icon.ClassList.Contains("bit-icon--Add")); - - component.SetParametersAndRender(parameters => - { - parameters.Add(p => p.IsEnabled, false); - parameters.Add(p => p.IconName, "Delete"); - }); - - Assert.IsTrue(button.ClassList.Contains("bit-dis")); - Assert.IsTrue(icon.ClassList.Contains("bit-icon--Delete")); - } - - [TestMethod] - public void BitActionButtonRelShouldUpdateWhenHrefChanges() - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Href, "https://bitplatform.dev/docs"); - parameters.Add(p => p.Rel, BitLinkRels.NoOpener | BitLinkRels.NoReferrer); - }); - - var anchor = component.Find(".bit-acb"); - - Assert.AreEqual("noopener noreferrer", anchor.GetAttribute("rel")); - - component.SetParametersAndRender(parameters => parameters.Add(p => p.Href, "#section-one")); - - anchor = component.Find(".bit-acb"); - Assert.IsFalse(anchor.HasAttribute("rel")); - - component.SetParametersAndRender(parameters => parameters.Add(p => p.Href, "/resources")); - - anchor = component.Find(".bit-acb"); - Assert.AreEqual("noopener noreferrer", anchor.GetAttribute("rel")); - } - - [TestMethod, - DataRow("5"), - DataRow("50"), - ] - public void BitActionButtonTabIndexShouldRecoverAfterReEnable(string tabIndex) - { - var component = RenderComponent(parameters => - { - parameters.Add(p => p.IsEnabled, false); - parameters.Add(p => p.TabIndex, tabIndex); - }); - - var button = component.Find(".bit-acb"); - - Assert.AreEqual("-1", button.GetAttribute("tabindex")); - - component.SetParametersAndRender(parameters => parameters.Add(p => p.IsEnabled, true)); - - Assert.AreEqual(tabIndex, button.GetAttribute("tabindex")); - } - - - private static string GetColorClass(BitColor? color) => color switch - { - BitColor.Primary => "bit-acb-pri", - BitColor.Secondary => "bit-acb-sec", - BitColor.Tertiary => "bit-acb-ter", - BitColor.Info => "bit-acb-inf", - BitColor.Success => "bit-acb-suc", - BitColor.Warning => "bit-acb-wrn", - BitColor.SevereWarning => "bit-acb-swr", - BitColor.Error => "bit-acb-err", - BitColor.PrimaryBackground => "bit-acb-pbg", - BitColor.SecondaryBackground => "bit-acb-sbg", - BitColor.TertiaryBackground => "bit-acb-tbg", - BitColor.PrimaryForeground => "bit-acb-pfg", - BitColor.SecondaryForeground => "bit-acb-sfg", - BitColor.TertiaryForeground => "bit-acb-tfg", - BitColor.PrimaryBorder => "bit-acb-pbr", - BitColor.SecondaryBorder => "bit-acb-sbr", - BitColor.TertiaryBorder => "bit-acb-tbr", - _ => "bit-acb-pri" - }; - - private static string GetSizeClass(BitSize? size) => size switch - { - BitSize.Small => "bit-acb-sm", - BitSize.Medium => "bit-acb-md", - BitSize.Large => "bit-acb-lg", - _ => "bit-acb-md" - }; -} diff --git a/src/BlazorUI/Bit.BlazorUI.Tests/Components/Extras/AppShell/BitAppShellTests.cs b/src/BlazorUI/Bit.BlazorUI.Tests/Components/Extras/AppShell/BitAppShellTests.cs deleted file mode 100644 index 4f8fef172b..0000000000 --- a/src/BlazorUI/Bit.BlazorUI.Tests/Components/Extras/AppShell/BitAppShellTests.cs +++ /dev/null @@ -1,131 +0,0 @@ -using System.Reflection; -using Bunit; -using Microsoft.AspNetCore.Components.Routing; -using Microsoft.VisualStudio.TestTools.UnitTesting; - -namespace Bit.BlazorUI.Tests.Components.Extras.AppShell; - -[TestClass] -public class BitAppShellTests : BunitTestContext -{ - [TestMethod] - public void BitAppShellShouldRenderStructureAndContent() - { - var component = RenderComponent(parameters => - { - parameters.AddChildContent("
Hello
"); - }); - - var root = component.Find(".bit-ash"); - Assert.IsNotNull(root); - - component.Find(".bit-ash-top"); - component.Find(".bit-ash-center"); - component.Find(".bit-ash-left"); - component.Find(".bit-ash-main"); - component.Find(".bit-ash-right"); - component.Find(".bit-ash-bottom"); - - var content = component.Find(".content"); - - Assert.AreEqual("Hello", content.TextContent); - } - - [TestMethod] - public void BitAppShellShouldRespectClassesAndStyles() - { - var classes = new BitAppShellClassStyles - { - Root = "root-class", - Top = "top-class", - Center = "center-class", - Left = "left-class", - Main = "main-class", - Right = "right-class", - Bottom = "bottom-class" - }; - - var styles = new BitAppShellClassStyles - { - Root = "margin:1px;", - Top = "padding:2px;", - Center = "gap:3px;", - Left = "width:4px;", - Main = "height:5px;", - Right = "border:6px solid transparent;", - Bottom = "background:red;" - }; - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.Classes, classes); - parameters.Add(p => p.Styles, styles); - }); - - var root = component.Find(".bit-ash"); - - Assert.IsTrue(root.ClassList.Contains("root-class")); - StringAssert.Contains(root.GetAttribute("style") ?? string.Empty, "margin:1px"); - - Assert.IsTrue(component.Find(".bit-ash-top").ClassList.Contains("top-class")); - Assert.IsTrue(component.Find(".bit-ash-center").ClassList.Contains("center-class")); - Assert.IsTrue(component.Find(".bit-ash-left").ClassList.Contains("left-class")); - Assert.IsTrue(component.Find(".bit-ash-main").ClassList.Contains("main-class")); - Assert.IsTrue(component.Find(".bit-ash-right").ClassList.Contains("right-class")); - Assert.IsTrue(component.Find(".bit-ash-bottom").ClassList.Contains("bottom-class")); - - StringAssert.Contains(component.Find(".bit-ash-top").GetAttribute("style") ?? string.Empty, "padding:2px"); - StringAssert.Contains(component.Find(".bit-ash-center").GetAttribute("style") ?? string.Empty, "gap:3px"); - StringAssert.Contains(component.Find(".bit-ash-left").GetAttribute("style") ?? string.Empty, "width:4px"); - StringAssert.Contains(component.Find(".bit-ash-main").GetAttribute("style") ?? string.Empty, "height:5px"); - StringAssert.Contains(component.Find(".bit-ash-right").GetAttribute("style") ?? string.Empty, "border:6px"); - StringAssert.Contains(component.Find(".bit-ash-bottom").GetAttribute("style") ?? string.Empty, "background:red"); - } - - [TestMethod] - public void BitAppShellShouldPersistScroll() - { - Context.JSInterop.SetupVoid("BitBlazorUI.AppShell.initScroll"); - Context.JSInterop.SetupVoid("BitBlazorUI.AppShell.locationChangedScroll"); - Context.JSInterop.SetupVoid("BitBlazorUI.AppShell.afterRenderScroll"); - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.PersistScroll, true); - }); - - Context.JSInterop.VerifyInvoke("BitBlazorUI.AppShell.initScroll"); - - InvokeLocationChanged(component.Instance, "https://example.com/page2"); - - Context.JSInterop.VerifyInvoke("BitBlazorUI.AppShell.locationChangedScroll"); - - component.Render(); // trigger OnAfterRenderAsync for non-first render - - Context.JSInterop.VerifyInvoke("BitBlazorUI.AppShell.afterRenderScroll"); - } - - [TestMethod] - public void BitAppShellShouldGoToTopWhenAutoGoToTop() - { - Context.JSInterop.SetupVoid("BitBlazorUI.Extras.goToTop"); - - var component = RenderComponent(parameters => - { - parameters.Add(p => p.AutoGoToTop, true); - }); - - InvokeLocationChanged(component.Instance, "https://example.com/other"); - - Context.JSInterop.VerifyInvoke("BitBlazorUI.Extras.goToTop"); - } - - private static void InvokeLocationChanged(BitAppShell instance, string uri) - { - var method = instance.GetType().GetMethod("LocationChanged", BindingFlags.Instance | BindingFlags.NonPublic); - - Assert.IsNotNull(method); - - method!.Invoke(instance, new object?[] { null, new LocationChangedEventArgs(uri, false) }); - } -} diff --git a/src/BlazorUI/Bit.BlazorUI.Web.slnf b/src/BlazorUI/Bit.BlazorUI.Web.slnf index 6b4261c28c..55cdd71702 100644 --- a/src/BlazorUI/Bit.BlazorUI.Web.slnf +++ b/src/BlazorUI/Bit.BlazorUI.Web.slnf @@ -2,12 +2,12 @@ "solution": { "path": "Bit.BlazorUI.slnx", "projects": [ + "Bit.BlazorUI\\Bit.BlazorUI.csproj", "Bit.BlazorUI.Assets\\Bit.BlazorUI.Assets.csproj", "Bit.BlazorUI.Extras\\Bit.BlazorUI.Extras.csproj", "Bit.BlazorUI.Icons\\Bit.BlazorUI.Icons.csproj", "Bit.BlazorUI.SourceGenerators\\Bit.BlazorUI.SourceGenerators.csproj", - "Bit.BlazorUI.Tests\\Bit.BlazorUI.Tests.csproj", - "Bit.BlazorUI\\Bit.BlazorUI.csproj", + "Tests\\Bit.BlazorUI.Tests\\Bit.BlazorUI.Tests.csproj", "Demo\\Bit.BlazorUI.Demo.Server\\Bit.BlazorUI.Demo.Server.csproj", "Demo\\Bit.BlazorUI.Demo.Shared\\Bit.BlazorUI.Demo.Shared.csproj", "Demo\\Client\\Bit.BlazorUI.Demo.Client.Core\\Bit.BlazorUI.Demo.Client.Core.csproj", diff --git a/src/BlazorUI/Bit.BlazorUI.slnx b/src/BlazorUI/Bit.BlazorUI.slnx index 683dcfb98b..ece4313cf9 100644 --- a/src/BlazorUI/Bit.BlazorUI.slnx +++ b/src/BlazorUI/Bit.BlazorUI.slnx @@ -28,10 +28,17 @@
+ + + + + + + + - diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor index edf7f9b814..f97a726780 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor +++ b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor @@ -1,6 +1,11 @@ -@namespace Bit.BlazorUI +@namespace Bit.BlazorUI @inherits BitComponentBase +@{ + RenderFragment b = @
@(Body ?? ChildContent)
; + var body = ((Body ?? ChildContent) is not null && IconOnly is false) ? b : null; +} + @if (Href.HasNoValue()) { } else @@ -36,22 +56,20 @@ else class="@ClassBuilder.Value" dir="@Dir?.ToString().ToLower()" disabled="@(IsEnabled is false)" - href="@(IsEnabled ? Href : null)" + href="@(IsEnabled? Href : null)" title="@Title" target="@Target" tabindex="@_tabIndex" aria-label="@AriaLabel" aria-hidden="@AriaHidden" aria-describedby="@AriaDescription"> - @if (IconName is not null) - { - - } - @if (ChildContent is not null && IconOnly is false) - { -
- @ChildContent -
- } + + @if (IconName is not null) + { + + } + + @body + } \ No newline at end of file diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor.cs b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor.cs index b2a7093600..af250a4176 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor.cs +++ b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.razor.cs @@ -1,4 +1,5 @@ -using Microsoft.AspNetCore.Components.Forms; +using System.Diagnostics.CodeAnalysis; +using Microsoft.AspNetCore.Components.Forms; namespace Bit.BlazorUI; @@ -9,6 +10,7 @@ public partial class BitActionButton : BitComponentBase { private string? _rel; private string? _tabIndex; + private bool _isLoading; private BitButtonType _buttonType; @@ -19,6 +21,17 @@ public partial class BitActionButton : BitComponentBase /// [CascadingParameter] public EditContext? EditContext { get; set; } + /// + /// Gets or sets the cascading parameters for the action button component. + /// + /// + /// This property receives its value from an ancestor component via Blazor's cascading parameter mechanism. + ///
+ /// The intended use is to allow shared configuration or settings to be applied to multiple action button components through the component. + ///
+ [CascadingParameter(Name = BitActionButtonParams.ParamName)] + public BitActionButtonParams? CascadingParameters { get; set; } + /// @@ -41,6 +54,11 @@ public partial class BitActionButton : BitComponentBase /// [Parameter] public BitButtonType? ButtonType { get; set; } + /// + /// Alias for , the custom body of the action button (text and/or any render fragment). + /// + [Parameter] public RenderFragment? Body { get; set; } + /// /// The custom body of the action button (text and/or any render fragment). /// @@ -100,6 +118,17 @@ public partial class BitActionButton : BitComponentBase [Parameter, ResetClassBuilder] public BitIconPosition? IconPosition { get; set; } + /// + /// Determines whether the action button is in loading mode or not. + /// + [Parameter, ResetClassBuilder] + public bool IsLoading { get; set; } + + /// + /// The custom template used to replace the default loading indicator inside the action button in the loading state. + /// + [Parameter] public RenderFragment? LoadingTemplate { get; set; } + /// /// Gets or sets the callback that is invoked when the component is clicked. /// @@ -109,16 +138,6 @@ public partial class BitActionButton : BitComponentBase /// [Parameter] public EventCallback OnClick { get; set; } - /// - /// Gets or sets the custom CSS inline styles to apply to the action button component. - /// - /// - /// Use this property to override the default styles of the action button. - /// If not set, the component uses its built-in styling. - /// This property is typically used to provide additional visual customization. - /// - [Parameter] public BitActionButtonClassStyles? Styles { get; set; } - /// /// Gets or sets the relationship type between the current element and the linked resource, as defined by the link's rel attribute. /// @@ -140,6 +159,16 @@ public partial class BitActionButton : BitComponentBase [Parameter, ResetClassBuilder] public BitSize? Size { get; set; } + /// + /// Gets or sets the custom CSS inline styles to apply to the action button component. + /// + /// + /// Use this property to override the default styles of the action button. + /// If not set, the component uses its built-in styling. + /// This property is typically used to provide additional visual customization. + /// + [Parameter] public BitActionButtonClassStyles? Styles { get; set; } + /// /// Gets or sets the name of the target frame or window for the navigation action when the action button renders as an anchor (by providing the Href parameter). /// @@ -155,6 +184,12 @@ public partial class BitActionButton : BitComponentBase /// [Parameter] public string? Title { get; set; } + /// + /// Adds an underline to the action button text, useful for link-style buttons. + /// + [Parameter, ResetClassBuilder] + public bool Underlined { get; set; } + protected override string RootElementClass => "bit-acb"; @@ -187,6 +222,8 @@ protected override void RegisterCssClasses() ClassBuilder.Register(() => FullWidth ? "bit-acb-fwi" : string.Empty); + ClassBuilder.Register(() => IsLoading ? "bit-acb-lod" : string.Empty); + ClassBuilder.Register(() => Size switch { BitSize.Small => "bit-acb-sm", @@ -195,6 +232,8 @@ protected override void RegisterCssClasses() _ => "bit-acb-md" }); + ClassBuilder.Register(() => Underlined ? "bit-acb-und" : string.Empty); + ClassBuilder.Register(() => IconPosition is BitIconPosition.End ? "bit-acb-eni" : string.Empty); } @@ -203,14 +242,19 @@ protected override void RegisterCssStyles() StyleBuilder.Register(() => Styles?.Root); } + [DynamicDependency(DynamicallyAccessedMemberTypes.All, typeof(BitActionButtonParams))] protected override void OnParametersSet() { + CascadingParameters?.UpdateParameters(this); + _tabIndex = IsEnabled ? TabIndex : AllowDisabledFocus ? TabIndex : "-1"; _buttonType = ButtonType ?? (EditContext is null ? BitButtonType.Button : BitButtonType.Submit); + _isLoading = IsLoading; + base.OnParametersSet(); } @@ -226,7 +270,7 @@ protected virtual async Task HandleOnClick(MouseEventArgs e) - private void OnSetHrefAndRel() + internal void OnSetHrefAndRel() { if (Rel.HasValue is false || Href.HasNoValue() || Href!.StartsWith('#')) { diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.scss b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.scss index 61f4b5d06b..2419f9bc96 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.scss +++ b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButton.scss @@ -4,12 +4,9 @@ border: none; display: flex; cursor: pointer; - gap: spacing(1); color: $clr-fg-pri; - align-items: center; text-decoration: none; box-sizing: border-box; - justify-content: center; font-family: $tg-font-family; font-weight: $tg-font-weight; background-color: transparent; @@ -46,11 +43,39 @@ color: var(--bit-acb-clr-ico); } +.bit-acb-spn { + border-radius: 50%; + border-width: spacing(0.2125); + width: var(--bit-acb-spn-size); + height: var(--bit-acb-spn-size); + border-color: $clr-brd-sec; + border-style: $shp-border-style; + border-top-color: var(--bit-acb-clr-ico); + animation: bit-acb-spin 1.3s linear infinite; + animation-timing-function: cubic-bezier(0.53, 0.21, 0.29, 0.67); +} + +@keyframes bit-acb-spin { + 0% { + transform: rotate(0deg); + } + + 100% { + transform: rotate(360deg); + } +} + .bit-acb-con { flex-grow: 1; display: flex; } +.bit-acb-inn { + display: inline-flex; + align-items: center; + gap: spacing(1); +} + .bit-acb-eni { flex-direction: row-reverse; } @@ -60,6 +85,34 @@ justify-content: flex-start; } +.bit-acb-lod { + cursor: default; + pointer-events: none; +} + +.bit-acb-und { + .bit-acb-inn { + text-decoration: none; + position: relative; + + &::after { + left: 0; + right: 0; + bottom: 0; + height: 1px; + content: ""; + position: absolute; + background-color: currentColor; + } + + @media (hover: hover) { + &:hover::after { + height: 2px; + } + } + } +} + .bit-acb-pri { --bit-acb-clr-ico: #{$clr-pri}; @@ -167,15 +220,18 @@ .bit-acb-sm { --bit-acb-fontsize: #{spacing(1.25)}; + --bit-acb-spn-size: #{spacing(1.25)}; --bit-acb-padding: #{spacing(0.50)} #{spacing(1)}; } .bit-acb-md { --bit-acb-fontsize: #{spacing(1.75)}; + --bit-acb-spn-size: #{spacing(1.75)}; --bit-acb-padding: #{spacing(0.75)} #{spacing(1.5)}; } .bit-acb-lg { --bit-acb-fontsize: #{spacing(2.25)}; + --bit-acb-spn-size: #{spacing(2.25)}; --bit-acb-padding: #{spacing(1.00)} #{spacing(2.00)}; -} +} \ No newline at end of file diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButtonClassStyles.cs b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButtonClassStyles.cs index d8b37e614f..c340dc4c95 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButtonClassStyles.cs +++ b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButtonClassStyles.cs @@ -19,4 +19,9 @@ public class BitActionButtonClassStyles /// Custom class or style applied to the content container. /// public string? Content { get; set; } + + /// + /// Custom class or style applied to the loading spinner element. + /// + public string? Spinner { get; set; } } diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButtonParams.cs b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButtonParams.cs new file mode 100644 index 0000000000..fe9665c91c --- /dev/null +++ b/src/BlazorUI/Bit.BlazorUI/Components/Buttons/ActionButton/BitActionButtonParams.cs @@ -0,0 +1,283 @@ +namespace Bit.BlazorUI; + +/// +/// The parameters for component. +/// +public class BitActionButtonParams : BitComponentBaseParams, IBitComponentParams +{ + /// + /// Represents the parameter name used to identify the cascading parameters within . + /// + /// + /// This constant is typically used when referencing or accessing the BitActionButton value in + /// parameterized APIs or configuration settings. Using this constant helps ensure consistency and reduces the risk + /// of typographical errors. + /// + public const string ParamName = $"{nameof(BitParams)}.{nameof(BitActionButton)}"; + + + + public string Name => ParamName; + + + + /// + /// Keeps the disabled action button focusable by not forcing a negative tabindex when is false. + /// + public bool? AllowDisabledFocus { get; set; } + + /// + /// Detailed description of the button for the benefit of screen readers (rendered into aria-describedby). + /// + public string? AriaDescription { get; set; } + + /// + /// If true, adds an aria-hidden attribute instructing screen readers to ignore the button. + /// + public bool? AriaHidden { get; set; } + + /// + /// The type of the button element; defaults to submit inside an otherwise button. + /// + public BitButtonType? ButtonType { get; set; } + + /// + /// Custom CSS classes for the root, icon, and content sections of the action button. + /// + public BitActionButtonClassStyles? Classes { get; set; } + + /// + /// The general color of the button that applies to the icon and text of the action button. + /// + public BitColor? Color { get; set; } + + /// + /// Gets or sets a value indicating whether the component should expand to occupy the full available width. + /// + public bool? FullWidth { get; set; } + + /// + /// The value of the href attribute of the link rendered by the button. + /// If provided, the component will be rendered as an anchor tag instead of button. + /// + public string? Href { get; set; } + + /// + /// Gets or sets the name of the icon to display. + /// + /// + /// The icon name should be from the Fluent UI icon set (e.g., BitIconName.AddFriend). + ///
+ /// Browse available names in BitIconName of the Bit.BlazorUI.Icons nuget package or the gallery: + /// . + ///
+ /// The value is case-sensitive and must match a valid icon identifier. + /// If not set or set to null, no icon will be rendered. + ///
+ public string? IconName { get; set; } + + /// + /// Gets or sets a value indicating whether only the icon is displayed, without accompanying text. + /// + /// + /// Set this property to to render the component with only its icon visible. + /// When , both icon and text are shown if available. + /// + public bool? IconOnly { get; set; } + + /// + /// Gets or sets the position of the icon relative to the component's content. + /// + public BitIconPosition? IconPosition { get; set; } + + /// + /// Determines whether the action button is in loading mode or not. + /// + public bool? IsLoading { get; set; } + + /// + /// Gets or sets the relationship type between the current element and the linked resource, as defined by the link's rel attribute. + /// + /// + /// Sets the rel attribute for link-rendered buttons when is a non-anchor URL; ignored for empty or hash-only hrefs. + /// The rel attribute specifies the relationship between the current document and the linked document. + ///
+ /// Set this property to specify how the linked resource is related to the current context. + /// Common values include "stylesheet", "noopener", or "nofollow". The value determines how browsers and search + /// engines interpret the link. + ///
+ public BitLinkRels? Rel { get; set; } + + /// + /// Sets the preset size (Small, Medium, Large) for typography and padding of the action button. + /// + public BitSize? Size { get; set; } + + /// + /// Gets or sets the custom CSS inline styles to apply to the action button component. + /// + /// + /// Use this property to override the default styles of the action button. + /// If not set, the component uses its built-in styling. + /// This property is typically used to provide additional visual customization. + /// + public BitActionButtonClassStyles? Styles { get; set; } + + /// + /// Gets or sets the name of the target frame or window for the navigation action when the action button renders as an anchor (by providing the Href parameter). + /// + /// + /// Specify a value to control where the linked content will be displayed. Common values include + /// "_blank" to open in a new window or tab, "_self" for the same frame, "_parent" for the parent frame, and "_top" + /// for the full body of the window. If not set, the default browser behavior is used. + /// + public string? Target { get; set; } + + /// + /// The tooltip to show when the mouse is placed on the button. + /// + public string? Title { get; set; } + + /// + /// Adds an underline to the action button text, useful for link-style buttons. + /// + public bool? Underlined { get; set; } + + + + /// + /// Updates the properties of the specified instance with any values that have been set on + /// this object, if those properties have not already been set on the . + /// + /// + /// Only properties that have a value set and have not already been set on the will be updated. + /// This method does not overwrite existing values on . + /// + /// + /// The instance whose properties will be updated. Cannot be null. + /// + public void UpdateParameters(BitActionButton bitActionButton) + { + if (bitActionButton is null) return; + + UpdateBaseParameters(bitActionButton); + + if (AllowDisabledFocus.HasValue && bitActionButton.HasNotBeenSet(nameof(AllowDisabledFocus))) + { + bitActionButton.AllowDisabledFocus = AllowDisabledFocus.Value; + } + + if (AriaDescription.HasValue() && bitActionButton.HasNotBeenSet(nameof(AriaDescription))) + { + bitActionButton.AriaDescription = AriaDescription; + } + + if (AriaHidden.HasValue && bitActionButton.HasNotBeenSet(nameof(AriaHidden))) + { + bitActionButton.AriaHidden = AriaHidden.Value; + } + + if (ButtonType.HasValue && bitActionButton.HasNotBeenSet(nameof(ButtonType))) + { + bitActionButton.ButtonType = ButtonType.Value; + } + + if (Classes is not null && bitActionButton.HasNotBeenSet(nameof(Classes))) + { + bitActionButton.Classes = Classes; + + bitActionButton.ClassBuilder.Reset(); + } + + if (Color.HasValue && bitActionButton.HasNotBeenSet(nameof(Color))) + { + bitActionButton.Color = Color.Value; + + bitActionButton.ClassBuilder.Reset(); + } + + if (FullWidth.HasValue && bitActionButton.HasNotBeenSet(nameof(FullWidth))) + { + bitActionButton.FullWidth = FullWidth.Value; + + bitActionButton.ClassBuilder.Reset(); + } + + bool hrefWasSet = false; + bool relWasSet = false; + + if (Href.HasValue() && bitActionButton.HasNotBeenSet(nameof(Href))) + { + bitActionButton.Href = Href; + + hrefWasSet = true; + } + + if (IconName.HasValue() && bitActionButton.HasNotBeenSet(nameof(IconName))) + { + bitActionButton.IconName = IconName; + } + + if (IconOnly.HasValue && bitActionButton.HasNotBeenSet(nameof(IconOnly))) + { + bitActionButton.IconOnly = IconOnly.Value; + } + + if (IconPosition.HasValue && bitActionButton.HasNotBeenSet(nameof(IconPosition))) + { + bitActionButton.IconPosition = IconPosition.Value; + + bitActionButton.ClassBuilder.Reset(); + } + + if (IsLoading.HasValue && bitActionButton.HasNotBeenSet(nameof(IsLoading))) + { + bitActionButton.IsLoading = IsLoading.Value; + + bitActionButton.ClassBuilder.Reset(); + } + + if (Rel.HasValue && bitActionButton.HasNotBeenSet(nameof(Rel))) + { + bitActionButton.Rel = Rel.Value; + + relWasSet = true; + } + + // Call OnSetHrefAndRel if either Href or Rel was set, to update the _rel field + if (hrefWasSet || relWasSet) + { + bitActionButton.OnSetHrefAndRel(); + } + + if (Size.HasValue && bitActionButton.HasNotBeenSet(nameof(Size))) + { + bitActionButton.Size = Size.Value; + + bitActionButton.ClassBuilder.Reset(); + } + + if (Styles is not null && bitActionButton.HasNotBeenSet(nameof(Styles))) + { + bitActionButton.Styles = Styles; + } + + if (Target.HasValue() && bitActionButton.HasNotBeenSet(nameof(Target))) + { + bitActionButton.Target = Target; + } + + if (Title.HasValue() && bitActionButton.HasNotBeenSet(nameof(Title))) + { + bitActionButton.Title = Title; + } + + if (Underlined.HasValue && bitActionButton.HasNotBeenSet(nameof(Underlined))) + { + bitActionButton.Underlined = Underlined.Value; + + bitActionButton.ClassBuilder.Reset(); + } + } +} + diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/CircularTimePicker/BitCircularTimePicker.razor.cs b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/CircularTimePicker/BitCircularTimePicker.razor.cs index 17f22ef1d4..345f455392 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/CircularTimePicker/BitCircularTimePicker.razor.cs +++ b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/CircularTimePicker/BitCircularTimePicker.razor.cs @@ -254,7 +254,10 @@ public async Task _HandlePointerMove(MouseEventArgs e) - public Task OpenCallout() => HandleOnClick(); + public Task OpenCallout() + { + return HandleOnClick(); + } @@ -382,14 +385,6 @@ private void OnSetCulture() _culture = Culture ?? CultureInfo.CurrentUICulture; } - private string GetTransformStyle(int index, double radius, double offsetX, double offsetY) - { - double angle = (6 - index) * 30 / 180d * Math.PI; - var x = Math.Sin(angle) * radius + offsetX; - var y = (Math.Cos(angle) + 1) * radius + offsetY; - return $"{x:F3}px, {y:F3}px"; - } - private string GetHoursMinutesClass(int hourMinute) { StringBuilder classes = new(); @@ -436,9 +431,19 @@ private string GetHoursMinutesStyle(int hourMinute, int index, double radius, do return styles.ToString(); } - private int GetClockHandHeightPercent() => (_showHourView && TimeFormat == BitTimeFormat.TwentyFourHours && _hour > 0 && _hour < 13) ? 26 : 40; + private int GetClockHandHeightPercent() + { + return (_showHourView && TimeFormat == BitTimeFormat.TwentyFourHours && _hour > 0 && _hour < 13) + ? 26 + : 40; + } - private double GetPointerDegree() => _showHourView ? ((_hour.GetValueOrDefault() * 30) % 360) : ((_minute.GetValueOrDefault() * 6) % 360); + private double GetPointerDegree() + { + return _showHourView + ? (_hour.GetValueOrDefault() * 30 % 360) + : (_minute.GetValueOrDefault() * 6 % 360); + } private async Task HandleOnPointerDown(MouseEventArgs e) { @@ -449,7 +454,9 @@ private async Task HandleOnPointerDown(MouseEventArgs e) private async Task UpdateCurrentValue() { - CurrentValue = (_hour.HasValue is false || _minute.HasValue is false) ? null : new TimeSpan(_hour.Value, _minute.Value, 0); + CurrentValue = _hour.HasValue is true && _minute.HasValue is true + ? new TimeSpan(_hour.Value, _minute.Value, 0) + : null; await OnSelectTime.InvokeAsync(CurrentValue); } @@ -462,15 +469,19 @@ private string GetHourString() return Math.Min(23, Math.Max(0, hours)).ToString(CultureInfo.InvariantCulture); } - private string GetMinuteString() => _minute.HasValue ? $"{Math.Min(59, Math.Max(0, _minute.Value)):D2}" : "--"; - - private int GetAmPmHours(int hours) + private string GetMinuteString() { - var result = hours % 12; - return result == 0 ? 12 : result; + return _minute.HasValue + ? $"{Math.Min(59, Math.Max(0, _minute.Value)):D2}" + : "--"; } - private int GetHours() => TimeFormat == BitTimeFormat.TwelveHours ? GetAmPmHours(_hour.GetValueOrDefault()) : _hour.GetValueOrDefault(); + private int GetHours() + { + return TimeFormat == BitTimeFormat.TwelveHours + ? GetAmPmHours(_hour.GetValueOrDefault()) + : _hour.GetValueOrDefault(); + } private void HandleOnHourClick() { @@ -515,7 +526,10 @@ private void HandleOnValueChanged(object? sender, EventArgs args) _minute = CurrentValue?.Minutes; } - private bool IsAm() => _hour.GetValueOrDefault() >= 00 && _hour < 12; // am is 00:00 to 11:59 + private bool IsAm() + { + return _hour.GetValueOrDefault() >= 00 && _hour < 12; // am is 00:00 to 11:59 + } private string GetValueFormat() { @@ -666,6 +680,22 @@ private string GetCalloutCssClasses() + private static string GetTransformStyle(int index, double radius, double offsetX, double offsetY) + { + double angle = (6 - index) * 30 / 180d * Math.PI; + var x = Math.Sin(angle) * radius + offsetX; + var y = (Math.Cos(angle) + 1) * radius + offsetY; + return FormattableString.Invariant($"{x:F3}px, {y:F3}px"); + } + + private static int GetAmPmHours(int hours) + { + var result = hours % 12; + return result == 0 ? 12 : result; + } + + + /// protected override bool TryParseValueFromString(string? value, [MaybeNullWhen(false)] out TimeSpan? result, [NotNullWhen(false)] out string? validationErrorMessage) diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor index 77ad9fc913..8f5fd05bae 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor +++ b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor @@ -1,6 +1,13 @@ -@namespace Bit.BlazorUI +@namespace Bit.BlazorUI @inherits BitComponentBase +@{ + var thumbStyle = FormattableString.Invariant($"top:{_saturationPickerThumbPosition?.Top}px;left:{_saturationPickerThumbPosition?.Left}px;background-color:{Rgb}"); + + var alphaSliderBg = "url()"; + var alphaSliderStyle = $"background:linear-gradient(to left,{Rgb} 0%, transparent 100%), {alphaSliderBg};"; +} +
-
+
@@ -45,8 +51,7 @@ @if (ShowAlphaSlider) { -
+
+ value="@(FormattableString.Invariant($"{_color.A}"))" + aria-valuenow="@(FormattableString.Invariant($"{_color.A}"))" + aria-valuetext="@(FormattableString.Invariant($"{_color.A}"))">
}
diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor.cs b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor.cs index 60f7a55a16..638e91e4c4 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor.cs +++ b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/ColorPicker/BitColorPicker.razor.cs @@ -1,7 +1,11 @@ -namespace Bit.BlazorUI; +using System.Globalization; + +namespace Bit.BlazorUI; /// -/// The color picker (ColorPicker) is used to browse through and select colors. By default, it lets people navigate through colors on a color spectrum, or specify a color in either Red-Green-Blue (RGB), or alpha color code; or Hexadecimal textboxes. +/// The color picker (ColorPicker) is used to browse through and select colors. +/// By default, it lets people navigate through colors on a color spectrum, +/// or specify a color in either Red-Green-Blue (RGB), or alpha color code; or Hexadecimal textboxes. /// public partial class BitColorPicker : BitComponentBase { @@ -35,6 +39,7 @@ public double Alpha if (_color.A == value) return; _color.A = value; + AlphaChanged.InvokeAsync(value); } } @@ -48,7 +53,7 @@ public string Color get => _colorType == BitInternalColorType.Hex ? _color.Hex! : _color.Rgb!; set { - _colorType = value.HasValue() && value.StartsWith("#", StringComparison.InvariantCultureIgnoreCase) + _colorType = value.HasValue() && value.StartsWith('#') ? BitInternalColorType.Hex : BitInternalColorType.Rgb; @@ -85,10 +90,15 @@ public string Color public string? Hex => _color.Hex; + public string? Rgb => FormattableString.Invariant($"rgb({_color.R},{_color.G},{_color.B})"); + public string? Rgba => FormattableString.Invariant($"rgba({_color.R},{_color.G},{_color.B},{_color.A})"); + public (double Hue, double Saturation, double Value) Hsv => _color.Hsv; + + [JSInvokable(nameof(HandlePointerUp))] public void HandlePointerUp(MouseEventArgs e) { @@ -132,6 +142,7 @@ protected override async Task OnAfterRenderAsync(bool firstRender) private async Task SetSaturationPickerThumbPositionAsync() { var (_, saturation, value) = _color.Hsv; + var saturationPickerRect = await _js.BitUtilsGetBoundingClientRect(_saturationPickerRef); var width = saturationPickerRect?.Width ?? 0; @@ -145,6 +156,7 @@ private async Task SetSaturationPickerThumbPositionAsync() private void SetSaturationPickerStyle() { var rgb = BitInternalColor.ToRgb(_selectedHue, 1, 1).ToString(); + _saturationPickerStyle = $"background-color:rgb{rgb}"; } @@ -153,14 +165,17 @@ private async Task UpdateColor(MouseEventArgs e) if (ColorHasBeenSet && ColorChanged.HasDelegate is false) return; var pickerRect = await _js.BitUtilsGetBoundingClientRect(_saturationPickerRef); - var left = e.ClientX < pickerRect.Left ? 0 + + var left = e.ClientX < pickerRect.Left + ? 0 : e.ClientX > pickerRect.Left + pickerRect.Width - ? pickerRect.Width - : (e.ClientX - pickerRect.Left); - var top = e.ClientY < pickerRect.Top ? 0 + ? pickerRect.Width + : (e.ClientX - pickerRect.Left); + var top = e.ClientY < pickerRect.Top + ? 0 : e.ClientY > pickerRect.Top + pickerRect.Height - ? pickerRect.Height - : (e.ClientY - pickerRect.Top); + ? pickerRect.Height + : (e.ClientY - pickerRect.Top); _saturationPickerThumbPosition = new(left, top); @@ -168,6 +183,7 @@ private async Task UpdateColor(MouseEventArgs e) _selectedValue = Math.Clamp((pickerRect.Height - e.ClientY + pickerRect.Top) / pickerRect.Height, 0, 1); _color.Update(_selectedHue, _selectedSaturation, _selectedValue, _color.A); + var colorValue = _colorType == BitInternalColorType.Hex ? _color.Hex : _color.Rgb; SetSaturationPickerStyle(); @@ -183,7 +199,8 @@ private async Task HandleOnHueInput(ChangeEventArgs args) { if (ColorHasBeenSet && ColorChanged.HasDelegate is false) return; - _selectedHue = Convert.ToDouble(args.Value); + _selectedHue = Convert.ToDouble(args.Value, CultureInfo.InvariantCulture); + _color.Update(_selectedHue, _selectedSaturation, _selectedValue, _color.A); var colorValue = _colorType == BitInternalColorType.Hex ? _color.Hex : _color.Rgb; @@ -199,7 +216,8 @@ private async Task HandleOnAlphaInput(ChangeEventArgs args) { if (AlphaHasBeenSet && AlphaChanged.HasDelegate is false) return; - _color.A = Convert.ToDouble(args.Value); + _color.A = Convert.ToDouble(args.Value, CultureInfo.InvariantCulture); + var colorValue = _colorType == BitInternalColorType.Hex ? _color.Hex : _color.Rgb; await ColorChanged.InvokeAsync(colorValue); @@ -210,20 +228,20 @@ private async Task HandleOnAlphaInput(ChangeEventArgs args) private async Task HandleOnSaturationPickerPointerDown(MouseEventArgs e) { _saturationPickerPointerDown = true; + await UpdateColor(e); } private string GetRootElAriaLabel() { - var ariaLabel = $"Color picker, Red {_color.R} Green {_color.G} Blue {_color.B} "; + var ariaLabel = $"Color picker, Red {_color.R} Green {_color.G} Blue {_color.B}"; + if (ShowAlphaSlider) { - ariaLabel += $"Alpha {_color.A * 100}% selected."; - } - else - { - ariaLabel += "selected."; + ariaLabel += FormattableString.Invariant($" and Alpha {_color.A * 100}%"); } + + ariaLabel += " selected."; return ariaLabel; } diff --git a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/DatePicker/BitDatePicker.razor b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/DatePicker/BitDatePicker.razor index f0597b9ad2..4c7526594b 100644 --- a/src/BlazorUI/Bit.BlazorUI/Components/Inputs/DatePicker/BitDatePicker.razor +++ b/src/BlazorUI/Bit.BlazorUI/Components/Inputs/DatePicker/BitDatePicker.razor @@ -381,7 +381,7 @@ var monthName = _culture.DateTimeFormat.GetMonthName(month); var disabled = IsEnabled is false || IsMonthOutOfMinAndMaxDate(month); var selected = month == _currentMonth; -