diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml
index 062bf71d89..6ba3ad7a6d 100644
--- a/.github/workflows/build-and-test.yml
+++ b/.github/workflows/build-and-test.yml
@@ -17,7 +17,7 @@ jobs:
- uses: actions/checkout@v2
- uses: actions/setup-dotnet@master
with:
- dotnet-version: 5.0.100-rc.2.20479.15
+ dotnet-version: 5.0.100
- name: Build All
run: .\build-all.ps1
diff --git a/.gitignore b/.gitignore
index 5b8dfe9c11..b9576fda4c 100644
--- a/.gitignore
+++ b/.gitignore
@@ -6,6 +6,7 @@
*.user
*.userosscache
*.sln.docstates
+*.editorconfig
# User-specific files (MonoDevelop/Xamarin Studio)
*.userprefs
diff --git a/Directory.Build.props b/Directory.Build.props
new file mode 100644
index 0000000000..dea0a6c6f7
--- /dev/null
+++ b/Directory.Build.props
@@ -0,0 +1,29 @@
+
+
+
+
+ 5.0.0
+
+
+ 16.6.1
+
+
+ 4.2.2
+
+
+ 3.0.2
+
+
+ 2.4.1
+
+
+ 2.4.1
+
+
+ 2.4.2
+
+
+ 2.2.14
+
+
+
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Account/Localization/Resources/de-DE.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Account/Localization/Resources/de-DE.json
new file mode 100644
index 0000000000..b0418f082f
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Account/Localization/Resources/de-DE.json
@@ -0,0 +1,14 @@
+{
+ "culture": "de-DE",
+ "texts": {
+ "Account": "ABP Benutzerkonto - Anmeldung & Registrierung | ABP.IO",
+ "Welcome": "Willkommen",
+ "UseOneOfTheFollowingLinksToContinue": "Nutzen Sie einen der nachfolgenden Links um fortzusetzen",
+ "FrameworkHomePage": "Framework Website",
+ "FrameworkDocumentation": "Framework Dokumentation",
+ "OfficialBlog": "Offizieller Blog",
+ "CommercialHomePage": "Commercial Website",
+ "CommercialSupportWebSite": "Commercial Support-Website",
+ "CommunityWebSite": "ABP Community-Website"
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/de-DE.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/de-DE.json
new file mode 100644
index 0000000000..60725c437b
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/de-DE.json
@@ -0,0 +1,199 @@
+{
+ "culture": "de-DE",
+ "texts": {
+ "Permission:Organizations": "Organisationen",
+ "Permission:Manage": "Organisationen verwalten",
+ "Permission:DiscountRequests": "Rabattanfragen",
+ "Permission:DiscountManage": "Rabattanfragen verwalten",
+ "Permission:Disable": "Deaktivieren",
+ "Permission:Enable": "Aktivieren",
+ "Permission:EnableSendEmail": "E-Mail-Senden aktivieren",
+ "Permission:SendEmail": "E-Mail senden",
+ "Permission:NpmPackages": "NPM-Pakete",
+ "Permission:NugetPackages": "Nuget-Pakete",
+ "Permission:Maintenance": "Wartung",
+ "Permission:Maintain": "Warten",
+ "Permission:ClearCaches": "Caches leeren",
+ "Permission:Modules": "Module",
+ "Permission:Packages": "Pakete",
+ "Permission:Edit": "Bearbeiten",
+ "Permission:Delete": "Löschen",
+ "Permission:Create": "Erstellen",
+ "Permission:Accounting": "Abrechnung",
+ "Permission:Accounting:Quotation": "Angebot",
+ "Permission:Accounting:Invoice": "Rechnung",
+ "Menu:Organizations": "Organisationen",
+ "Menu:Accounting": "Abrechnung",
+ "Menu:Packages": "Pakete",
+ "Menu:DiscountRequests": "Rabattanfragen",
+ "NpmPackageDeletionWarningMessage": "Dieses NPM-Paket wird entfernt. Bestätigen Sie das?",
+ "NugetPackageDeletionWarningMessage": "Dieses Nuget-Paket wird entfernt. Bestägiten Sie das?",
+ "ModuleDeletionWarningMessage": "Dieses Modul wird entfernt. Bestätigen Sie das?",
+ "Name": "Name",
+ "DisplayName": "Anzeigename",
+ "ShortDescription": "Kurzbeschreibung",
+ "NameFilter": "Name",
+ "CreationTime": "Erstellungszeitpunkt",
+ "IsPro": "Ist pro",
+ "ShowOnModuleList": "In Modulliste anzeigen",
+ "EfCoreConfigureMethodName": "Methodenname konfigurieren",
+ "IsProFilter": "Ist pro",
+ "ApplicationType": "Anwendungstyp",
+ "Target": "Ziel",
+ "TargetFilter": "Ziel",
+ "ModuleClass": "Modulklasse",
+ "NugetPackageTarget.DomainShared": "Gemeinsame Domain",
+ "NugetPackageTarget.Domain": "Domain",
+ "NugetPackageTarget.Application": "Anwendung",
+ "NugetPackageTarget.ApplicationContracts": "Anwedungsverträge",
+ "NugetPackageTarget.HttpApi": "HTTP-API",
+ "NugetPackageTarget.HttpApiClient": "HTTP-API-Client",
+ "NugetPackageTarget.Web": "Web",
+ "NugetPackageTarget.EntityFrameworkCore": "DeleteAllEntityFramework Core",
+ "NugetPackageTarget.MongoDB": "MongoDB",
+ "Edit": "Bearbeiten",
+ "Delete": "Löschen",
+ "Refresh": "Aktualisieren",
+ "NpmPackages": "NPM-Pakete",
+ "NugetPackages": "Nuget-Pakete",
+ "NpmPackageCount": "NPM-Paketanzahl",
+ "NugetPackageCount": "Nuget-Paketanzahl",
+ "Module": "Module",
+ "ModuleInfo": "Modulinfo",
+ "CreateANpmPackage": "Erstellen Sie ein NPM Paket",
+ "CreateAModule": "Erstellen Sie in Modul",
+ "CreateANugetPackage": "Erstellen Sei ein Nuget-Paket",
+ "AddNew": "Neu hinzufügen",
+ "PackageAlreadyExist{0}": "\"{0}\" Paket ist bereits hinzugefügt.",
+ "ModuleAlreadyExist{0}": "\"{0}\" Modul ist bereits hinzugefügt.",
+ "ClearCache": "Cache leeren",
+ "SuccessfullyCleared": "Erfolgreich geleert",
+ "Menu:NpmPackages": "NPM-Pakete",
+ "Menu:Modules": "Module",
+ "Menu:Maintenance": "Wartung",
+ "Menu:NugetPackages": "Nuget-Pakete",
+ "CreateAnOrganization": "Erstellen Sie eine Organisation",
+ "Organizations": "Organisationen",
+ "LongName": "Lange Name",
+ "LicenseType": "Lizenztyp",
+ "MissingLicenseTypeField": "Das Feld Lizenztyp ist erforderlich!",
+ "LicenseStartTime": "Startzeit der Lizenz",
+ "LicenseEndTime": "Endzeit der Lizenz",
+ "AllowedDeveloperCount": "Zulässige Entwickleranzahl",
+ "UserNameOrEmailAddress": "Benutzername oder E-Mail-Adresse",
+ "AddOwner": "Besitzer hinzufügen",
+ "UserName": "Benutzername",
+ "Email": "E-Mail",
+ "Developers": "Entwickler",
+ "AddDeveloper": "Entwickler hinzufügen",
+ "Create": "Erstellen",
+ "UserNotFound": "Benutzer nicht gefunden",
+ "{0}WillBeRemovedFromDevelopers": "{0} wird von den Entwicklern entfernt. Bestätigen Sie das?",
+ "{0}WillBeRemovedFromOwners": "{0} wird von den Besitzern entfernt. Bestätigen Sie das?",
+ "Computers": "Computer",
+ "UniqueComputerId": "Eindeutig Computer-ID",
+ "LastSeenDate": "Zuletzt gesehenes Datum",
+ "{0}Computer{1}WillBeRemovedFromRecords": "Computer von {0} ({1}) wird aus den Datensätzen entfernt",
+ "OrganizationDeletionWarningMessage": "Organisation wird gelöscht",
+ "DeletingLastOwnerWarningMessage": "Eine Organisation muss zumindest einen Besitzer aufweisen! Daher können Sie diesen Besitzer nicht entfernen",
+ "This{0}AlreadyExistInThisOrganization": "Dies {0} existiert bereits in dieser Organisation",
+ "AreYouSureYouWantToDeleteAllComputers": "Sind Sie sicher, dass Sie alle Computer löschen möchten?",
+ "DeleteAll": "Alles Löschen",
+ "DoYouWantToCreateNewUser": "Möchten Sie einen neuen Benutzer erstellen?",
+ "MasterModules": "Master-Module",
+ "OrganizationName": "Organisationsname",
+ "CreationDate": "Erstellungsdatum",
+ "LicenseStartDate": "Startdatum der Lizenz",
+ "LicenseEndDate": "Enddatum der Lizenz",
+ "OrganizationNamePlaceholder": "Organisationsname...",
+ "TotalQuestionCountPlaceholder": "Gesamtzahl der Fragen...",
+ "RemainingQuestionCountPlaceholder": "Anzahl verbleibender Fragen...",
+ "LicenseTypePlaceholder": "Lizenztyp...",
+ "CreationDatePlaceholder": "Erstellungsdatum...",
+ "LicenseStartDatePlaceholder": "Startdatum der Lizenz...",
+ "LicenseEndDatePlaceholder": "Enddatum der Lizenz...",
+ "UsernameOrEmail": "Benutzername oder E-Mail-Adresse",
+ "UsernameOrEmailPlaceholder": "Benutzername oder E-Mail-Adresse...",
+ "Member": "Mitglied",
+ "PurchaseOrderNo": "Bestellnummer",
+ "QuotationDate": "Angebotsdatum",
+ "CompanyName": "Firmenname",
+ "CompanyAddress": "Firmenanschrift",
+ "Price": "Preis",
+ "DiscountText": "Rabatttext",
+ "DiscountQuantity": "Rabattmenge",
+ "DiscountPrice": "Rabattpreis",
+ "Quotation": "Angebot",
+ "ExtraText": "Zusätzlicher Text",
+ "ExtraAmount": "Zusätzliche Menge",
+ "DownloadQuotation": "Angebot herunterladen",
+ "Invoice": "Rechnung",
+ "TaxNumber": "Steuernummer",
+ "InvoiceNumber": "Rechnungsnummer",
+ "InvoiceDate": "Rechnungsdatum",
+ "InvoiceNote": "Rechnungsnotiz",
+ "Quantity": "Menge",
+ "AddProduct": "Produkt hinzufügen",
+ "AddProductWarning": "Sie müssen ein Produkt hinzufügen!",
+ "TotalPrice": "Gesamtpreis",
+ "Generate": "Generieren",
+ "MissingQuantityField": "Das Feld Menge ist erforderlich!",
+ "MissingPriceField": "Das Feld Preis ist erforderlich!",
+ "CodeUsageStatus": "Status",
+ "Country": "Land",
+ "DeveloperCount": "Entwickleranzahl",
+ "RequestCode": "Anfrage-Code",
+ "WebSite": "Webseite",
+ "GithubUsername": "Github Benutzername",
+ "PhoneNumber": "Telefonnummer",
+ "ProjectDescription": "Projektbeschreibung",
+ "Referrer": "Referrer",
+ "DiscountRequests": "Rabattanfrage",
+ "Copylink": "Link kopieren",
+ "Disable": "Deaktivieren",
+ "Enable": "Aktivieren",
+ "EnableSendEmail": "E-Mail-Senden aktivieren",
+ "SendEmail": "E-Mail senden",
+ "SuccessfullyDisabled": "Erfolgreich deaktiviert",
+ "SuccessfullyEnabled": "Erfolgreich aktiviert",
+ "EmailSent": "E-Mail gesendet",
+ "SuccessfullySent": "Erfolgreich gesendet",
+ "SuccessfullyDeleted": "Erfolgreich gelöscht",
+ "DiscountRequestDeletionWarningMessage": "Rabattanfrage wird gelöscht",
+ "BusinessType": "Unternehmensart",
+ "TotalQuestionCount": "Gesamtzahl der Fragen",
+ "RemainingQuestionCount": "Anzahl verbleibender Fragen",
+ "TotalQuestionMustBeGreaterWarningMessage": "TotalQuestionCount muss größer sein als RemainingQuestionCount !",
+ "QuestionCountsMustBeGreaterThanZero": "TotalQuestionCount und RemainingQuestionCount müssen Null oder größer als Null sein !",
+ "UnlimitedQuestionCount": "Unbegrenzte Anzahl von Fragen",
+ "Notes": "Anmerkungen",
+ "Menu:Community": "Community",
+ "Menu:Articles": "Beiträge",
+ "Wait": "Warten",
+ "Approve": "Genehmigen",
+ "Reject": "Ablehnen",
+ "Details": "Details",
+ "Url": "URL",
+ "Title": "Titel",
+ "ContentSource": "Inhaltsquelle",
+ "Status": "Status",
+ "ReadArticle": "Beitrag lesen",
+ "ArticleHasBeenWaiting": "Beitrag hat gewartet",
+ "ArticleHasBeenApproved": "Beitrag wurde genehmigt",
+ "ArticleHasBeenRejected": "Beitrag wurde abgelehnt",
+ "Permission:Community": "Community",
+ "Permission:CommunityArticle": "Beitrag",
+ "Link": "Link",
+ "Enum:ContentSource:0": "Github",
+ "Enum:ContentSource:1": "Extern",
+ "Enum:Status:0": "Wartend",
+ "Enum:Status:1": "Abgelehnt",
+ "Enum:Status:2": "Genehmigt",
+ "Summary": "Zusammenfassung",
+ "AuthorName": "Autorenname",
+ "CoverImage": "Titelbild",
+ "RemoveCacheConfirmationMessage": "Sind Sie sicher, dass Sie den Cache für den Artikel \"{0}\" entfernen wollen?",
+ "SuccessfullyRemoved": "Erfolgreich geleert",
+ "RemoveCache": "Cache entfernen"
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/es.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/es.json
index dbd028baae..89ef8706e0 100644
--- a/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/es.json
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/es.json
@@ -116,7 +116,7 @@
"UsernameOrEmailPlaceholder": "Usuario o email...",
"Member": "Miembro",
"PurchaseOrderNo": "Número de orden de compra",
- "QuotationDate": "",
+ "QuotationDate": "Fecha de presupuesto",
"CompanyName": "Nombre de empresa",
"CompanyAddress": "Dirección de empresa",
"Price": "Precio",
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Base/Localization/Resources/de-DE.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Base/Localization/Resources/de-DE.json
new file mode 100644
index 0000000000..77e1b2090a
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Base/Localization/Resources/de-DE.json
@@ -0,0 +1,33 @@
+{
+ "culture": "de-DE",
+ "texts": {
+ "Volo.AbpIo.Domain:010004": "Maximale Mitgliederanzahl erreicht!",
+ "Volo.AbpIo.Domain:010005": "Miximale Besizeranzahl erreicht!",
+ "Volo.AbpIo.Domain:010006": "Dieser Benutzer ist bereits ein Besitzer in dieser Organisation!",
+ "Volo.AbpIo.Domain:010007": "Dieser Benutzer ist bereits ein Entwickler in dieser Organisation!",
+ "Volo.AbpIo.Domain:010008": "Die zulässige Entwickleranzahl darf nicht geringer sein als die aktuelle Entwickleranzahl!",
+ "Volo.AbpIo.Domain:010009": "Die zulässige Entwickleranzahl darf nicht kleiner als 0 sein!",
+ "Volo.AbpIo.Domain:010010": "Die maximale Anzahl der Mac-Adressen ist überschritten!",
+ "Volo.AbpIo.Domain:010011": "Die persönliche Lizenz kann nicht mehr als 1 Entwickler haben!",
+ "Volo.AbpIo.Domain:010012": "Die Lizenz kann nicht einen Monat nach Ablauf der Lizenz verlängert werden!",
+ "Volo.AbpIo.Domain:020001": "Dieses NPM-Paket konnte nicht gelöscht werden, da \"{NugetPackages}\" Nuget-Pakete von diesem Paket abhängig sind.",
+ "Volo.AbpIo.Domain:020002": "Dieses NPM-Paket konnte nicht gelöscht werden, da \"{Module}\" Module dieses Paket verwenden.",
+ "Volo.AbpIo.Domain:020003": "Dieses NPM-Paket konnte nicht gelöscht werden, da \"{Module}\" Module dieses Paket verwenden und \"{NugetPackages}\" Nuget-Pakete von diesem Paket abhängig sind.",
+ "Volo.AbpIo.Domain:020004": "Dieses Nuget-Paket konnte nicht gelöscht werden, da \"{Module}\" Module dieses Paket verwenden.",
+ "WantToLearn?": "Wollen Sie sich einlernen?",
+ "ReadyToGetStarted?": "Bereit anzufangen?",
+ "JoinOurCommunity": "Tritt unserer Community bei",
+ "GetStartedUpper": "LOSLEGEN",
+ "ForkMeOnGitHub": "Fork me on GitHub",
+ "Features": "Features",
+ "GetStarted": "Loslegen",
+ "Documents": "Unterlagen",
+ "Community": "Community",
+ "ContributionGuide": "Leitfaden für Mitwirkende",
+ "Blog": "Blog",
+ "Commercial": "Commercial",
+ "MyAccount": "Mein Benutzerkonto",
+ "SeeDocuments": "Siehe Unterlagen",
+ "Samples": "Beispiele"
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/de-DE.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/de-DE.json
new file mode 100644
index 0000000000..ddc493e63b
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/de-DE.json
@@ -0,0 +1,35 @@
+{
+ "culture": "de-DE",
+ "texts": {
+ "OrganizationManagement": "Organisationsverwaltung",
+ "OrganizationList": "Organisationsauflistung",
+ "Volo.AbpIo.Commercial:010003": "Sie sind der Besitzer dieser Organisation!",
+ "OrganizationNotFoundMessage": "Keine Organisation gefunden!",
+ "DeveloperCount": "Zugeordnete / Gesamte Entwickler",
+ "QuestionCount": "Verbleibende / Gesamte Fragen",
+ "Unlimited": "Unbegrenzt",
+ "Owners": "Besitzer",
+ "AddMember": "Mitglied hinzufügen",
+ "AddOwner": "Besizer hinzufügen",
+ "AddDeveloper": "Entwickler hinzufügen",
+ "UserName": "Benutzername",
+ "Name": "Name",
+ "EmailAddress": "E-Mail-Adress",
+ "Developers": "Entwickler",
+ "LicenseType": "Lizenztyp",
+ "Manage": "Verwalten",
+ "StartDate": "Startdatum",
+ "EndDate": "Enddatum",
+ "Modules": "Module",
+ "LicenseExtendMessage": "Ihr Lizenzenddatum wird auf {0} verlängert",
+ "LicenseUpgradeMessage": "Ihre Lizenz wird auf {0} aktualisiert",
+ "LicenseAddDeveloperMessage": "{0} Entwickler zu Ihrer Lizenz hinzugefügt",
+ "Volo.AbpIo.Commercial:010004": "Kann den angegebenen Benutzer nicht finden! Der Benutzer muss sich bereits registriert haben.",
+ "MyOrganizations": "Meine Organisationen",
+ "ApiKey": "API-Schlüssel",
+ "UserNameNotFound": "Es gibt keinen Benutzer mit dem Benutzernamen {0}",
+ "SuccessfullyAddedToNewsletter": "Vielen Dank, dass Sie unseren Newsletter abonniert haben!",
+ "MyProfile": "Mein Profil",
+ "EmailNotValid": "Bitte geben Sie eine gültige E-Mail-Adresse ein."
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/en.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/en.json
index 2a57c6b25d..e029499b06 100644
--- a/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/en.json
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/en.json
@@ -30,6 +30,8 @@
"UserNameNotFound": "There is no user with username {0}",
"SuccessfullyAddedToNewsletter": "Thanks you for subscribing to our newsletter!",
"MyProfile": "My profile",
- "EmailNotValid": "Please enter a valid email address."
+ "EmailNotValid": "Please enter a valid email address.",
+ "JoinOurMarketingNewsletter": "Join our marketing newsletter",
+ "WouldLikeToReceiveMarketingMaterials": "I would like to receive marketing materials like product deals & special offers."
}
}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/de-DE.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/de-DE.json
new file mode 100644
index 0000000000..0db3c9b29b
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/de-DE.json
@@ -0,0 +1,90 @@
+{
+ "culture": "de-DE",
+ "texts": {
+ "Permission:CommunityArticle": "Community-Beitrag",
+ "Permission:Edit": "Bearbeiten",
+ "Waiting": "Wartend",
+ "Approved": "Genehmigt",
+ "Rejected": "Abgelehnt",
+ "Wait": "Warten",
+ "Approve": "Genehmigen",
+ "Reject": "Ablehnen",
+ "ReadArticle": "Beitrag lesen",
+ "Status": "Status",
+ "ContentSource": "Inhaltsquelle",
+ "Details": "Details",
+ "Url": "URL",
+ "Title": "Titel",
+ "CreationTime": "Erstellungszeitpunkt",
+ "Save": "Speichern",
+ "SameUrlAlreadyExist": "Dieselbe URL existiert bereits, wenn Sie diesen Beitrag hinzufügen möchten, sollten Sie die URL ändern!",
+ "UrlIsNotValid": "Der URL ist nicht korrekt.",
+ "UrlNotFound": "URL nicht gefunden.",
+ "UrlContentNotFound": "URL-Inhalt nicht gefunden",
+ "Summary": "Zusammenfassung",
+ "MostRead": "Meist gelesen",
+ "Latest": "Neueste",
+ "ContributeAbpCommunity": "Tragen Sie zur ABP Community bei",
+ "SubmitYourArticle": "Reichen Sie Ihren Beitrag ein",
+ "ContributionGuide": "Leitfaden für Mitwirkende",
+ "BugReport": "Fehler melden",
+ "SeeAllArticles": "Alle Beiträge anzeigen",
+ "WelcomeToABPCommunity!": "Willkommen in der ABP Community!",
+ "MyProfile": "Mein Profil",
+ "MyOrganizations": "Meine Organisationen",
+ "EmailNotValid": "Bitte geben Sie eine gültige E-Mail-Adresse ein.",
+ "FeatureRequest": "Featureanfrage",
+ "CreateArticleTitleInfo": "Titel des Beitrags, der in der Beitragsliste angezeigt werden soll.",
+ "CreateArticleUrlInfo": "Original GitHub-/externe URL des Beitrags.",
+ "CreateArticleSummaryInfo": "Eine kurze Zusammenfassung des Beitrags, der in der Beitragsliste angezeigt werden soll.",
+ "CreateArticleCoverInfo": "Fügen Sie zum Erstellen eines effektiven Beitrags ein Titelbild hinzu. Laden Sie Bilder mit einem Seitenverhältnis von 16: 9 hoch, um die beste Ansicht zu erhalten.",
+ "ThisExtensionIsNotAllowed": "Diese Erweiterung ist nicht zulässig.",
+ "TheFileIsTooLarge": "Die Datei ist zu groß.",
+ "GoToTheArticle": "Gehe zum Beitrag",
+ "Contribute": "Beitragen",
+ "OverallProgress": "Gesamtfortschritt",
+ "Done": "Fertig",
+ "Open": "Offen",
+ "Closed": "Geschlossen",
+ "LatestQuestionOnThe": "Letzte Frage zum",
+ "Stackoverflow": "Stackoverflow",
+ "Votes": "Stimmen",
+ "Answer": "Antwort",
+ "Views": "Ansichten",
+ "Answered": "Beantwortet",
+ "WaitingForYourAnswer": "Warten auf Ihre Antwort",
+ "Asked": "gefragt",
+ "AllQuestions": "Alle Fragen",
+ "NextVersion": "Nächste Version",
+ "MilestoneErrorMessage": "Die aktuellen Meilensteindetails konnten von Github nicht abgerufen werden.",
+ "QuestionItemErrorMessage": "Die neuesten Fragendetails konnten von Stackoverflow nicht abgerufen werden.",
+ "Oops": "Hoppla!",
+ "CreateArticleSuccessMessage": "Der Beitrag wurde erfolgreich eingereicht. Er wird nach einer Überprüfung durch den Site-Administrator veröffentlicht.",
+ "ChooseCoverImage": "Ein Titelbild auswählen...",
+ "CoverImage": "Titelbild",
+ "ShareYourExperiencesWithTheABPFramework": "Ihre Erfahrungen mit dem ABP Framework teilen!",
+ "Optional": "Optional",
+ "UpdateUserWebSiteInfo": "Beispiel: https://johndoe.com",
+ "UpdateUserTwitterInfo": "Beispiel: johndoe",
+ "UpdateUserGithubInfo": "Beispiel: johndoe",
+ "UpdateUserLinkedinInfo": "Beispiel: https://www.linkedin.com/...",
+ "UpdateUserCompanyInfo": "Beispiel: Volosoft",
+ "UpdateUserJobTitleInfo": "Beispiel: Software Developer",
+ "UserName": "Benutzername",
+ "Company": "Firma",
+ "PersonalWebsite": "Persönliche Website",
+ "RegistrationDate": "Registrierungsdatum",
+ "Social": "Social",
+ "Biography": "Biographie",
+ "HasNoPublishedArticlesYet": "hat noch keine Beiträge veröffentlicht",
+ "Author": "Autor",
+ "LatestGithubAnnouncements": "Neueste Github-Ankündigungen",
+ "SeeAllAnnouncements": "Alle Ankündigungen anzeigen",
+ "LatestBlogPost": "Letzter Blog-Beitrag",
+ "Edit": "Bearbeiten",
+ "ProfileImageChange": "Ändern Sie das Profilbild",
+ "BlogItemErrorMessage": "Die neuesten Blogpost-Details konnten nicht abgerufen werden.",
+ "PlannedReleaseDate": "Geplantes Erscheinungsdatum",
+ "CommunityArticleRequestErrorMessage": "Die Anfrage nach den neuesten Beiträgen von Github konnte nicht abgerufen werden."
+ }
+}
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json
index 87db8c931a..c570e6fb1e 100644
--- a/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json
@@ -85,6 +85,20 @@
"ProfileImageChange": "Change the profile image",
"BlogItemErrorMessage": "Could not get the latest blog post details from ABP.",
"PlannedReleaseDate": "Planned release date",
- "CommunityArticleRequestErrorMessage": "Could not get the latest article request from Github."
+ "CommunityArticleRequestErrorMessage": "Could not get the latest article request from Github.",
+ "ArticleRequestFromGithubIssue": "There are not any article requests now.",
+ "LatestArticles": "Latest Articles",
+ "ArticleRequests": "Article Requests",
+ "AllArticleRequests": "See All Article Requests",
+ "SubscribeToTheNewsletter": "Subscribe to the Newsletter",
+ "NewsletterEmailDefinition": "Get information about happenings in ABP like new releases, free sources, articles, and more.",
+ "NoThanks": "No, thanks",
+ "MaybeLater": "Maybe later",
+ "JoinOurArticleNewsletter": "Join our article newsletter",
+ "Community": "Community",
+ "Marketing": "Marketing",
+ "CommunityPrivacyPolicyConfirmation": "I agree to the Terms & Conditions and Privacy Policy.",
+ "ArticleRequestMessageTitle": "Open an issue on the GitHub to request an article/tutorial you want to see on this web site.",
+ "ArticleRequestMessageBody": "Here, the list of the requested articles by the community. Do you want to write a requested article? Please click to the request and join to the discussion."
}
}
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/de-DE.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/de-DE.json
new file mode 100644
index 0000000000..6f9f0726cf
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/de-DE.json
@@ -0,0 +1,189 @@
+{
+ "culture": "de-DE",
+ "texts": {
+ "GetStarted": "Erste Schritte - Startvorlagen",
+ "Create": "Erstellen",
+ "NewProject": "Neues Projekt",
+ "DirectDownload": "Direkter Download",
+ "ProjectName": "Proejktname",
+ "ProjectType": "Projekttyp",
+ "DatabaseProvider": "Datenbankanbieter",
+ "NTier": "N-Tier",
+ "IncludeUserInterface": "Benutzeroberfläche einschließen",
+ "CreateNow": "Jetzt erstellen",
+ "TheStartupProject": "Das Startprojekt",
+ "Tutorial": "Lernprogramm",
+ "UsingCLI": "mit CLI",
+ "SeeDetails": "Details ansehen",
+ "AbpShortDescription": "ABP Framework ist eine vollständige Infrastruktur zum Erstellen moderner Webanwendungen unter Befolgung von Best Practices und Konventionen für Softwareentwicklung.",
+ "SourceCodeUpper": "QUELLCODE",
+ "LatestReleaseLogs": "Akteulle Release",
+ "Infrastructure": "Infrastrutkur",
+ "Architecture": "Architektur",
+ "Modular": "Modular",
+ "DontRepeatYourself": "Don't Repeat Yourself",
+ "DeveloperFocused": "Entwickler-Zentriert",
+ "FullStackApplicationInfrastructure": "Full-Stack-Anwendungsinfrastruktur.",
+ "DomainDrivenDesign": "Domain Driven Design",
+ "DomainDrivenDesignExplanation": "Entworfen und entwickelt basierend auf DDD-Mustern und -Prinzipien. Bietet ein Schichtenmodell für Ihre Anwendung.",
+ "Authorization": "Authorization",
+ "AuthorizationExplanation": "Erweiterte Autorisierung mit Benutzer-, Rollen- und fein abgestimmtem Berechtigungssystem. Aufbauend auf der Microsoft Identity-Bibliothek.",
+ "MultiTenancy": "Multi-Tenancy",
+ "MultiTenancyExplanationShort": "SaaS-Anwendungen leicht gemacht! Integrierte Mandantenfähigkeit von der Datenbank bis zur Benutzeroberfläche.",
+ "CrossCuttingConcerns": "Cross Cutting Concerns",
+ "CrossCuttingConcernsExplanationShort": "Komplette Infrastruktur für Autorisierung, Validierung, Ausnahmebehandlung, Caching, Überwachungsprotokollierung, Transaktionsverwaltung und mehr.",
+ "BuiltInBundlingMinification": "Built-In Bundling & Minification",
+ "BuiltInBundlingMinificationExplanation": "Für die Bundling und Minification müssen keine externen Tools verwendet werden. ABP bietet eine einfachere, dynamische, leistungsstarke, modulare und integrierte Methode!",
+ "VirtualFileSystem": "Virtual File System",
+ "VirtualFileSystemExplanation": "Betten Sie Ansichten, Skripte, Stile, Bilder ... in Pakete/Bibliotheken ein und verwenden Sie sie in verschiedenen Anwendungen wieder.",
+ "Theming": "Theming",
+ "ThemingExplanationShort": "Verwenden und passen Sie das Bootstrap-basierte Standard-UI-Design an oder erstellen Sie Ihr eigenes.",
+ "BootstrapTagHelpersDynamicForms": "Bootstrap Tag Helpers & Dynamic Forms",
+ "BootstrapTagHelpersDynamicFormsExplanation": "Anstatt die sich wiederholenden Details von Bootstrap-Komponenten manuell zu schreiben, verwenden Sie die Tag-Helper von ABP, um diese zu vereinfachen und dabei die Vorteile von Intellisense zu nutzen. Erstellen Sie mit dem dynamischen Formular-Tag-Helfer schnell UI-Formulare basierend auf einem C#-Modell.",
+ "HTTPAPIsDynamicProxies": "HTTP APIs & Dynamic Proxies",
+ "HTTPAPIsDynamicProxiesExplanation": "Stellen Sie Anwendungsdienste automatisch als HTTP-APIs im REST-Stil bereit und verwenden Sie diese mit dynamischen JavaScript- und C#-Proxys.",
+ "CompleteArchitectureInfo": "Moderne Architektur zur Erstellung wartbarer Softwarelösungen.",
+ "DomainDrivenDesignBasedLayeringModelExplanation": "Hilft Ihnen bei der Implementierung einer DDD-basierten Schichtarchitektur und beim Aufbau einer wartbaren Codebasis.",
+ "DomainDrivenDesignBasedLayeringModelExplanationCont": "Bietet Startvorlagen, Abstraktionen, Basisklassen, Dienste, Dokumentation und Anleitungen, mit denen Sie Ihre Anwendung basierend auf DDD-Mustern und -Prinzipien entwickeln können.",
+ "MicroserviceCompatibleModelExplanation": "Das Kernframework und die vorgefertigten Module sind unter Berücksichtigung der Microservice-Architektur konzipiert.",
+ "MicroserviceCompatibleModelExplanationCont": "Bietet Infrastruktur, Integrationen, Beispiele und Dokumentation zur einfacheren Implementierung von Microservice-Lösungen, ohne zusätzliche Komplexität zu verursachen, wenn Sie eine monolithische Anwendung wünschen.",
+ "ModularInfo": "ABP bietet ein Modulsystem, mit dem Sie wiederverwendbare Anwendungsmodule entwickeln, Ereignisse im Anwendungslebenszyklus verknüpfen und Abhängigkeiten zwischen Kernteilen Ihres Systems ausdrücken können.",
+ "PreBuiltModulesThemes": "Vorgefertigte Module & Themes",
+ "PreBuiltModulesThemesExplanation": "Open Source- und kommerzielle Module und Themes stehen bereit, um in Ihrer Geschäftsanwendung verwendet zu werden.",
+ "NuGetNPMPackages": "NuGet- & NPM-Pakete",
+ "NuGetNPMPackagesExplanation": "Bereitgestellt als NuGet- & NPM-Pakete. Einfach zu installieren und zu aktualisieren.",
+ "ExtensibleReplaceable": "Erweiterbar/Austauschbar",
+ "ExtensibleReplaceableExplanation": "Alle Dienste und Module sind auf Erweiterbarkeit ausgelegt. Sie können Dienste, Seiten, Stile und Komponenten ersetzen.",
+ "CrossCuttingConcernsExplanation2": "Halten Sie Ihre Codebasis kleiner, damit Sie sich auf ihre geschäftsspezifischen Code konzentrieren können.",
+ "CrossCuttingConcernsExplanation3": "Verbringen Sie keine Zeit damit, grundlegende Anwendungsanforderungen für jedes neue Projekte zu implementieren.",
+ "AuthenticationAuthorization": "Authentifizierung & Autorisierung",
+ "ExceptionHandling": "Fehlerbehandlung",
+ "Validation": "Validierung",
+ "DatabaseConnection": "Datenbankverbindung",
+ "TransactionManagement": "Transaktionsmanagement",
+ "AuditLogging": "Audit Logging",
+ "Caching": "Caching",
+ "Multitenancy": "Multimandantenfähigkeit",
+ "DataFiltering": "Datenfilterung",
+ "ConventionOverConfiguration": "Convention Over Configuration",
+ "ConventionOverConfigurationExplanation": "ABP implementiert standardmäßig allgemeine Anwendungskonventionen mit einer minimalen oder Null-Konfiguration.",
+ "ConventionOverConfigurationExplanationList1": "Automatische Registrierung bekannter Services für Dependency Injection.",
+ "ConventionOverConfigurationExplanationList2": "Stellt Anwendungsdienste mittels Namenskonventionen als HTTP-APIs bereit.",
+ "ConventionOverConfigurationExplanationList3": "Erstellt dynamische HTTP-Client-Proxys für C# und JavaScript.",
+ "ConventionOverConfigurationExplanationList4": "Bietet Standard-Repositorys für Ihre Entities.",
+ "ConventionOverConfigurationExplanationList5": "Verwaltet die Unit-of-Work gemäß Webanforderung oder Anwendungsdienstmethode.",
+ "ConventionOverConfigurationExplanationList6": "Triggert Erstellungs-, Aktualisierungs- und Lösch-Events für Ihre Entities.",
+ "BaseClasses": "Basisklassen",
+ "BaseClassesExplanation": "Vorgefertigte Basisklassen für gängige Anwendungsmuster.",
+ "DeveloperFocusedExplanation": "ABP ist für Entwickler",
+ "DeveloperFocusedExplanationCont": "Es zielt darauf ab, Ihre tägliche Softwareentwicklung zu vereinfachen, ohne Sie daran zu hindern, Low-Level-Code zu schreiben.",
+ "SeeAllFeatures": "Alle Features anzeigen",
+ "CLI_CommandLineInterface": "CLI (Command Line Interface)",
+ "CLI_CommandLineInterfaceExplanation": "Enthält eine CLI, mit der Sie die Erstellung neuer Projekte und das Hinzufügen neuer Module automatisieren können.",
+ "StartupTemplates": "Startvorlagen",
+ "StartupTemplatesExplanation": "Verschiedene Startvorlagen bieten eine vollständig konfigurierte Lösung, um Ihre Entwicklung zu beschleunigen.",
+ "BasedOnFamiliarTools": "Basierend auf vertrauten Tools",
+ "BasedOnFamiliarToolsExplanation": "Aufbauend auf und integriert mit beliebten Tools, die Sie bereits kennen. Geringe Lernkurve, einfache Anpassung, komfortable Entwicklung.",
+ "ORMIndependent": "ORM-unabhängig",
+ "ORMIndependentExplanation": "Das Kernframework ist ORM-/datenbankunabhängig und kann mit jeder Datenquelle arbeiten. Entity Framework Core- und MongoDB-Anbieter sind bereits verfügbar.",
+ "Features": "Entdecken Sie die ABP Framework-Features",
+ "ABPCLI": "ABP CLI",
+ "Modularity": "Modularität",
+ "BootstrapTagHelpers": "Bootstrap Tag Helpers",
+ "DynamicForms": "Dynamische Formulare",
+ "BundlingMinification": "Bundling & Minification",
+ "BackgroundJobs": "Background Jobs",
+ "BackgroundJobsExplanation": "Definieren Sie einfache Klassen, um Jobs im Hintergrund in der Warteschlange auszuführen. Verwenden Sie den integrierten Jobmanager oder integrieren Sie Ihren eigenen. Hangfire & RabbitMQ -Integrationen sind bereits verfügbar.",
+ "DDDInfrastructure": "DDD-Infrastruktur",
+ "DomainDrivenDesignInfrastructure": "Domain Driven Design-Infrastruktur",
+ "AutoRESTAPIs": "Auto REST APIs",
+ "DynamicClientProxies": "Dynamische Client-Proxies",
+ "DistributedEventBus": "Distributed Event Bus",
+ "DistributedEventBusWithRabbitMQIntegration": "Distributed Event Bus mit RabbitMQ-Integration",
+ "TestInfrastructure": "Test-Infrastruktur",
+ "AuditLoggingEntityHistories": "Audit Logging & Entity Histories",
+ "ObjectToObjectMapping": "Object to Object Mapping",
+ "ObjectToObjectMappingExplanation": " Object to Object Mapping Abstraktion mit AutoMapper-Integration.",
+ "EmailSMSAbstractions": "E-Mail & SMS Abstraktionen",
+ "EmailSMSAbstractionsWithTemplatingSupport": "E-Mail- und SMS-Abstraktionen mit Vorlagenunterstützung",
+ "Localization": "Lokalisierung",
+ "SettingManagement": "Einstellungsverwaltung",
+ "ExtensionMethods": "Erweiterungsmethoden",
+ "ExtensionMethodsHelpers": "Erweiterungsmethoden & Helfer",
+ "AspectOrientedProgramming": "Aspektorientierte Programmierung",
+ "DependencyInjection": "Dependency Injection",
+ "DependencyInjectionByConventions": "Dependency Injection durch Konventionen",
+ "ABPCLIExplanation": "ABP CLI (Command Line Interface) ist ein Befehlszeilenprogramm zum Ausführen einiger gängiger Vorgänge für ABP-basierte Lösungen.",
+ "ModularityExplanation": "ABP bietet eine vollständige Infrastruktur zum Erstellen eigener Anwendungsmodule, die Entities, Services, Datenbankintegration, APIs, UI-Komponenten usw. enthalten können.",
+ "MultiTenancyExplanation": "Das ABP-Framework unterstützt nicht nur die Entwicklung von Multi-Mandantenanwendungen, sondern macht Ihren Code von der Mandantenfähigkeit auch weitgehend unabhängig.",
+ "MultiTenancyExplanation2": "Kann den aktuellen Mandanten automatisch ermitteln und Daten verschiedener Mandanten voneinander isolieren.",
+ "MultiTenancyExplanation3": "Unterstützt einzelne Datenbank-, Datenbank-pro-Mandanten- und Hybrid-Ansätze.",
+ "MultiTenancyExplanation4": "Sie konzentrieren sich auf Ihren geschäftsspezifischen Code und lassen das Framework die Mandantenfähigkeit für Sie übernehmen.",
+ "BootstrapTagHelpersExplanation": "Anstatt die sich wiederholenden Details von Bootstrap-Komponenten manuell zu schreiben, verwenden Sie die Tag-Helper von ABP, um diese zu vereinfachen und dabei die Vorteile von Intellisense zu nutzen. Sie können Bootstrap weitherhin verwenden, wann immer Sie es benötigen.",
+ "DynamicFormsExplanation": "Helfer für dynamische Formular- und Input-Tags können das vollständige Formular anhand einer C#-Klasse als Model erstellen.",
+ "AuthenticationAuthorizationExplanation": "Umfangreiche Authentifizierungs- und Autorisierungsoptionen, die in ASP.NET Core Identity & IdentityServer4 integriert sind. Bietet ein erweiterbares und detailliertes Berechtigungssystem.",
+ "CrossCuttingConcernsExplanation": "Wiederholen Sie sich nicht, um all diese allgemeinen Dinge immer wieder zu implementieren. Konzentrieren Sie sich auf Ihren geschäftsspezifischen Code und lassen Sie ihn von ABP durch Konventionen automatisieren.",
+ "DatabaseConnectionTransactionManagement": "Datenbankverbindungs- und Transaktionsmanagement",
+ "CorrelationIdTracking": "Correlation-ID-Verfolgung",
+ "BundlingMinificationExplanation": "ABP bietet ein einfaches, dynamisches, leistungsstarkes, modulares und integriertes Bundling- und Minification-System.",
+ "VirtualFileSystemnExplanation": "Das virtuelle Dateisystem ermöglicht die Verwaltung von Dateien, die physisch nicht auf dem Dateisystem (Datenträger) vorhanden sind. Es wird hauptsächlich verwendet, um Dateien (js, css, image, cshtml ...) in Assemblys einzubetten und sie zur Laufzeit wie physische Dateien zu verwenden.",
+ "ThemingExplanation": "Mit dem Theming-System können Sie Ihre Anwendung & Module Theme-unabhängig entwickeln, indem Sie eine Reihe gemeinsamer Basisbibliotheken und Layouts definieren, die auf dem neuesten Bootstrap-Framework basieren.",
+ "DomainDrivenDesignInfrastructureExplanation": "Eine vollständige Infrastruktur zum Erstellen von mehrschichtigen Anwendungen basierend auf den Domain Driven Design Entwurfsmustern und -prinzipien;",
+ "Specification": "Specification",
+ "Repository": "Repository",
+ "DomainService": "Domain Service",
+ "ValueObject": "Value Object",
+ "ApplicationService": "Application Service",
+ "DataTransferObject": "Data Transfer Object",
+ "AggregateRootEntity": "Aggregate Root, Entity",
+ "AutoRESTAPIsExplanation": "ABP kann Ihre Anwendungsservices gemäß Konvention automatisch als API-Controller konfigurieren.",
+ "DynamicClientProxiesExplanation": "Verwenden Sie Ihre APIs ganz einfach in JavaScript- und C#-Clients.",
+ "DistributedEventBusWithRabbitMQIntegrationExplanation": "Veröffentlichen und konsumieren Sie Distributed Events einfach mithilfe des integrierten Distributed Event Bus mit verfügbarer RabbitMQ-Integration.",
+ "TestInfrastructureExplanation": "Das Framework wurde unter Berücksichtigung von Unit- und Integrationstests entwickelt. Bietet Ihnen Basisklassen, um es einfacher zu machen. Startvorlagen werden mit vorkonfiguriert Tests geliefert.",
+ "AuditLoggingEntityHistoriesExplanation": "Integriertes Audit Logging für geschäftskritische Anwendungen. Audit Logging auf Request-, Service-, und Methodenebene sowie Entity-Historien mit Details auf Property-Ebene.",
+ "EmailSMSAbstractionsWithTemplatingSupportExplanation": "IEmailSender- und ISmsSender-Abstraktionen entkoppeln Ihre Anwendungslogik von der Infrastruktur. Das erweiterte E-Mail-Vorlagensystem ermöglicht das Erstellen und Lokalisieren von E-Mail-Vorlagen und deren einfache Verwendung bei Bedarf.",
+ "LocalizationExplanation": "Das Lokalisierungssystem ermöglicht das Erstellen von Ressourcen in einfachen JSON-Dateien und die Lokalisierung Ihrer Benutzeroberfläche. Es unterstützt erweiterte Szenarien wie Vererbung, Erweiterungen und JavaScript-Integration und ist vollständig mit dem Lokalisierungssystem von AspNet Core kompatibel.",
+ "SettingManagementExplanation": "Definieren Sie Einstellungen für Ihre Anwendung und erhalten Sie zur Laufzeit Werte basierend auf der aktuellen Konfiguration, dem Mandanten und dem Benutzer.",
+ "ExtensionMethodsHelpersExplanation": "Wiederholen Sie sich nicht einmal für triviale Codeteile. Erweiterungen und Helfer für Standardtypen machen Ihren Code viel sauberer und einfacher zu schreiben.",
+ "AspectOrientedProgrammingExplanation": "Bietet eine komfortable Infrastruktur zum Erstellen dynamischer Proxys und zum Implementieren der aspektorientierten Programmierung. Fangen Sie eine Klasse ab und führen Sie Ihren Code vor und nach jeder Methodenausführung aus.",
+ "DependencyInjectionByConventionsExplanation": "Sie müssen Ihre Klassen nicht manuell für die Dependency Injection registrieren. Registriert gängige Servicetypen automatisch gemäß Konvention. Für andere Arten von Services können Sie Schnittstellen und Attribute verwenden, um dies einfacher gestalten und an Ort und Stelle zu ermöglichen.",
+ "DataFilteringExplanation": "Definieren und verwenden Sie Datenfilter, die automatisch angewendet werden, wenn Sie Entities aus der Datenbank abfragen. Soft Delete- und Multimandanten-Filter sind sofort verfügbar, wenn Sie einfache Schnittstellen implementieren.",
+ "PublishEvents": "Events veröffentlichen",
+ "HandleEvents": "Auf Events reagieren",
+ "AndMore": "und mehr...",
+ "Code": "Code",
+ "Result": "Resultat",
+ "SeeTheDocumentForMoreInformation": "Weitere Informationen finden Sie in der {0} -Dokumentation ",
+ "IndexPageHeroSection": "Open SourceWebanwendung Framework für ASP.Net Core",
+ "UiFramework": "UI-Framework",
+ "EmailAddress": "E-Mail-Adresse",
+ "Mobile": "Mobile",
+ "ReactNative": "React Native",
+ "Strong": "Stark",
+ "Complete": "Vollständig",
+ "BasedLayeringModel": "Based Layering Model",
+ "Microservice": "Microservice",
+ "Compatible": "Kompatibel",
+ "MeeTTheABPCommunityInfo": "Unsere Mission ist es, eine Umgebung zu schaffen, in der Entwickler sich gegenseitig mit Beiträgen, Tutorials, Fallstudien usw. helfen und Gleichgesinnte treffen können.",
+ "JoinTheABPCommunityInfo": "Beteiligen Sie sich an einer lebendigen Community und tragen Sie zum ABP Framework bei!",
+ "AllArticles": "Alle Beiträge",
+ "SubmitYourArticle": "Reichen Sie Ihren Beitrag ein",
+ "DynamicClientProxyDocument": "In der Dokumentation zu den Dynamischen Client-Proxies finden Sie Informationen zu JavaScript & C#.",
+ "EmailSMSAbstractionsDocument": "Weitere Informationen finden Sie in den Unterlagen E-Mail-Senden and SMS-Senden.",
+ "CreateProjectWizard": "Dieser Assistent erstellt ein neues Projekt aus der Startvorlage, die ordnungsgemäß konfiguriert ist, um Ihr Projekt zu starten.",
+ "TieredOption": "Erstellt eine Tiered Lösung, bei der Web- und HTTP-API-Ebenen physisch getrennt sind. Wenn diese Option nicht aktiviert ist, wird eine mehrschichtige Lösung erstellt, die weniger komplex und für die meisten Szenarien geeignet ist.",
+ "SeparateIdentityServerOption": "Trennt die Serverseite in zwei Anwendungen: Die erste ist für den Identitätsserver und die zweite für die serverseitige HTTP-API.",
+ "UseslatestPreVersion": "Verwendet die neueste Vorabversion",
+ "ReadTheDocumentation": "Lesen SieDie Dokumentation",
+ "Documentation": "Dokumentation",
+ "GettingStartedTutorial": "Erste Schritte Tutorial",
+ "ApplicationDevelopmentTutorial": "Tutorial zur Anwendungsentwicklung",
+ "TheStartupTemplate": "Die Startvorlage",
+ "InstallABPCLIInfo": "ABP CLI ist der schnellste Weg, um eine neue Lösung mit dem ABP-Framework zu starten. Installieren Sie die ABP-CLI über die Eingabeaufforderung:",
+ "DifferentLevelOfNamespaces": "Sie können verschiedene Ebenen von Namespaces verwenden; z.B. BookStore, Acme.BookStore or Acme.Retail.BookStore.",
+ "ABPCLIExamplesInfo": "Der Befehl new erstellt eine mehrschichtige MVC-Anwendung mit Entity Framework Core als Datenbankanbieter. Es gibt jedoch zusätzliche Optionen. Beispiele:",
+ "SeeCliDocumentForMoreInformation": "Weitere Optionen finden Sie im ABP CLI-Dokument oder wählen Sie oben die Registerkarte \"Direkter Download\".",
+ "Optional": "Optional",
+ "LocalFrameworkRef": "Behalten Sie die lokale Projektreferenz für die Framework-Pakete bei."
+ }
+}
\ No newline at end of file
diff --git a/common.DotSettings b/common.DotSettings
index 6f40d029a7..7edbd05e75 100644
--- a/common.DotSettings
+++ b/common.DotSettings
@@ -20,6 +20,17 @@
FalseFalseSQL
+ False
+ Never
+ Never
+ False
+ Never
+ Never
+ Never
+ Never
+ Never
+ True
+ TrueFalseFalseFalse
diff --git a/common.props b/common.props
index 9f6b677e72..1528abbcc6 100644
--- a/common.props
+++ b/common.props
@@ -1,7 +1,7 @@
latest
- 4.0.0
+ 4.1.0$(NoWarn);CS1591;CS0436https://abp.io/assets/abp_nupkg.pnghttps://abp.io/
diff --git a/docs/en/API/Dynamic-CSharp-API-Clients.md b/docs/en/API/Dynamic-CSharp-API-Clients.md
index 23ec336acf..d92846f5e8 100644
--- a/docs/en/API/Dynamic-CSharp-API-Clients.md
+++ b/docs/en/API/Dynamic-CSharp-API-Clients.md
@@ -1,4 +1,4 @@
-# Dynamic C# API Clients
+# Dynamic C# API Client Proxies
ABP can dynamically create C# API client proxies to call remote HTTP services (REST APIs). In this way, you don't need to deal with `HttpClient` and other low level HTTP features to call remote services and get results.
diff --git a/docs/en/Application-Services.md b/docs/en/Application-Services.md
index f48ff9cb5b..7fe9825f2a 100644
--- a/docs/en/Application-Services.md
+++ b/docs/en/Application-Services.md
@@ -342,12 +342,12 @@ public class DistrictAppService
{
}
- protected override async Task DeleteByIdAsync(DistrictKey id)
+ protected async override Task DeleteByIdAsync(DistrictKey id)
{
await Repository.DeleteAsync(d => d.CityId == id.CityId && d.Name == id.Name);
}
- protected override async Task GetEntityByIdAsync(DistrictKey id)
+ protected async override Task GetEntityByIdAsync(DistrictKey id)
{
return await AsyncQueryableExecuter.FirstOrDefaultAsync(
Repository.Where(d => d.CityId == id.CityId && d.Name == id.Name)
@@ -400,7 +400,7 @@ public class MyPeopleAppService : CrudAppService
{
}
- protected override async Task CheckDeletePolicyAsync()
+ protected async override Task CheckDeletePolicyAsync()
{
await AuthorizationService.CheckAsync("...");
}
diff --git a/docs/en/Authorization.md b/docs/en/Authorization.md
index eef4237039..158466d153 100644
--- a/docs/en/Authorization.md
+++ b/docs/en/Authorization.md
@@ -234,7 +234,7 @@ context
When you write this code inside your permission definition provider, it finds the "role deletion" permission of the [Identity Module](Modules/Identity.md) and disabled the permission, so no one can delete a role on the application.
-> Tip: It is better to check the value returned by the `GetPermissionOrNull` method since it may return null if the given permission was not defined.
+> Tip: It is better to check the value returned by the `GetPermissionOrNull` method since it may return null if the given permission was not defined.
## IAuthorizationService
@@ -354,7 +354,7 @@ public class SystemAdminPermissionValueProvider : PermissionValueProvider
public override string Name => "SystemAdmin";
- public override async Task
+ public async override Task
CheckAsync(PermissionValueCheckContext context)
{
if (context.Principal?.FindFirst("User_Type")?.Value == "SystemAdmin")
diff --git a/docs/en/Background-Jobs-RabbitMq.md b/docs/en/Background-Jobs-RabbitMq.md
index a984a2a3ab..2ee478ed17 100644
--- a/docs/en/Background-Jobs-RabbitMq.md
+++ b/docs/en/Background-Jobs-RabbitMq.md
@@ -1,3 +1,133 @@
# RabbitMQ Background Job Manager
-TODO
\ No newline at end of file
+RabbitMQ is an industry standard message broker. While it is typically used for inter-process communication (messaging / distributed events), it is pretty useful to store and execute background jobs in FIFO (First In First Out) order.
+
+ABP Framework provides the [Volo.Abp.BackgroundJobs.RabbitMQ](https://www.nuget.org/packages/Volo.Abp.BackgroundJobs.RabbitMQ) NuGet package to use the RabbitMQ for background job execution.
+
+> See the [background jobs document](Background-Jobs.md) to learn how to use the background job system. This document only shows how to install and configure the RabbitMQ integration.
+
+## Installation
+
+Use the ABP CLI to add [Volo.Abp.BackgroundJobs.RabbitMQ](https://www.nuget.org/packages/Volo.Abp.BackgroundJobs.RabbitMQ) NuGet package to your project:
+
+* Install the [ABP CLI](https://docs.abp.io/en/abp/latest/CLI) if you haven't installed before.
+* Open a command line (terminal) in the directory of the `.csproj` file you want to add the `Volo.Abp.BackgroundJobs.RabbitMQ` package.
+* Run `abp add-package Volo.Abp.BackgroundJobs.RabbitMQ` command.
+
+If you want to do it manually, install the [Volo.Abp.BackgroundJobs.RabbitMQ](https://www.nuget.org/packages/Volo.Abp.BackgroundJobs.RabbitMQ) NuGet package to your project and add `[DependsOn(typeof(AbpBackgroundJobsRabbitMqModule))]` to the [ABP module](Module-Development-Basics.md) class inside your project.
+
+## Configuration
+
+### Default Configuration
+
+The default configuration automatically connects to the local RabbitMQ server (localhost) with the standard port. **In this case, no configuration needed.**
+
+### RabbitMQ Connection(s)
+
+You can configure the RabbitMQ connections using the standard [configuration system](Configuration.md), like using the `appsettings.json` file, or using the [options](Options.md) classes.
+
+#### `appsettings.json` file configuration
+
+This is the simplest way to configure the RabbitMQ connections. It is also very strong since you can use any other configuration source (like environment variables) that is [supported by the AspNet Core](https://docs.microsoft.com/en-us/aspnet/core/fundamentals/configuration/).
+
+**Example: Configuring the Default RabbitMQ Connection**
+
+````json
+{
+ "RabbitMQ": {
+ "Connections": {
+ "Default": {
+ "HostName": "123.123.123.123",
+ "Port": "5672"
+ }
+ }
+ }
+}
+````
+
+You can use any of the [ConnectionFactry](http://rabbitmq.github.io/rabbitmq-dotnet-client/api/RabbitMQ.Client.ConnectionFactory.html#properties) properties as the connection properties. See [the RabbitMQ document](https://www.rabbitmq.com/dotnet-api-guide.html#exchanges-and-queues) to understand these options better.
+
+Defining multiple connections is allowed. In this case, you can use different connections for different background job types (see the `AbpRabbitMqBackgroundJobOptions` section below).
+
+**Example: Declare two connections**
+
+````json
+{
+ "RabbitMQ": {
+ "Connections": {
+ "Default": {
+ "HostName": "123.123.123.123"
+ },
+ "SecondConnection": {
+ "HostName": "321.321.321.321"
+ }
+ }
+ }
+}
+````
+
+#### AbpRabbitMqOptions
+
+`AbpRabbitMqOptions` class can be used to configure the connection strings for the RabbitMQ. You can configure this options inside the `ConfigureServices` of your [module](Module-Development-Basics.md).
+
+**Example: Configure the connection**
+
+````csharp
+Configure(options =>
+{
+ options.Connections.Default.UserName = "user";
+ options.Connections.Default.Password = "pass";
+ options.Connections.Default.HostName = "123.123.123.123";
+ options.Connections.Default.Port = 5672;
+});
+````
+
+Using these options classes can be combined with the `appsettings.json` way. Configuring an option property in the code overrides the value in the configuration file.
+
+### AbpRabbitMqBackgroundJobOptions
+
+#### Job Queue Names
+
+By default, each job type uses a separate queue. Queue names are calculated by combining a standard prefix and the job name. Default prefix is `AbpBackgroundJobs.` So, if the job name is `EmailSending` then the queue name in the RabbitMQ becomes `AbpBackgroundJobs.EmailSending`
+
+> Use `BackgroundJobName` attribute on the background **job argument** class to specify the job name. Otherwise, the job name will be the full name (with namespace) of the job class.
+
+#### Job Connections
+
+By default, all the job types use the `Default` RabbitMQ connection.
+
+#### Customization
+
+`AbpRabbitMqBackgroundJobOptions` can be used to customize the queue names and the connections used by the jobs.
+
+**Example:**
+
+````csharp
+Configure(options =>
+{
+ options.DefaultQueueNamePrefix = "my_app_jobs.";
+ options.JobQueues[typeof(EmailSendingArgs)] =
+ new JobQueueConfiguration(
+ typeof(EmailSendingArgs),
+ queueName: "my_app_jobs.emails",
+ connectionName: "SecondConnection"
+ );
+});
+````
+
+* This example sets the default queue name prefix to `my_app_jobs.`. If different applications use the same RabbitMQ server, it would be important to use different prefixes for each application to not consume jobs of each other.
+* Also specifies a different connection string for the `EmailSendingArgs`.
+
+`JobQueueConfiguration` class has some additional options in its constructor;
+
+* `queueName`: The queue name that is used for this job. The prefix is not added, so you need to specify the full name of the queue.
+* `connectionName`: The RabbitMQ connection name (see the connection configuration above). This is optional and the default value is `Default`.
+* `durable` (optional, default: `true`).
+* `exclusive` (optional, default: `false`).
+* `autoDelete` (optional, default: `false`)
+
+See the RabbitMQ documentation if you want to understand the `durable`, `exclusive` and `autoDelete` options better, while most of the times the default configuration is what you want.
+
+## See Also
+
+* [Background Jobs](Background-Jobs.md)
\ No newline at end of file
diff --git a/docs/en/Background-Jobs.md b/docs/en/Background-Jobs.md
index e5ef87bb2d..2e0b5d3f65 100644
--- a/docs/en/Background-Jobs.md
+++ b/docs/en/Background-Jobs.md
@@ -13,7 +13,7 @@ Background jobs are **persistent** that means they will be **re-tried** and **ex
ABP provides an **abstraction** module and **several implementations** for background jobs. It has a built-in/default implementation as well as Hangfire, RabbitMQ and Quartz integrations.
-`Volo.Abp.BackgroundJobs.Abstractions` nuget package provides needed services to create background jobs and queue background job items. If your module only depend on this package, it can be independent from the actual implementation/integration.
+`Volo.Abp.BackgroundJobs.Abstractions` NuGet package provides needed services to create background jobs and queue background job items. If your module only depend on this package, it can be independent from the actual implementation/integration.
> `Volo.Abp.BackgroundJobs.Abstractions` package is installed to the startup templates by default.
@@ -24,23 +24,29 @@ A background job is a class that implements the `IBackgroundJob` interfac
This example is used to send emails in background. First, define a class to store arguments of the background job:
````csharp
-public class EmailSendingArgs
+namespace MyProject
{
- public string EmailAddress { get; set; }
- public string Subject { get; set; }
- public string Body { get; set; }
+ public class EmailSendingArgs
+ {
+ public string EmailAddress { get; set; }
+ public string Subject { get; set; }
+ public string Body { get; set; }
+ }
}
````
Then create a background job class that uses an `EmailSendingArgs` object to send an email:
````csharp
+using System.Threading.Tasks;
using Volo.Abp.BackgroundJobs;
+using Volo.Abp.DependencyInjection;
using Volo.Abp.Emailing;
namespace MyProject
{
- public class EmailSendingJob : BackgroundJob, ITransientDependency
+ public class EmailSendingJob
+ : AsyncBackgroundJob, ITransientDependency
{
private readonly IEmailSender _emailSender;
@@ -49,9 +55,9 @@ namespace MyProject
_emailSender = emailSender;
}
- public override void Execute(EmailSendingArgs args)
+ public override async Task ExecuteAsync(EmailSendingArgs args)
{
- _emailSender.Send(
+ await _emailSender.SendAsync(
args.EmailAddress,
args.Subject,
args.Body
@@ -63,10 +69,35 @@ namespace MyProject
This job simply uses `IEmailSender` to send emails (see [email sending document](Emailing.md)).
+> `AsyncBackgroundJob` is used to create a job needs to perform async calls. You can inherit from `BackgroundJob` and override the `Execute` method if the method doesn't need to perform any async call.
+
#### Exception Handling
A background job should not hide exceptions. If it throws an exception, the background job is automatically re-tried after a calculated waiting time. Hide exceptions only if you don't want to re-run the background job for the current argument.
+#### Job Name
+
+Each background job has a name. Job names are used in several places. For example, RabbitMQ provider uses job names to determine the RabbitMQ Queue names.
+
+Job name is determined by the **job argument type**. For the `EmailSendingArgs` example above, the job name is `MyProject.EmailSendingArgs` (full name, including the namespace). You can use the `BackgroundJobName` attribute to set a different job name.
+
+**Example**
+
+```csharp
+using Volo.Abp.BackgroundJobs;
+
+namespace MyProject
+{
+ [BackgroundJobName("emails")]
+ public class EmailSendingArgs
+ {
+ public string EmailAddress { get; set; }
+ public string Subject { get; set; }
+ public string Body { get; set; }
+ }
+}
+```
+
### Queue a Job Item
Now, you can queue an email sending job using the `IBackgroundJobManager` service:
diff --git a/docs/en/Background-Workers-Quartz.md b/docs/en/Background-Workers-Quartz.md
index e1d8287019..ce82222c14 100644
--- a/docs/en/Background-Workers-Quartz.md
+++ b/docs/en/Background-Workers-Quartz.md
@@ -37,7 +37,7 @@ public class YourModule : AbpModule
````
> Quartz background worker integration provided `QuartzPeriodicBackgroundWorkerAdapter` to adapt `PeriodicBackgroundWorkerBase` and `AsyncPeriodicBackgroundWorkerBase` derived class. So, you can still fllow the [background workers document](Background-Workers.md) to define the background worker.
-> `BackgroundJobWorker` checks todo jobs every 5 seconds, but quartz will not block when long time jobs are executing. So,after Added Quartz background worker integration, you also need to add [Quartz Background Jobs](Background-Jobs-Quartz.md) or [Hangfire Background Jobs](Background-Jobs-Hangfire.md) to avoid duplicate execution jobs.
+> `BackgroundJobWorker` checks jobs every 5 seconds, but quartz will not block when long time jobs are executing. So,after Added Quartz background worker integration, you also need to add [Quartz Background Jobs](Background-Jobs-Quartz.md) or [Hangfire Background Jobs](Background-Jobs-Hangfire.md) to avoid duplicate execution jobs.
## Configuration
diff --git a/docs/en/Background-Workers.md b/docs/en/Background-Workers.md
index c8a617efec..2e1b9d2a90 100644
--- a/docs/en/Background-Workers.md
+++ b/docs/en/Background-Workers.md
@@ -54,7 +54,7 @@ public class PassiveUserCheckerWorker : AsyncPeriodicBackgroundWorkerBase
Timer.Period = 600000; //10 minutes
}
- protected override async Task DoWorkAsync(
+ protected async override Task DoWorkAsync(
PeriodicBackgroundWorkerContext workerContext)
{
Logger.LogInformation("Starting: Setting status of inactive users...");
diff --git a/docs/en/Blob-Storing.md b/docs/en/Blob-Storing.md
index f477e6178a..63281a5ee6 100644
--- a/docs/en/Blob-Storing.md
+++ b/docs/en/Blob-Storing.md
@@ -21,7 +21,7 @@ The ABP Framework has already the following storage provider implementations;
* [Azure](Blob-Storing-Azure.md): Stores BLOBs on the [Azure BLOB storage](https://azure.microsoft.com/en-us/services/storage/blobs/).
* [Aliyun](Blob-Storing-Aliyun.md): Stores BLOBs on the [Aliyun Storage Service](https://help.aliyun.com/product/31815.html).
* [Minio](Blob-Storing-Minio.md): Stores BLOBs on the [MinIO Object storage](https://min.io/).
-* [Aws](Blob-Storing-Aws.md): Stores BLOBs on the[Amazon Simple Storage Service](https://min.io/).
+* [Aws](Blob-Storing-Aws.md): Stores BLOBs on the[Amazon Simple Storage Service](https://aws.amazon.com/s3/).
More providers will be implemented by the time. You can [request](https://github.com/abpframework/abp/issues/new) it for your favorite provider or [create it yourself](Blob-Storing-Custom-Provider.md) and [contribute](Contribution/Index.md) to the ABP Framework.
diff --git a/docs/en/Blog-Posts/2020-05-08 v2_7_Release/Post.md b/docs/en/Blog-Posts/2020-05-08 v2_7_Release/Post.md
index b6c1d3f952..c09b0eb75c 100644
--- a/docs/en/Blog-Posts/2020-05-08 v2_7_Release/Post.md
+++ b/docs/en/Blog-Posts/2020-05-08 v2_7_Release/Post.md
@@ -16,7 +16,7 @@ ABP.IO Platform is rapidly growing and we are getting more and more contribution
### Object Extending System
-In the last few releases, we've mostly focused on providing ways to extend existing modules when you use them as NuGet/NPM Packages.
+In the last few releases, we've mostly focused on providing ways to extend existing modules when you use them as NuGet/NPM Packages.
The Object Extending System allows module developers to create extensible modules and allows application developers to customize and extend a module easily.
@@ -43,7 +43,7 @@ ObjectExtensionManager.Instance
options.Attributes.Add(new RequiredAttribute());
options.Attributes.Add(
new StringLengthAttribute(32) {
- MinimumLength = 6
+ MinimumLength = 6
}
);
});
@@ -121,7 +121,7 @@ Just create a class derived from the `ExceptionSubscriber` class in your applica
````csharp
public class MyExceptionSubscriber : ExceptionSubscriber
{
- public override async Task HandleAsync(ExceptionNotificationContext context)
+ public async override Task HandleAsync(ExceptionNotificationContext context)
{
//TODO...
}
@@ -244,4 +244,4 @@ We ([Volosoft](https://volosoft.com/) - the core team behind the ABP.IO platform
[ABP Framework](https://abp.io/) provides all the infrastructure and application independent framework features to make you more productive, focus on your own business code and implement software development best practices. It provides you a well defined and comfortable development experience without repeating yourself.
-[ABP Commercial](https://commercial.abp.io/) provides pre-built functionalities, themes and tooling to save your time if your requirements involve these functionalities in addition to the premium support for the framework and the pre-built modules.
\ No newline at end of file
+[ABP Commercial](https://commercial.abp.io/) provides pre-built functionalities, themes and tooling to save your time if your requirements involve these functionalities in addition to the premium support for the framework and the pre-built modules.
diff --git a/docs/en/Blog-Posts/2020-11-12 v4_0_Preview/POST.md b/docs/en/Blog-Posts/2020-11-12 v4_0_Preview/POST.md
new file mode 100644
index 0000000000..03e3d31f97
--- /dev/null
+++ b/docs/en/Blog-Posts/2020-11-12 v4_0_Preview/POST.md
@@ -0,0 +1,143 @@
+# ABP Framework 4.0 RC Has Been Published based on .NET 5.0!
+
+Today, we have released the [ABP Framework](https://abp.io/) (and the [ABP Commercial](https://commercial.abp.io/)) 4.0.0 RC that is based on the **.NET 5.0**. This blog post introduces the new features and important changes in the new version.
+
+> **The planned release date for the [4.0.0 final](https://github.com/abpframework/abp/milestone/45) version is November 26, 2020**.
+
+## Get Started with the 4.0 RC
+
+If you want to try the version `4.0.0` today, follow the steps below;
+
+1) **Upgrade** the ABP CLI to the version `4.0.0-rc.3` using a command line terminal:
+
+````bash
+dotnet tool update Volo.Abp.Cli -g --version 4.0.0-rc.3
+````
+
+**or install** if you haven't installed before:
+
+````bash
+dotnet tool install Volo.Abp.Cli -g --version 4.0.0-rc.3
+````
+
+2) Create a **new application** with the `--preview` option:
+
+````bash
+abp new BookStore --preview
+````
+
+See the [ABP CLI documentation](https://docs.abp.io/en/abp/3.3/CLI) for all the available options.
+
+> You can also use the *Direct Download* tab on the [Get Started](https://abp.io/get-started) page by selecting the **Preview checkbox**.
+
+## Migrating From 3.x to 4.0
+
+The version 4.0 comes with some major changes including the **migration from .NET Core 3.1 to .NET 5.0**.
+
+We've prepared a **detailed [migration document](https://docs.abp.io/en/abp/4.0/Migration-Guides/Abp-4_0)** to explain all the changes and the actions you need to take while upgrading your existing solutions.
+
+## What's new with the ABP Framework 4.0
+
+### The Blazor UI
+
+The Blazor UI is now stable and officially supported. The [web application development tutorial](https://docs.abp.io/en/abp/4.0/Tutorials/Part-1?UI=Blazor) has been updated based on the version 4.0.
+
+#### abp bundle command
+
+Introducing the `abp bundle` CLI command to manage static JavaScript & CSS file dependencies of a Blazor application. This command is currently used to add the dependencies to the `index.html` file in the dependency order by respecting to modularity. In the next version it will automatically unify & minify the files. The documentation is being prepared.
+
+#### Removed the JQuery & Bootstrap JavaScript
+
+Removed JQuery & Bootstrap JavaScript dependencies for the Blazor UI.
+
+>There are some other changes in the startup template and some public APIs. Follow the [Migration Guide](https://docs.abp.io/en/abp/4.0/Migration-Guides/Abp-4_0) to apply changes for existing solutions that you're upgrading from the version 3.3. While we will continue to make improvements add new features, we no longer make breaking changes on the existing APIs until the version 5.0.
+
+#### Others
+
+A lot of minor and major improvements have been done for the Blazor UI. Some of them are listed below:
+
+* Implemented `IComponentActivator` to resolve the component from the `IServiceProvider`. So, you can now inject dependencies into the constructor of your razor component.
+* Introduced the `AbpComponentBase` base class that you derive your components from. It has useful base properties that you can use in your pages/components.
+* Introduced `IUiNotificationService` service to show toast notifications on the UI.
+* Improved the `IUiMessageService` to show message & confirmation dialogs.
+
+### System.Text.Json
+
+ABP Framework 4.0 uses the System.Text.Json by default as the JSON serialization library. It, actually, using a hybrid approach: Continues to use the Newtonsoft.Json when it needs to use the features not supported by the System.Text.Json.
+
+Follow the [Migration Guide](https://docs.abp.io/en/abp/4.0/Migration-Guides/Abp-4_0) to learn how to configure to use the Newtonsoft.Json for some specific types or switch back to the Newtonsoft.Json as the default JSON serializer.
+
+### Identity Server 4 Upgrade
+
+ABP Framework upgrades the [IdentityServer4](https://www.nuget.org/packages/IdentityServer4) library from 3.x to 4.1.1 with the ABP Framework version 4.0. IdentityServer 4.x has a lot of changes. Some of them are **breaking changes in the data structure**.
+
+Follow the [Migration Guide](https://docs.abp.io/en/abp/4.0/Migration-Guides/Abp-4_0) to upgrade existing solutions.
+
+### Creating a New Module Inside the Application
+
+ABP CLI has now a command to create a new module and add it to an existing solution. In this way, you can create modular applications easier than before.
+
+Example: Create a *ProductManagement* module into your solution.
+
+````bash
+abp add-module ProductManagement --new --add-to-solution-file
+````
+
+Execute this command in a terminal in the root folder of your solution. If you don't specify the `--add-to-solution-file` option, then the module projects will not be added to the main solution, but the project references still be added. In this case, you need to open the module's solution to develop the module.
+
+See the [CLI document](https://docs.abp.io/en/abp/4.0/CLI) for other options.
+
+### WPF Startup Template
+
+Introducing the WPF startup template for the ABP Framework. Use the ABP CLI new command to create a new WPF application:
+
+````bash
+abp new MyWpfApp -t wpf
+````
+
+This is a minimalist, empty project template that is integrated to the ABP Framework.
+
+### New Languages
+
+**Thanks to the contributors** from the ABP Community, the framework modules and the startup template have been localized to **German** language by [Alexander Pilhar](https://github.com/alexanderpilhar) & [Nico Lachmuth](https://github.com/tntwist).
+
+### Other Notes
+
+* Upgraded to Angular 11.
+* Since [Mongo2Go](https://github.com/Mongo2Go/Mongo2Go) library not supports transactions, you can use transactions in unit tests for MongoDB.
+
+## What's new with the ABP Commercial 4.0
+
+### The Blazor UI
+
+The Blazor UI for the ABP Commercial is also becomes stable and feature rich with the version 4.0;
+
+* [ABP Suite](https://commercial.abp.io/tools/suite) now supports to generate CRUD pages for the Blazor UI.
+* Completed the [Lepton Theme](https://commercial.abp.io/themes) for the Blazor UI.
+* Implemented the [File Management](https://commercial.abp.io/modules/Volo.FileManagement) module for the Blazor UI.
+
+### The ABP Suite
+
+While creating create/edit modals with a navigation property, we had two options: A dropdown to select the target entity and a modal to select the entity by searching with a data table.
+
+Dropdown option now supports **lazy load, search and auto-complete**. In this way, selecting a navigation property becomes much easier and supports large data sets on the dropdown.
+
+**Example: Select an author while creating a new book**
+
+
+
+With the new version, you can **disable backend code generation** on CRUD page generation. This is especially useful if you want to regenerate the page with a different UI framework, but don't want to regenerate the server side code.
+
+### Identity Server Management UI Revised
+
+Completely revised the Identity Server Management UI based on the IDS 4.x changes.
+
+## About the Next Release
+
+The next feature version, `4.1.0`, will mostly focus on completing the missing documents, fixing bugs, performance optimizations and improving the Blazor UI features. The planned preview release date for the version `4.1.0` is December 10 and the final (stable) version release date is December 24.
+
+Follow the [GitHub milestones](https://github.com/abpframework/abp/milestones) for all the planned ABP Framework version release dates.
+
+## Feedback
+
+Please check out the ABP Framework 4.0.0 RC and [provide feedback](https://github.com/abpframework/abp/issues/new) to help us to release a more stable version. **The planned release date for the [4.0.0 final](https://github.com/abpframework/abp/milestone/45) version is November 26**.
diff --git a/docs/en/Blog-Posts/2020-11-12 v4_0_Preview/abp-suite-auto-complete-dropdown.png b/docs/en/Blog-Posts/2020-11-12 v4_0_Preview/abp-suite-auto-complete-dropdown.png
new file mode 100644
index 0000000000..2021ef42bd
Binary files /dev/null and b/docs/en/Blog-Posts/2020-11-12 v4_0_Preview/abp-suite-auto-complete-dropdown.png differ
diff --git a/docs/en/CLI.md b/docs/en/CLI.md
index 8713bc9019..a119770286 100644
--- a/docs/en/CLI.md
+++ b/docs/en/CLI.md
@@ -41,6 +41,7 @@ Here, the list of all available commands before explaining their details:
* **`login`**: Authenticates on your computer with your [abp.io](https://abp.io/) username and password.
* **`logout`**: Logouts from your computer if you've authenticated before.
* **`build`**: Builds a GIT repository and depending repositories or a single .NET solution.
+* **`bundle`**: Generates script and style references for an ABP Blazor project.
### help
@@ -159,6 +160,8 @@ abp add-package Volo.Abp.MongoDB
Adds a [multi-package application module](Modules/Index) to a solution by finding all packages of the module, finding related projects in the solution and adding each package to the corresponding project in the solution.
+It can also create a new module for your solution and add it to your solution. See `--new-template` option.
+
> A business module generally consists of several packages (because of layering, different database provider options or other reasons). Using `add-module` command dramatically simplifies adding a module to a solution. However, each module may require some additional configurations which is generally indicated in the documentation of the related module.
Usage
@@ -167,21 +170,29 @@ Usage
abp add-module [options]
````
-Example:
+Examples:
```bash
abp add-module Volo.Blogging
```
-* This example add the Volo.Blogging module to the solution.
+* This example adds the `Volo.Blogging` module to the solution.
+
+```bash
+abp add-module ProductManagement --new --add-to-solution-file
+```
+
+* This command creates a fresh new module customized for your solution (named `ProductManagement`) and adds it to your solution.
+
#### Options
* `--solution` or `-s`: Specifies the solution (.sln) file path. If not specified, CLI tries to find a .sln file in the current directory.
* `--skip-db-migrations`: For EF Core database provider, it automatically adds a new code first migration (`Add-Migration`) and updates the database (`Update-Database`) if necessary. Specify this option to skip this operation.
* `-sp` or `--startup-project`: Relative path to the project folder of the startup project. Default value is the current folder.
-* `--with-source-code`: Downloads the source code of the module to your solution folder and uses local project references instead of NuGet/NPM packages.
-* `--add-to-solution-file`: Adds the downloaded module to your solution file, so you will also see the projects of the module when you open the solution on a IDE. (only available when `--with-source-code` is used.)
+* `--new`: Creates a fresh new module (customized for your solution) and adds it to your solution.
+* `--with-source-code`: Downloads the source code of the module to your solution folder and uses local project references instead of NuGet/NPM packages. This options is always `True` if `--new` is used.
+* `--add-to-solution-file`: Adds the downloaded/created module to your solution file, so you will also see the projects of the module when you open the solution on a IDE. (only available when `--with-source-code` is `True`.)
### get-source
@@ -383,10 +394,27 @@ abp build --build-name "prod" --dotnet-build-arguments "\"--no-dependencies\""
#### Options
-* ```--working-directory``` or ```-w```: Specifies the working directory. This option is useful when the command is executed outside of a GIT repository or when executing directory doesn't contain a .NET solution file.
+* ```--working-directory``` or ```-wd```: Specifies the working directory. This option is useful when the command is executed outside of a GIT repository or when executing directory doesn't contain a .NET solution file.
* ```--build-name``` or ```-n```: Specifies a name for the build. This option is useful when same repository is used for more than one different builds.
* ```--dotnet-build-arguments``` or ```-a```: Arguments to pass ```dotnet build``` when building project files. This parameter must be passed like ```"\"{params}\""``` .
* ```--force``` or ```-f```: Forces to build projects even they are not changed from the last successful build.
For more details, see [build command documentation](CLI-BuildCommand.md).
+
+#### bundle
+
+This command generates script and style references for an ABP Blazor project and updates the **index.html** file. It helps developers to manage dependencies required by ABP modules easily. In order ```bundle``` command to work, its **executing directory** or passed ```--working-directory``` parameter's directory must contain a Blazor project file(*.csproj).
+
+Usage:
+
+````bash
+abp bundle [options]
+````
+
+#### Options
+
+* ```--working-directory``` or ```-wd```: Specifies the working directory. This option is useful when executing directory doesn't contain a Blazor project file.
+* ```--force``` or ```-f```: Forces to build project before generating references.
+
+For more details about managing style and script references in Blazor apps, see [Managing Global Scripts & Styles](UI/Blazor/Global-Scripts-Styles.md)
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2020-04-19-Customize-the-SignIn-Manager/POST.md b/docs/en/Community-Articles/2020-04-19-Customize-the-SignIn-Manager/POST.md
index d46049526d..1342407880 100644
--- a/docs/en/Community-Articles/2020-04-19-Customize-the-SignIn-Manager/POST.md
+++ b/docs/en/Community-Articles/2020-04-19-Customize-the-SignIn-Manager/POST.md
@@ -1,6 +1,6 @@
# How to Customize the SignIn Manager for ABP Applications
-After creating a new application using the [application startup template](https://docs.abp.io/en/abp/latest/Startup-Templates/Application), you may want extend or change the default behavior of the SignIn Manager for your authentication and registration flow needs. ABP [Account Module](https://docs.abp.io/en/abp/latest/Modules/Account) uses the [Identity Management Module](https://docs.abp.io/en/abp/latest/Modules/Identity) for SignIn Manager and the [Identity Management Module](https://docs.abp.io/en/abp/latest/Modules/Identity) uses default [Microsoft Identity SignIn Manager](https://github.com/dotnet/aspnetcore/blob/master/src/Identity/Core/src/SignInManager.cs) ([see here](https://github.com/abpframework/abp/blob/be32a55449e270d2d456df3dabdc91f3ffdd4fa9/modules/identity/src/Volo.Abp.Identity.AspNetCore/Volo/Abp/Identity/AspNetCore/AbpIdentityAspNetCoreModule.cs#L17)).
+After creating a new application using the [application startup template](https://docs.abp.io/en/abp/latest/Startup-Templates/Application), you may want extend or change the default behavior of the SignIn Manager for your authentication and registration flow needs. ABP [Account Module](https://docs.abp.io/en/abp/latest/Modules/Account) uses the [Identity Management Module](https://docs.abp.io/en/abp/latest/Modules/Identity) for SignIn Manager and the [Identity Management Module](https://docs.abp.io/en/abp/latest/Modules/Identity) uses default [Microsoft Identity SignIn Manager](https://github.com/dotnet/aspnetcore/blob/master/src/Identity/Core/src/SignInManager.cs) ([see here](https://github.com/abpframework/abp/blob/be32a55449e270d2d456df3dabdc91f3ffdd4fa9/modules/identity/src/Volo.Abp.Identity.AspNetCore/Volo/Abp/Identity/AspNetCore/AbpIdentityAspNetCoreModule.cs#L17)).
To write your Custom SignIn Manager, you need to extend [Microsoft Identity SignIn Manager](https://github.com/dotnet/aspnetcore/blob/master/src/Identity/Core/src/SignInManager.cs) class and register it to the DI container.
@@ -27,7 +27,7 @@ public class CustomSignInManager : Microsoft.AspNetCore.Identity.SignInManager It is important to use **Volo.Abp.Identity.IdentityUser** type for SignInManager to inherit, not the AppUser of your application.
+> It is important to use **Volo.Abp.Identity.IdentityUser** type for SignInManager to inherit, not the AppUser of your application.
Afterwards you can override any of the SignIn Manager methods you need and add new methods and properties needed for your authentication or registration flow.
@@ -38,7 +38,7 @@ In this case we'll be overriding the `GetExternalLoginInfoAsync` method which is
A good way to override a method is copying its [source code](https://github.com/dotnet/aspnetcore/blob/c56aa320c32ee5429d60647782c91d53ac765865/src/Identity/Core/src/SignInManager.cs#L638-L674). In this case, we will be using a minorly modified version of the source code which explicitly shows the namespaces of the methods and properties to help better understanding of the concept.
````csharp
-public override async Task GetExternalLoginInfoAsync(string expectedXsrf = null)
+public async override Task GetExternalLoginInfoAsync(string expectedXsrf = null)
{
var auth = await Context.AuthenticateAsync(Microsoft.AspNetCore.Identity.IdentityConstants.ExternalScheme);
var items = auth?.Properties?.Items;
@@ -93,4 +93,4 @@ PreConfigure(identityBuilder =>
## The Source Code
-You can find the source code of the completed example [here](https://github.com/abpframework/abp-samples/tree/master/Authentication-Customization).
\ No newline at end of file
+You can find the source code of the completed example [here](https://github.com/abpframework/abp-samples/tree/master/Authentication-Customization).
diff --git a/docs/en/Community-Articles/2020-08-07-Passwordless-Authentication/POST.md b/docs/en/Community-Articles/2020-08-07-Passwordless-Authentication/POST.md
index b31ea023f2..9489c004bd 100644
--- a/docs/en/Community-Articles/2020-08-07-Passwordless-Authentication/POST.md
+++ b/docs/en/Community-Articles/2020-08-07-Passwordless-Authentication/POST.md
@@ -33,7 +33,7 @@ namespace PasswordlessAuthentication.Web
}
//We need to override this method as well.
- public override async Task GetUserModifierAsync(string purpose, UserManager manager, TUser user)
+ public async override Task GetUserModifierAsync(string purpose, UserManager manager, TUser user)
{
var userId = await manager.GetUserIdAsync(user);
@@ -105,7 +105,7 @@ namespace PasswordlessAuthentication.Web.Pages
UserManager = userManager;
_userRepository = userRepository;
}
-
+
public ActionResult OnGet()
{
if (!CurrentUser.IsAuthenticated)
@@ -115,15 +115,15 @@ namespace PasswordlessAuthentication.Web.Pages
return Page();
}
-
+
//added for passwordless authentication
public async Task OnPostGeneratePasswordlessTokenAsync()
{
var adminUser = await _userRepository.FindByNormalizedUserNameAsync("admin");
-
+
var token = await UserManager.GenerateUserTokenAsync(adminUser, "PasswordlessLoginProvider",
"passwordless-auth");
-
+
PasswordlessLoginUrl = Url.Action("Login", "Passwordless",
new {token = token, userId = adminUser.Id.ToString()}, Request.Scheme);
@@ -238,7 +238,7 @@ namespace PasswordlessAuthentication.Web.Controllers
return Redirect("/");
}
-
+
private static IEnumerable CreateClaims(IUser user, IEnumerable roles)
{
var claims = new List
@@ -272,4 +272,4 @@ That's all! We created a passwordless login with 7 steps.
## Source Code
-The completed sample is available on [GitHub repository](https://github.com/abpframework/abp-samples/tree/master/PasswordlessAuthentication).
\ No newline at end of file
+The completed sample is available on [GitHub repository](https://github.com/abpframework/abp-samples/tree/master/PasswordlessAuthentication).
diff --git a/docs/en/Community-Articles/2020-08-31-Adding-User-Navigation-In-Suite/POST.md b/docs/en/Community-Articles/2020-08-31-Adding-User-Navigation-In-Suite/POST.md
index 7bfd20637f..3fd6b845a9 100644
--- a/docs/en/Community-Articles/2020-08-31-Adding-User-Navigation-In-Suite/POST.md
+++ b/docs/en/Community-Articles/2020-08-31-Adding-User-Navigation-In-Suite/POST.md
@@ -22,6 +22,8 @@ Then add a string property called `Title`, as an example property.
### Create AppUserDto
+_Note that, creating `AppUserDto` is not necessary after ABP v4.X_
+
ABP Suite needs a DTO for the target entity (user, in this case) in order to define a navigation property.
To do this, create a new folder called "Users" in `*.Application.Contracts` then add a new class called `AppUserDto` inherited from `IdentityUserDto`.
diff --git a/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/POST.md b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/POST.md
new file mode 100644
index 0000000000..f1387b2a50
--- /dev/null
+++ b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/POST.md
@@ -0,0 +1,46 @@
+# How to add a new language to your ABP project?
+
+Adding a new language to your ABP project is pretty simple. Let's add the German language to our ABP project:
+
+
+
+1. Go to your solution's root folder and write the following CLI command. This command will generate an empty translation file from English.
+ ```bash
+ abp translate -c de-DE
+ ```
+
+ Check out for [the complete supported culture codes](https://docs.microsoft.com/en-us/bingmaps/rest-services/common-parameters-and-types/supported-culture-codes).
+ (For internal development `D:\Github\abp` and `D:\Github\volo\abp`)
+
+2. Fill the `target` fields in your target language.
+
+ 
+
+3. Copy `abp-translation.json` your solution's root folder (Do not change the filename!)
+
+4. Run the following command. This command will create the necessary `json` files.
+ ```bash
+ abp translate --apply
+ ```
+
+5. Open your solution and add the new language to the language list. To do this;
+
+ * open `MyProjectNameDomainModule.cs` and in `ConfigureServices` you'll find `Configure`. If you have `HttpApi.Host` project then you need to add this in `MyProjectNameHttpApiHostModule.cs`
+
+ ```
+ options.Languages.Add(new LanguageInfo("de-DE", "de-DE", "Deutsch", "de"));
+ ```
+
+ 
+
+ The last parameter is the flag icon. You can find the list of flag icons on https://flagicons.lipis.dev/
+
+ 6. The last step is running the DbMigrator project. It will seed the database for the new language.
+ 
+
+
+
+Close the IIS Express / Kestrel to invalidate the language cache and run the project. You will see the new language on your website.
+
+
+
diff --git a/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/add-to-languages.png b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/add-to-languages.png
new file mode 100644
index 0000000000..1dda317685
Binary files /dev/null and b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/add-to-languages.png differ
diff --git a/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/database-table.png b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/database-table.png
new file mode 100644
index 0000000000..2d512fabeb
Binary files /dev/null and b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/database-table.png differ
diff --git a/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/language-target.png b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/language-target.png
new file mode 100644
index 0000000000..7b5f104003
Binary files /dev/null and b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/language-target.png differ
diff --git a/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/website-new-language.png b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/website-new-language.png
new file mode 100644
index 0000000000..1e04841983
Binary files /dev/null and b/docs/en/Community-Articles/2020-11-02-How-To-Add-New-Language/website-new-language.png differ
diff --git a/docs/en/Customizing-Application-Modules-Overriding-Services.md b/docs/en/Customizing-Application-Modules-Overriding-Services.md
index b8706eea61..133b2e89eb 100644
--- a/docs/en/Customizing-Application-Modules-Overriding-Services.md
+++ b/docs/en/Customizing-Application-Modules-Overriding-Services.md
@@ -29,7 +29,7 @@ public class TestAppService : IIdentityUserAppService, ITransientDependency
}
````
-The dependency injection system allows to register multiple services for the same interface. The last registered one is used when the interface is injected. It is a good practice to explicitly replace the service.
+The dependency injection system allows to register multiple services for the same interface. The last registered one is used when the interface is injected. It is a good practice to explicitly replace the service.
Example:
@@ -59,7 +59,6 @@ In most cases, you will want to change one or a few methods of the current imple
### Example: Overriding an Application Service
````csharp
-//[RemoteService(IsEnabled = false)] // If you use dynamic controller feature you can disable remote service. Prevent creating duplicate controller for the application service.
[Dependency(ReplaceServices = true)]
[ExposeServices(typeof(IIdentityUserAppService), typeof(IdentityUserAppService), typeof(MyIdentityUserAppService))]
public class MyIdentityUserAppService : IdentityUserAppService
@@ -76,7 +75,7 @@ public class MyIdentityUserAppService : IdentityUserAppService
{
}
- public override async Task CreateAsync(IdentityUserCreateDto input)
+ public async override Task CreateAsync(IdentityUserCreateDto input)
{
if (input.PhoneNumber.IsNullOrWhiteSpace())
{
@@ -109,33 +108,33 @@ public class MyIdentityUserManager : IdentityUserManager
{
public MyIdentityUserManager(
IdentityUserStore store,
- IIdentityRoleRepository roleRepository,
+ IIdentityRoleRepository roleRepository,
IIdentityUserRepository userRepository,
- IOptions optionsAccessor,
+ IOptions optionsAccessor,
IPasswordHasher passwordHasher,
- IEnumerable> userValidators,
- IEnumerable> passwordValidators,
+ IEnumerable> userValidators,
+ IEnumerable> passwordValidators,
ILookupNormalizer keyNormalizer,
IdentityErrorDescriber errors,
IServiceProvider services,
- ILogger logger,
- ICancellationTokenProvider cancellationTokenProvider) :
+ ILogger logger,
+ ICancellationTokenProvider cancellationTokenProvider) :
base(store,
roleRepository,
- userRepository,
- optionsAccessor,
- passwordHasher,
- userValidators,
+ userRepository,
+ optionsAccessor,
+ passwordHasher,
+ userValidators,
passwordValidators,
- keyNormalizer,
- errors,
- services,
- logger,
+ keyNormalizer,
+ errors,
+ services,
+ logger,
cancellationTokenProvider)
{
}
- public override async Task CreateAsync(IdentityUser user)
+ public async override Task CreateAsync(IdentityUser user)
{
if (user.PhoneNumber.IsNullOrWhiteSpace())
{
@@ -182,7 +181,7 @@ namespace MyProject.Controllers
}
- public override async Task SendPasswordResetCodeAsync(
+ public async override Task SendPasswordResetCodeAsync(
SendPasswordResetCodeDto input)
{
Logger.LogInformation("Your custom logic...");
@@ -214,7 +213,7 @@ Assuming that you've already added a `SocialSecurityNumber` as described in the
You can use the [object extension system](Object-Extensions.md) to add the property to the `IdentityUserDto`. Write this code inside the `YourProjectNameDtoExtensions` class comes with the application startup template:
````csharp
-ObjectExtensionManager.Instance
+ObjectExtensionManager.Instance
.AddOrUpdateProperty(
"SocialSecurityNumber"
);
@@ -286,8 +285,8 @@ ObjectExtensionManager.Instance
.AddOrUpdateProperty(
new[]
{
- typeof(IdentityUserDto),
- typeof(IdentityUserCreateDto),
+ typeof(IdentityUserDto),
+ typeof(IdentityUserCreateDto),
typeof(IdentityUserUpdateDto)
},
"SocialSecurityNumber"
diff --git a/docs/en/Data-Filtering.md b/docs/en/Data-Filtering.md
index 626719b2fc..6df738cf18 100644
--- a/docs/en/Data-Filtering.md
+++ b/docs/en/Data-Filtering.md
@@ -32,6 +32,8 @@ namespace Acme.BookStore
> `ISoftDelete` filter is enabled by default and you can not get deleted entities from database unless you explicitly disable it. See the `IDataFilter` service below.
+> Soft-delete entities can be hard-deleted when you use `HardDeleteAsync` method on the repositories.
+
### IMultiTenant
[Multi-tenancy](Multi-Tenancy.md) is an efficient way of creating SaaS applications. Once you create a multi-tenant application, you typically want to isolate data between tenants. Implement `IMultiTenant` interface to make your entity "multi-tenant aware".
diff --git a/docs/en/Data-Seeding.md b/docs/en/Data-Seeding.md
index 8827fe96c0..f3e9414f0f 100644
--- a/docs/en/Data-Seeding.md
+++ b/docs/en/Data-Seeding.md
@@ -38,31 +38,37 @@ namespace Acme.BookStore
{
private readonly IRepository _bookRepository;
private readonly IGuidGenerator _guidGenerator;
+ private readonly ICurrentTenant _currentTenant;
public BookStoreDataSeedContributor(
IRepository bookRepository,
- IGuidGenerator guidGenerator)
+ IGuidGenerator guidGenerator,
+ ICurrentTenant currentTenant)
{
_bookRepository = bookRepository;
_guidGenerator = guidGenerator;
+ _currentTenant = currentTenant;
}
public async Task SeedAsync(DataSeedContext context)
{
- if (await _bookRepository.GetCountAsync() > 0)
+ using (_currentTenant.Change(context?.TenantId))
{
- return;
+ if (await _bookRepository.GetCountAsync() > 0)
+ {
+ return;
+ }
+
+ var book = new Book(
+ id: _guidGenerator.Create(),
+ name: "The Hitchhiker's Guide to the Galaxy",
+ type: BookType.ScienceFiction,
+ publishDate: new DateTime(1979, 10, 12),
+ price: 42
+ );
+
+ await _bookRepository.InsertAsync(book);
}
-
- var book = new Book(
- id: _guidGenerator.Create(),
- name: "The Hitchhiker's Guide to the Galaxy",
- type: BookType.ScienceFiction,
- publishDate: new DateTime(1979, 10, 12),
- price: 42
- );
-
- await _bookRepository.InsertAsync(book);
}
}
}
diff --git a/docs/en/Domain-Driven-Design-Implementation-Guide.md b/docs/en/Domain-Driven-Design-Implementation-Guide.md
new file mode 100644
index 0000000000..b0a9692f83
--- /dev/null
+++ b/docs/en/Domain-Driven-Design-Implementation-Guide.md
@@ -0,0 +1,95 @@
+# Implementing Domain Driven Design
+
+## Introduction
+
+### Goals
+
+This is an **integrative guide** for implementing the Domain Driven Design (DDD). The goals of this document are;
+
+* **Introduce and explain** the DDD architecture, concepts, principles, patterns and building blocks.
+* Explain the **layered architecture** & solution structure offered by the ABP Framework.
+* Introduce **explicit rules** to implement DDD patterns and best practices by giving **concrete examples**.
+* Show what **ABP Framework provides** you as the infrastructure for implementing DDD in a proper way.
+
+### Simple Code!
+
+> **Playing football** is very **simple**, but **playing simple football** is the **hardest thing** there is.
+> — Johan Cruyff
+
+If we take this famous quote for programming, we can say;
+
+> **Writing code** is very **simple**, but **writing simple code** is the **hardest thing** there is.
+> — ???
+
+In this document, we will introduce **simple rules** those are **easy to implement**.
+
+Once your **application grows**, it will be **hard to follow** these rules. Sometimes you find **breaking rules** will save your time in a short term. However, the saved time in the short term will bring much **more time loss** in the middle and long term. Your code base becomes **complicated** and hard to maintain. Most of the business applications are **re-written** just because you **can't maintain** it anymore.
+
+If you **follow the rules and best practices**, your code base will be simpler and easier to maintain. Your application **react to changes** faster.
+
+## What is the Domain Driven Design?
+
+Domain-driven design (DDD) is an approach to software development for **complex** needs by connecting the implementation to an **evolving** model;
+
+DDD is suitable for **complex domains** and **large-scale** applications rather than simple CRUD applications. It focuses on the **core domain logic** rather than the infrastructure details. It helps to build a **flexible**, modular and **maintainable** code base.
+
+### OOP & SOLID
+
+Implementing DDD highly relies on the Object Oriented Programming (OOP) and [SOLID](https://en.wikipedia.org/wiki/SOLID) principles. Actually, it **implements** and **extends** these principles. So, a **good understanding** of OOP & SOLID helps you a lot while truly implementing the DDD.
+
+### DDD Layers & Clean Architecture
+
+There are four fundamental layers of a Domain Driven Based Solution;
+
+
+
+**Business Logic** places into two layers, the *Domain layer* and the *Application Layer*, while they contains different kinds of business logic;
+
+* **Domain Layer** implements the core, use-case independent business logic of the domain/system.
+* **Application Layer** implements the use cases of the application based on the domain. A use case can be thought as a user interaction on the User Interface (UI).
+* **Presentation Layer** contains the UI elements (pages, components) of the application.
+* **Infrastructure Layer** supports other layer by implementing the abstractions and integrations to 3rd-party library and systems.
+
+The same layering can be shown as the diagram below and known as the **Clean Architecture**, or sometimes the **Onion Architecture**:
+
+
+
+In the Clean Architecture, each layer only **depends on the layer directly inside it**. The most independent layer is shown in the most inner circle and it is the Domain Layer.
+
+### Core Building Blocks
+
+DDD mostly **focuses on the Domain & Application Layers** and ignores the Presentation and Infrastructure. They are seen as *details* and the business layers should not depend on them.
+
+That doesn't mean the Presentation and Infrastructure layers are not important. They are very important. UI frameworks and database providers have their own rules and best practices that you need to know and apply. However these are not in the topics of DDD.
+
+This section introduces the essential building blocks of the Domain & Application Layers.
+
+#### Domain Layer Building Blocks
+
+* **Entity**: An [Entity](Entities.md) is an object with its own properties (state, data) and methods that implements the business logic that is executed on these properties. An entity is represented by its unique identifier (Id). Two entity object with different Ids are considered as different entities.
+* **Value Object**: A [Value Object](Value-Objects.md) is another kind of domain object that is identified by its properties rather than a unique Id. That means two Value Objects with same properties are considered as the same object. Value objects are generally implemented as immutable and mostly are much simpler than the Entities.
+* **Aggregate & Aggregate Root**: An [Aggregate](Entities.md) is a cluster of objects (entities and value objects) bound together by an **Aggregate Root** object. The Aggregate Root is a specific type of an entity with some additional responsibilities.
+* **Repository**: A [Repository](Repositories.md) is a collection-like interface that is used by the Domain and Application Layers to access to the data persistence system (the database). It hides the complexity of the DBMS from the business code.
+* **Domain Service**: A [Domain Service](Domain-Services.md) is a stateless service that implements core business rules of the domain. It is useful to implement domain logic that depends on multiple aggregate (entity) type or some external services.
+* **Specification**: A [Specification](Specifications.md) is used to define named, reusable and combinable filters for entities and other business objects.
+* **Domain Event**: A [Domain Event](Event-Bus.md) is a way of informing other services in a loosely coupled manner, when a domain specific event occurs.
+
+#### Application Layer Building Blocks
+
+* **Application Service**: An [Application Service](Application-Services.md) is a stateless service that implements use cases of the application. An application service typically gets and returns DTOs. It is used by the Presentation Layer. It uses and coordinates the domain objects to implement the use cases. A use case is typically considered as a Unit Of Work.
+* **Data Transfer Object (DTO)**: A [DTO](Data-Transfer-Objects.md) is a simple object without any business logic that is used to transfer state (data) between the Application and Presentation Layers.
+* **Unit of Work (UOW)**: A [Unit of Work](Unit-Of-Work.md) is an atomic work that should be done as a transaction unit. All the operations inside a UOW should be committed on success or rolled back on a failure.
+
+## Implementation: The Big Picture
+
+### Layering of a .NET Solution
+
+TODO
+
+### Execution Flow a DDD Based Application
+
+TODO
+
+### Common Principles
+
+TODO
\ No newline at end of file
diff --git a/docs/en/Domain-Driven-Design.md b/docs/en/Domain-Driven-Design.md
index ce33295492..f3d2699cab 100644
--- a/docs/en/Domain-Driven-Design.md
+++ b/docs/en/Domain-Driven-Design.md
@@ -19,15 +19,19 @@ ABP follows DDD principles and patterns to achieve a layered application model w
- **Domain Layer**: Includes business objects and the core (domain) business rules. This is the heart of the application.
- **Infrastructure Layer**: Provides generic technical capabilities that support higher layers mostly using 3rd-party libraries.
+DDD mostly interest in the **Domain** and the **Application** layers, rather than the Infrastructure and the Presentation layers.
+
## Contents
+See the following documents to learn what ABP Framework provides to you to implement DDD in your project.
+
* **Domain Layer**
* [Entities & Aggregate Roots](Entities.md)
- * Value Objects
* [Repositories](Repositories.md)
- * Domain Services
- * Specifications
+ * [Domain Services](Domain-Services.md)
+ * [Value Objects](Value-Objects.md)
+ * [Specifications](Specifications.md)
* **Application Layer**
* [Application Services](Application-Services.md)
* [Data Transfer Objects (DTOs)](Data-Transfer-Objects.md)
- * Unit of Work
\ No newline at end of file
+ * [Unit of Work](Unit-Of-Work.md)
\ No newline at end of file
diff --git a/docs/en/Domain-Services.md b/docs/en/Domain-Services.md
index 7eb6c917cd..5a4f6393e1 100644
--- a/docs/en/Domain-Services.md
+++ b/docs/en/Domain-Services.md
@@ -1,3 +1,135 @@
-# ABP Documentation
+# Domain Services
-TODO!
\ No newline at end of file
+## Introduction
+
+In a [Domain Driven Design](Domain-Driven-Design.md) (DDD) solution, the core business logic is generally implemented in aggregates ([entities](Entities.md)) and the Domain Services. Creating a Domain Service is especially needed when;
+
+* You implement a core domain logic that depends on some services (like repositories or other external services).
+* The logic you need to implement is related to more than one aggregate/entity, so it doesn't properly fit in any of the aggregates.
+
+## ABP Domain Service Infrastructure
+
+Domain Services are simple, stateless classes. While you don't have to derive from any service or interface, ABP Framework provides some useful base classes and conventions.
+
+### DomainService & IDomainService
+
+Either derive a Domain Service from the `DomainService` base class or directly implement the `IDomainService` interface.
+
+**Example: Create a Domain Service deriving from the `DomainService` base class.**
+
+````csharp
+using Volo.Abp.Domain.Services;
+
+namespace MyProject.Issues
+{
+ public class IssueManager : DomainService
+ {
+
+ }
+}
+````
+
+When you do that;
+
+* ABP Framework automatically registers the class to the Dependency Injection system with a Transient lifetime.
+* You can directly use some common services as base properties, without needing to manually inject (e.g. [ILogger](Logging.md) and [IGuidGenerator](Guid-Generation.md)).
+
+> It is suggested to name a Domain Service with a `Manager` or `Service` suffix. We typically use the `Manager` suffix as used in the sample above.
+
+**Example: Implement the domain logic of assigning an Issue to a User**
+
+````csharp
+public class IssueManager : DomainService
+{
+ private readonly IRepository _issueRepository;
+
+ public IssueManager(IRepository issueRepository)
+ {
+ _issueRepository = issueRepository;
+ }
+
+ public async Task AssignAsync(Issue issue, AppUser user)
+ {
+ var currentIssueCount = await _issueRepository
+ .CountAsync(i => i.AssignedUserId == user.Id);
+
+ //Implementing a core business validation
+ if (currentIssueCount >= 3)
+ {
+ throw new IssueAssignmentException(user.UserName);
+ }
+
+ issue.AssignedUserId = user.Id;
+ }
+}
+````
+
+Issue is an [aggregate root](Entities.md) defined as shown below:
+
+````csharp
+public class Issue : AggregateRoot
+{
+ public Guid? AssignedUserId { get; internal set; }
+
+ //...
+}
+````
+
+* Making the setter `internal` ensures that it can not directly set in the upper layers and forces to always use the `IssueManager` to assign an `Issue` to a `User`.
+
+### Using a Domain Service
+
+A Domain Service is typically used in an [application service](Application-Services.md).
+
+**Example: Use the `IssueManager` to assign an Issue to a User**
+
+````csharp
+using System;
+using System.Threading.Tasks;
+using MyProject.Users;
+using Volo.Abp.Application.Services;
+using Volo.Abp.Domain.Repositories;
+
+namespace MyProject.Issues
+{
+ public class IssueAppService : ApplicationService, IIssueAppService
+ {
+ private readonly IssueManager _issueManager;
+ private readonly IRepository _userRepository;
+ private readonly IRepository _issueRepository;
+
+ public IssueAppService(
+ IssueManager issueManager,
+ IRepository userRepository,
+ IRepository issueRepository)
+ {
+ _issueManager = issueManager;
+ _userRepository = userRepository;
+ _issueRepository = issueRepository;
+ }
+
+ public async Task AssignAsync(Guid id, Guid userId)
+ {
+ var issue = await _issueRepository.GetAsync(id);
+ var user = await _userRepository.GetAsync(userId);
+
+ await _issueManager.AssignAsync(issue, user);
+ await _issueRepository.UpdateAsync(issue);
+ }
+ }
+}
+````
+
+Since the `IssueAppService` is in the Application Layer, it can't directly assign an issue to a user. So, it uses the `IssueManager`.
+
+## Application Services vs Domain Services
+
+While both of [Application Services](Application-Services.md) and Domain Services implement the business rules, there are fundamental logical and formal differences;
+
+* Application Services implement the **use cases** of the application (user interactions in a typical web application), while Domain Services implement the **core, use case independent domain logic**.
+* Application Services get/return [Data Transfer Objects](Data-Transfer-Objects.md), Domain Service methods typically get and return the **domain objects** ([entities](Entities.md), [value objects](Value-Objects.md)).
+* Domain services are typically used by the Application Services or other Domain Services, while Application Services are used by the Presentation Layer or Client Applications.
+
+## Lifetime
+
+Lifetime of Domain Services are [transient](https://docs.abp.io/en/abp/latest/Dependency-Injection) and they are automatically registered to the dependency injection system.
\ No newline at end of file
diff --git a/docs/en/Entity-Framework-Core.md b/docs/en/Entity-Framework-Core.md
index 98163e7c28..8c0a1c9cc9 100644
--- a/docs/en/Entity-Framework-Core.md
+++ b/docs/en/Entity-Framework-Core.md
@@ -92,7 +92,7 @@ protected override void OnModelCreating(ModelBuilder builder)
b.ToTable("Books");
//Configure the base properties
- b.ConfigureByConvention();
+ b.ConfigureByConvention();
//Configure other properties (if you are using the fluent API)
b.Property(x => x.Name).IsRequired().HasMaxLength(128);
@@ -113,7 +113,7 @@ If you have multiple databases in your application, you can configure the connec
[ConnectionStringName("MySecondConnString")]
public class MyDbContext : AbpDbContext
{
-
+
}
```
@@ -174,7 +174,7 @@ public class Book : AggregateRoot
}
```
-(`BookType` is a simple enum here) And you want to create a new `Book` entity in a [domain service](Domain-Services.md):
+(`BookType` is a simple `enum` here and not important) And you want to create a new `Book` entity in a [domain service](Domain-Services.md):
````csharp
public class BookManager : DomainService
@@ -221,12 +221,13 @@ public interface IBookRepository : IRepository
}
````
-You generally want to derive from the `IRepository` to inherit standard repository methods. However, you don't have to. Repository interfaces are defined in the domain layer of a layered application. They are implemented in the data/infrastructure layer (`EntityFrameworkCore` project in a [startup template](https://abp.io/Templates)).
+You generally want to derive from the `IRepository` to inherit standard repository methods (while, you don't have to do). Repository interfaces are defined in the domain layer of a layered application. They are implemented in the data/infrastructure layer (`EntityFrameworkCore` project in a [startup template](https://abp.io/Templates)).
Example implementation of the `IBookRepository` interface:
````csharp
-public class BookRepository : EfCoreRepository, IBookRepository
+public class BookRepository
+ : EfCoreRepository, IBookRepository
{
public BookRepository(IDbContextProvider dbContextProvider)
: base(dbContextProvider)
@@ -254,7 +255,7 @@ If you want to replace default repository implementation with your custom reposi
context.Services.AddAbpDbContext(options =>
{
options.AddDefaultRepositories();
-
+
//Replaces IRepository
options.AddRepository();
});
@@ -263,7 +264,7 @@ context.Services.AddAbpDbContext(options =>
This is especially important when you want to **override a base repository method** to customize it. For instance, you may want to override `DeleteAsync` method to delete a specific entity in a more efficient way:
````csharp
-public override async Task DeleteAsync(
+public async override Task DeleteAsync(
Guid id,
bool autoSave = false,
CancellationToken cancellationToken = default)
@@ -272,6 +273,278 @@ public override async Task DeleteAsync(
}
````
+## Loading Related Entities
+
+Assume that you've an `Order` with a collection of `OrderLine`s and the `OrderLine` has a navigation property to the `Order`:
+
+````csharp
+using System;
+using System.Collections.Generic;
+using System.Collections.ObjectModel;
+using Volo.Abp.Auditing;
+using Volo.Abp.Domain.Entities;
+
+namespace MyCrm
+{
+ public class Order : AggregateRoot, IHasCreationTime
+ {
+ public Guid CustomerId { get; set; }
+ public DateTime CreationTime { get; set; }
+
+ public ICollection Lines { get; set; } //Sub collection
+
+ public Order()
+ {
+ Lines = new Collection();
+ }
+ }
+
+ public class OrderLine : Entity
+ {
+ public Order Order { get; set; } //Navigation property
+ public Guid OrderId { get; set; }
+
+ public Guid ProductId { get; set; }
+ public int Count { get; set; }
+ public double UnitPrice { get; set; }
+ }
+}
+
+````
+
+And defined the database mapping as shown below:
+
+````csharp
+builder.Entity(b =>
+{
+ b.ToTable("Orders");
+ b.ConfigureByConvention();
+
+ //Define the relation
+ b.HasMany(x => x.Lines)
+ .WithOne(x => x.Order)
+ .HasForeignKey(x => x.OrderId)
+ .IsRequired();
+});
+
+builder.Entity(b =>
+{
+ b.ToTable("OrderLines");
+ b.ConfigureByConvention();
+});
+````
+
+When you query an `Order`, you may want to **include** all the `OrderLine`s in a single query or you may want to **load them later** on demand.
+
+> Actually these are not directly related to the ABP Framework. You can follow the [EF Core documentation](https://docs.microsoft.com/en-us/ef/core/querying/related-data/) to learn all the details. This section will cover some topics related to the ABP Framework.
+
+### Eager Loading / Load With Details
+
+You have different options when you want to load the related entities while querying an entity.
+
+#### Repository.WithDetails
+
+`IRepository.WithDetails(...)` can be used to include one relation collection/property to the query.
+
+**Example: Get an order with lines**
+
+````csharp
+using System;
+using System.Linq;
+using System.Threading.Tasks;
+using Volo.Abp.Domain.Repositories;
+using Volo.Abp.Domain.Services;
+
+namespace MyCrm
+{
+ public class OrderManager : DomainService
+ {
+ private readonly IRepository _orderRepository;
+
+ public OrderManager(IRepository orderRepository)
+ {
+ _orderRepository = orderRepository;
+ }
+
+ public async Task TestWithDetails(Guid id)
+ {
+ var query = _orderRepository
+ .WithDetails(x => x.Lines)
+ .Where(x => x.Id == id);
+
+ var order = await AsyncExecuter.FirstOrDefaultAsync(query);
+ }
+ }
+}
+````
+
+> `AsyncExecuter` is used to execute async LINQ extensions without depending on the EF Core. If you add EF Core NuGet package reference to your project, then you can directly use `await _orderRepository.WithDetails(x => x.Lines).FirstOrDefaultAsync()`. But, this time you depend on the EF Core in your domain layer. See the [repository document](Repositories.md) to learn more.
+
+**Example: Get a list of orders with their lines**
+
+````csharp
+public async Task TestWithDetails()
+{
+ var query = _orderRepository
+ .WithDetails(x => x.Lines);
+
+ var orders = await AsyncExecuter.ToListAsync(query);
+}
+````
+
+> `WithDetails` method can get more than one expression parameter if you need to include more than one navigation property or collection.
+
+#### DefaultWithDetailsFunc
+
+If you don't pass any expression to the `WithDetails` method, then it includes all the details using the `DefaultWithDetailsFunc` option you provide.
+
+You can configure `DefaultWithDetailsFunc` for an entity in the `ConfigureServices` method of your [module](Module-Development-Basics.md) in your `EntityFrameworkCore` project.
+
+**Example: Include `Lines` while querying an `Order`**
+
+````csharp
+Configure(options =>
+{
+ options.Entity(orderOptions =>
+ {
+ orderOptions.DefaultWithDetailsFunc = query => query.Include(o => o.Lines);
+ });
+});
+````
+
+> You can fully use the EF Core API here since this is located in the EF Core integration project.
+
+Then you can use the `WithDetails` without any parameter:
+
+````csharp
+public async Task TestWithDetails()
+{
+ var query = _orderRepository.WithDetails();
+ var orders = await AsyncExecuter.ToListAsync(query);
+}
+````
+
+`WithDetails()` executes the expression you've setup as the `DefaultWithDetailsFunc`.
+
+#### Repository Get/Find Methods
+
+Some of the standard [Repository](Repositories.md) methods have optional `includeDetails` parameters;
+
+* `GetAsync` and `FindAsync` gets `includeDetails` with default value is `true`.
+* `GetListAsync` and `GetPagedListAsync` gets `includeDetails` with default value is `false`.
+
+That means, the methods return a **single entity includes details** by default while list returning methods don't include details by default. You can explicitly pass `includeDetails` to change the behavior.
+
+> These methods use the `DefaultWithDetailsFunc` option that is explained above.
+
+**Example: Get an order with details**
+
+````csharp
+public async Task TestWithDetails(Guid id)
+{
+ var order = await _orderRepository.GetAsync(id);
+}
+````
+
+**Example: Get an order without details**
+
+````csharp
+public async Task TestWithoutDetails(Guid id)
+{
+ var order = await _orderRepository.GetAsync(id, includeDetails: false);
+}
+````
+
+**Example: Get list of entities with details**
+
+````csharp
+public async Task TestWithDetails()
+{
+ var orders = await _orderRepository.GetListAsync(includeDetails: true);
+}
+````
+
+#### Alternatives
+
+The repository patters tries to encapsulate the EF Core, so your options are limited. If you need an advanced scenario, you can follow one of the options;
+
+* Create a custom repository method and use the complete EF Core API.
+* Reference to the `Volo.Abp.EntityFrameworkCore` package from your project. In this way, you can directly use `Include` and `ThenInclude` in your code.
+
+See also [eager loading document](https://docs.microsoft.com/en-us/ef/core/querying/related-data/eager) of the EF Core.
+
+### Explicit / Lazy Loading
+
+If you don't include relations while querying an entity and later need to access to a navigation property or collection, you have different options.
+
+#### EnsurePropertyLoadedAsync / EnsureCollectionLoadedAsync
+
+Repositories provide `EnsurePropertyLoadedAsync` and `EnsureCollectionLoadedAsync` extension methods to **explicitly load** a navigation property or sub collection.
+
+**Example: Load Lines of an Order when needed**
+
+````csharp
+public async Task TestWithDetails(Guid id)
+{
+ var order = await _orderRepository.GetAsync(id, includeDetails: false);
+ //order.Lines is empty on this stage
+
+ await _orderRepository.EnsureCollectionLoadedAsync(order, x => x.Lines);
+ //order.Lines is filled now
+}
+````
+
+`EnsurePropertyLoadedAsync` and `EnsureCollectionLoadedAsync` methods do nothing if the property or collection was already loaded. So, calling multiple times has no problem.
+
+See also [explicit loading document](https://docs.microsoft.com/en-us/ef/core/querying/related-data/explicit) of the EF Core.
+
+#### Lazy Loading with Proxies
+
+Explicit loading may not be possible in some cases, especially when you don't have a reference to the `Repository` or `DbContext`. Lazy Loading is a feature of the EF Core that loads the related properties / collections when you first access to it.
+
+To enable lazy loading;
+
+1. Install the [Microsoft.EntityFrameworkCore.Proxies](https://www.nuget.org/packages/Microsoft.EntityFrameworkCore.Proxies/) package into your project (typically to the EF Core integration project)
+2. Configure `UseLazyLoadingProxies` for your `DbContext` (in the `ConfigureServices` method of your module in your EF Core project). Example:
+
+````csharp
+Configure(options =>
+{
+ options.PreConfigure(opts =>
+ {
+ opts.DbContextOptions.UseLazyLoadingProxies(); //Enable lazy loading
+ });
+
+ options.UseSqlServer();
+});
+````
+
+3. Make your navigation properties and collections `virtual`. Examples:
+
+````csharp
+public virtual ICollection Lines { get; set; } //virtual collection
+public virtual Order Order { get; set; } //virtual navigation property
+````
+
+Once you enable lazy loading and arrange your entities, you can freely access to the navigation properties and collections:
+
+````csharp
+public async Task TestWithDetails(Guid id)
+{
+ var order = await _orderRepository.GetAsync(id);
+ //order.Lines is empty on this stage
+
+ var lines = order.Lines;
+ //order.Lines is filled (lazy loaded)
+}
+````
+
+Whenever you access to a property/collection, EF Core automatically performs an additional query to load the property/collection from the database.
+
+> Lazy loading should be carefully used since it may cause performance problems in some specific cases.
+
+See also [lazy loading document](https://docs.microsoft.com/en-us/ef/core/querying/related-data/lazy) of the EF Core.
+
## Access to the EF Core API
In most cases, you want to hide EF Core APIs behind a repository (this is the main purpose of the repository pattern). However, if you want to access the `DbContext` instance over the repository, you can use `GetDbContext()` or `GetDbSet()` extension methods. Example:
@@ -296,7 +569,7 @@ public class BookService
* `GetDbContext` returns a `DbContext` reference instead of `BookStoreDbContext`. You can cast it, however in most cases you don't need it.
-> Important: You must reference to the `Volo.Abp.EntityFrameworkCore` package from the project you want to access to the DbContext. This breaks encapsulation, but this is what you want in that case.
+> Important: You must reference to the `Volo.Abp.EntityFrameworkCore` package from the project you want to access to the `DbContext`. This breaks encapsulation, but this is what you want in that case.
## Extra Properties & Object Extension Manager
@@ -365,7 +638,7 @@ public class MyRepositoryBase
: EfCoreRepository
where TEntity : class, IEntity
{
- public MyRepositoryBase(IDbContextProvider dbContextProvider)
+ public MyRepositoryBase(IDbContextProvider dbContextProvider)
: base(dbContextProvider)
{
}
@@ -395,7 +668,7 @@ context.Services.AddAbpDbContext(options =>
typeof(MyRepositoryBase<,>),
typeof(MyRepositoryBase<>)
);
-
+
//...
});
```
@@ -446,6 +719,22 @@ context.Services.AddAbpDbContext(options =>
In this example, `OtherDbContext` implements `IBookStoreDbContext`. This feature allows you to have multiple DbContext (one per module) on development, but single DbContext (implements all interfaces of all DbContexts) on runtime.
+### Split Queries
+
+ABP enables [split queries](https://docs.microsoft.com/en-us/ef/core/querying/single-split-queries) globally by default for better performance. You can change it as needed.
+
+**Example**
+
+````csharp
+Configure(options =>
+{
+ options.UseSqlServer(optionsBuilder =>
+ {
+ optionsBuilder.UseQuerySplittingBehavior(QuerySplittingBehavior.SingleQuery);
+ });
+});
+````
+
## See Also
-* [Entities](Entities.md)
\ No newline at end of file
+* [Entities](Entities.md)
diff --git a/docs/en/Exception-Handling.md b/docs/en/Exception-Handling.md
index 59a2c6bd0e..04cc6fc695 100644
--- a/docs/en/Exception-Handling.md
+++ b/docs/en/Exception-Handling.md
@@ -85,7 +85,7 @@ Error **details** in an optional field of the JSON error message. Thrown `Except
### Logging
-Caught exceptions are automatically logged.
+Caught exceptions are automatically logged.
#### Log Level
@@ -300,7 +300,7 @@ In this case, create a class derived from the `ExceptionSubscriber` class in you
````csharp
public class MyExceptionSubscriber : ExceptionSubscriber
{
- public override async Task HandleAsync(ExceptionNotificationContext context)
+ public async override Task HandleAsync(ExceptionNotificationContext context)
{
//TODO...
}
diff --git a/docs/en/Getting-Started-AspNetCore-Application.md b/docs/en/Getting-Started-AspNetCore-Application.md
index 9698b1dfb0..9202ed29f1 100644
--- a/docs/en/Getting-Started-AspNetCore-Application.md
+++ b/docs/en/Getting-Started-AspNetCore-Application.md
@@ -4,7 +4,7 @@ This tutorial explains how to start ABP from scratch with minimal dependencies.
## Create A New Project
-1. Create a new AspNet Core Web Application from Visual Studio 2019 (16.4.0+):
+1. Create a new AspNet Core Web Application from Visual Studio 2019 (16.8.0+):

diff --git a/docs/en/Getting-Started.md b/docs/en/Getting-Started.md
index 1e473b266f..6bfacc9a53 100644
--- a/docs/en/Getting-Started.md
+++ b/docs/en/Getting-Started.md
@@ -22,15 +22,14 @@ First things first! Let's setup your development environment before creating the
The following tools should be installed on your development machine:
-* [Visual Studio 2019](https://visualstudio.microsoft.com/vs/) for Windows / [Visual Studio for Mac](https://visualstudio.microsoft.com/vs/mac/). [1](#f-editor)
-* [.NET Core 3.1+](https://www.microsoft.com/net/download/dotnet-core/)
-
+* [Visual Studio 2019](https://visualstudio.microsoft.com/vs/) (v16.8+) for Windows / [Visual Studio for Mac](https://visualstudio.microsoft.com/vs/mac/). [1](#f-editor)
+* [.NET Core 5.0+](https://www.microsoft.com/net/download/dotnet-core/)
+{{ if UI != "Blazor" }}
* [Node v12 or v14](https://nodejs.org/)
* [Yarn v1.20+ (not v2)](https://classic.yarnpkg.com/en/docs/install) [2](#f-yarn) or npm v6+ (already installed with Node)
+{{ end }}
{{ if Tiered == "Yes" }}
-
* [Redis](https://redis.io/) (the startup solution uses the Redis as the [distributed cache](Caching.md)).
-
{{ end }}
1 _You can use another editor instead of Visual Studio as long as it supports .NET Core and ASP.NET Core._ [↩](#a-editor)
@@ -202,6 +201,8 @@ Right click to the `.DbMigrator` project and select **Set as StartUp Project**
{{ if Tiered == "Yes" }}
+> Tiered solutions use Redis as the distributed cache. Ensure that it is installed and running in your local computer. If you are using a remote Redis Server, set the configuration in the `appsettings.json` files of the projects below.
+
1. Ensure that the `.IdentityServer` project is the startup project. Run this application that will open a **login** page in your browser.
> Use Ctrl+F5 in Visual Studio (instead of F5) to run the application without debugging. If you don't have a debug purpose, this will be faster.
@@ -238,6 +239,8 @@ Ensure that the `.Web` project is the startup project. Run the application which
{{ if Tiered == "Yes" }}
+> Tiered solutions use Redis as the distributed cache. Ensure that it is installed and running in your local computer. If you are using a remote Redis Server, set the configuration in the `appsettings.json` files of the projects below.
+
Ensure that the `.IdentityServer` project is the startup project. Run the application which will open a **login** page in your browser.
> Use Ctrl+F5 in Visual Studio (instead of F5) to run the application without debugging. If you don't have a debug purpose, this will be faster.
diff --git a/docs/en/Integration-Tests.md b/docs/en/Integration-Tests.md
index f9f6830956..35d66171a6 100644
--- a/docs/en/Integration-Tests.md
+++ b/docs/en/Integration-Tests.md
@@ -1,3 +1 @@
-# Integration Tests
-
-TODO!
\ No newline at end of file
+This document has been [moved to here](Testing.md).
\ No newline at end of file
diff --git a/docs/en/Localization.md b/docs/en/Localization.md
index a33dff1779..089567422c 100644
--- a/docs/en/Localization.md
+++ b/docs/en/Localization.md
@@ -162,8 +162,8 @@ Getting the localized text is pretty standard.
Just inject the `IStringLocalizer` service and use it like shown below:
-````C#
-public class MyService
+````csharp
+public class MyService : ITransientDependency
{
private readonly IStringLocalizer _localizer;
diff --git a/docs/en/Migration-Guides/Abp-4_0-Angular.md b/docs/en/Migration-Guides/Abp-4_0-Angular.md
new file mode 100644
index 0000000000..6095a0f447
--- /dev/null
+++ b/docs/en/Migration-Guides/Abp-4_0-Angular.md
@@ -0,0 +1,121 @@
+# Angular UI 3.3 to 4.0 Migration Guide
+
+## Angular v11
+
+The new ABP Angular UI is based on Angular v11 and TypeScript v4. The difference between v10 and v11 is non-breaking so you do not have to update right away but it is recommended. Nevertheless, ABP modules will keep working with Angular v10. Therefore, if your project is Angular v10, you do not need to update to Angular 11. The update is usually very easy though.
+
+You can read more about Angular v11 [here](https://blog.angular.io/version-11-of-angular-now-available-74721b7952f7)
+
+## **Breaking Changes**
+
+### **Localization**
+
+Prior to ABP 4.x, we'd handled what locale files of Angular should be created to load them lazily. However, this made it impossible to add new locale files (to be lazily loaded) for our users. With ABP 4.x, we enabled an option to pass a function to `CoreModule`.
+
+The quickest solution is as follows:
+
+```typescript
+// app.module.ts
+
+import { registerLocale } from '@abp/ng.core/locale';
+// or
+// import { registerLocale } from '@volo/abp.ng.language-management/locale';
+// if you have commercial license
+
+
+@NgModule({
+imports: [
+ // ...
+ CoreModule.forRoot({
+ // ...other options,
+ registerLocaleFn: registerLocale()
+ }),
+ //...
+]
+export class AppModule {}
+```
+
+You can find the related issue [here](https://github.com/abpframework/abp/issues/6066)
+Also, please refer to [the docs](https://docs.abp.io/en/abp/latest/UI/Angular/Localization#registering-a-new-locale) for more information.
+
+### **Removed the Angular Account Module Public UI**
+
+With ABP 4.x, we have retired `@abp/ng.account`, it is no longer a part of our framework. There won't be any newer versions of this package as well. Therefore, you can delete anything related to this package.
+There should be a config in `app-routing.module` for path `account` and `AccountConfigModule` import in `app.module`
+
+However, if you are using the commercial version of this package, a.k.a `@volo/abp.ng.account`, this package will continue to exist because it contains `AccountAdminModule` which is still being maintained and developed. You only need to delete the route config from `app-routing.module`
+
+You can find the related issue [here](https://github.com/abpframework/abp/issues/5652)
+
+Angular UI is using the Authorization Code Flow to authenticate since version 3.1.0 by default. Starting from version 4.0, this is becoming the only option, because it is the recommended way of authenticating SPAs.
+
+If you haven't done it yet, see [this post](https://blog.abp.io/abp/ABP-Framework-v3.1-RC-Has-Been-Released) to change the authentication of your application.
+
+### State Management
+
+In the ABP Angular UI, we've been using `NGXS` for state management. However, we've decided that the Angular UI should be agnostic with regard to state management. Our users should be able to handle the state in any way they prefer. They should be able to use any library other than `NGXS` or no library at all. That's why we have created our internal store in version 3.2. It is a simple utility class that employs `BehaviorSubject` internally.
+
+You can examine it [here](https://github.com/abpframework/abp/blob/dev/npm/ng-packs/packages/core/src/lib/utils/internal-store-utils.ts)
+
+With version 4.0, we will keep utilizing our `InternalStore` instead of `@ngxs/store` in our services and move away from `@ngxs/store`. We plan to remove any dependency of `NGXS` by version 5.0.
+
+With this in mind, we've already deprecated some services and implemented some breaking changes.
+
+#### Removed the `SessionState`
+
+Use `SessionStateService` instead of the `SessionState`. See [this issue](https://github.com/abpframework/abp/issues/5606) for details.
+
+#### Deprecated the `ConfigState`
+
+`ConfigState` is now deprecated and should not be used.
+
+`ConfigState` reference removed from `CoreModule`. If you want to use the `ConfigState` (not recommended), you should pass the state to `NgxsModule` as shown below:
+
+```typescript
+//app.module.ts
+
+import { ConfigState } from '@abp/ng.core';
+
+// ...
+
+imports: [
+ NgxsModule.forRoot([ConfigState]),
+// ...
+```
+
+Moving away from the global store, we create small services with a single responsibility. There are two new services available in version 4.0 which are `EnvironmentService` and `PermissionService`.
+
+See [the related issue](https://github.com/abpframework/abp/issues/6154)
+
+Please refer to the following docs for detail information and examples
+- [`ConfigStateService`](../UI/Angular/Config-State-Service)
+- [`EnvironmentService`](../UI/Angular/Environment#EnvironmentService)
+- [`PermissionService`](../UI/Angular/Permission-Management#)
+
+### Deprecated Interfaces
+
+ Some interfaces have long been marked as deprecated and now they are removed.
+
+- Removed replaceable components state.
+- Removed legacy identity types and service.
+- Removed legacy tenant management types and service.
+- Removed legacy feature management types and services.
+- Removed legacy permission management types and service.
+
+### Deprecated commercial interfaces
+- Removed legacy audit logging types and services.
+- Removed legacy identity types and services.
+- Removed legacy language management types and services.
+- Removed legacy saas types and services.
+
+### Identity Server [COMMERCIAL]
+
+With the new version of Identity Server, there happened some breaking changes in the backend (also in the database). We've implemented those in the Angular UI.
+If you are just using the package `@volo/abp.ng.identity-server` as is, you will not need to do anything.
+However, there are a couple of breaking changes we need to mention.
+
+- As we have stated above, we want to remove the dependency of `NGXS`. Thus, we have deleted all of the actions defined in `identity-server.actions`. Those actions are not needed anymore and the state is managed locally. With the actions gone, `IdentityServerStateService` became unused and got deleted as well.
+
+- `ApiScope` is also available as a new entity (It was part of `ApiResource` before). It provides tokens for entity prop, entity actions, toolbar, edit and create form contributors like the existing ones which are `Client`, `IdentityResource` and `ApiResource`
+
+- There were some deprecated interfaces within the `IdentityServer` namespace. Those are no longer being used, instead, their replacements were generated by `ABP Cli` using the `generate-proxy` command.
diff --git a/docs/en/Migration-Guides/Abp-4_0-Blazor.md b/docs/en/Migration-Guides/Abp-4_0-Blazor.md
new file mode 100644
index 0000000000..d381ffceb6
--- /dev/null
+++ b/docs/en/Migration-Guides/Abp-4_0-Blazor.md
@@ -0,0 +1,88 @@
+# Blazor UI 3.3 to 4.0 Migration Guide
+
+## Startup Template Changes
+
+These changes are required to manually applied in your own solution. It would be easier if you create a new solution based on 4.0 with the same name of your current solution then compare the files.
+
+### Csproj File / Dependencies
+
+* Add `true` to the `PropertyGroup` section of your project (`.csproj`) file.
+* Update the `Blazorise.*` packages to the latest version (to the latest RC for the ABP 4.0 preview).
+
+### wwwroot/index.html
+
+There are some changes made in the index.html file;
+
+* Removed JQuery & Bootstrap JavaScript dependencies
+* Replaced Bootstrap and FontAwesome imports with local files instead of CDN usages.
+* Re-arranged some ABP CSS file locations.
+* Introduced the `abp bundle` CLI command to manage global Style/Script file imports.
+
+Follow the steps below to apply the changes;
+
+1. Add the bundle contributor class into your project (it will be slightly different based on your solution namespaces):
+
+````csharp
+using Volo.Abp.Bundling;
+
+namespace MyCompanyName.MyProjectName.Blazor
+{
+ public class MyProjectNameBundleContributer : IBundleContributer
+ {
+ public void AddScripts(BundleContext context)
+ {
+ }
+
+ public void AddStyles(BundleContext context)
+ {
+ context.Add("main.css");
+ }
+ }
+}
+````
+
+If you are using another global style/script files, add them here.
+
+2. Remove all the `` elements and replace with the following comment tags:
+
+````html
+
+
+````
+
+3. Remove all the `` elements and replace with the following comment tags:
+
+````html
+
+
+````
+
+4. Execute the following command in a terminal in the root folder of the Blazor project (`.csproj`) file (ensure that you're using the ABP CLI version 4.0):
+
+````bash
+abp bundle
+````
+
+This will fill in the `Styles` and `Scripts` tags based on the dependencies.
+
+5. You can clean the `blazor-error-ui` related sections from your `main.css` file since they are not needed anymore.
+
+### The Root Element
+
+This change is optional but recommended.
+
+* Change `...` to `
...
` in the `wwwroot/index.html`.
+* Change `builder.RootComponents.Add("app");` to `builder.RootComponents.Add("#ApplicationContainer");` in the *YourProjectBlazorModule.cs*.
+
+## AbpCrudPageBase Changes
+
+If you've derived your pages from the `AbpCrudPageBase` class, then you may need to apply the following changes;
+
+- `OpenEditModalAsync` method gets `EntityDto` instead of id (`Guid`) parameter. Pass `context` instead of `context.Id`.
+- `DeleteEntityAsync` method doesn't display confirmation dialog anymore. You can use the new `EntityActions` component in Data Grids to show confirmation messages. You can also inject `IUiMessageService` to your page or component and call the `ConfirmAsync` explicitly.
+- Added `GetListInput` as a base property that is used to filter while getting the entities from the server.
+
+## Others
+
+- Refactored namespaces for some Blazor components ([#6015](https://github.com/abpframework/abp/issues/6015)).
+- Removed Async Suffix from IUiMessageService methods ([#6123](https://github.com/abpframework/abp/pull/6123)).
\ No newline at end of file
diff --git a/docs/en/Migration-Guides/Abp-4_0-MVC-Razor-Pages.md b/docs/en/Migration-Guides/Abp-4_0-MVC-Razor-Pages.md
new file mode 100644
index 0000000000..d2fa97ec5e
--- /dev/null
+++ b/docs/en/Migration-Guides/Abp-4_0-MVC-Razor-Pages.md
@@ -0,0 +1,6 @@
+# MVC / Razor Pages UI 3.3 to 4.0 Migration Guide
+
+## Use IBrandingProvider in the Volo.Abp.UI Package
+
+This will be a breaking change for MVC UI, but very easy to fix. `IBrandingProvider` is being moved from `Volo.Abp.AspNetCore.Mvc.UI.Theme.Shared.Components` to `Volo.Abp.Ui.Branding` namespace. So, just update the namespace imports.
+
diff --git a/docs/en/Migration-Guides/Abp-4_0.md b/docs/en/Migration-Guides/Abp-4_0.md
index 0d193d476b..8bc4924000 100644
--- a/docs/en/Migration-Guides/Abp-4_0.md
+++ b/docs/en/Migration-Guides/Abp-4_0.md
@@ -1,96 +1,113 @@
# ABP Framework 3.3 to 4.0 Migration Guide
-## Auto API Controller Route Changes
+This document introduces the breaking changes done in the ABP Framework 4.0 and explains how to fix your 3.x based solutions while upgrading to the ABP Framework 4.0.
-The route calculation for the [Auto API Controllers](https://docs.abp.io/en/abp/latest/API/Auto-API-Controllers) is changing with the ABP Framework version 4.0 ([#5325](https://github.com/abpframework/abp/issues/5325)). Previously, **camelCase** route paths were being used. Beginning from the version 4.0, it uses **kebab-case** route paths where it is possible.
+> See [the blog post](https://blog.abp.io/abp/ABP.IO-Platform-v4.0-RC-Has-Been-Released-based-on-.NET-5.0) to learn what's new with the ABP Framework 4.0. This document only focuses on the breaking changes.
-**A typical auto API before v4.0**
+## Overall
-
+Here, the overall list of the changes;
-**camelCase route parts become kebab-case with 4.0**
+* Upgraded to the .NET 5.0 [(#6118](https://github.com/abpframework/abp/issues/6118)).
+* Moved from Newtonsoft.Json to System.Text.Json [(#1198](https://github.com/abpframework/abp/issues/1198)).
+* Upgraded to the Identity Server 4.1.1 ([#4461](https://github.com/abpframework/abp/issues/4461)).
+* Switched to `kebab-case` for conventional URLs for the auto API controller routes ([#5325](https://github.com/abpframework/abp/issues/5325)).
+* Removed Retry for the Dynamic HTTP Client Proxies ([#6090](https://github.com/abpframework/abp/issues/6090)).
+* Creation audit properties of the entities made read-only ([#6020](https://github.com/abpframework/abp/issues/6020)).
+* Changed type of the IHasExtraProperties.ExtraProperties ([#3751](https://github.com/abpframework/abp/issues/3751)).
+* Use IBrandingProvider in the Volo.Abp.UI package and remove the one in the Volo.Abp.AspNetCore.Mvc.UI.Theme.Shared ([#5375](https://github.com/abpframework/abp/issues/5375)).
+* Removed the Angular Account Module Public UI (login, register... pages) since they are not being used in the default (authorization code) flow ([#5652](https://github.com/abpframework/abp/issues/5652)).
+* Removed the SessionState in the @abp/ng.core package ([#5606](https://github.com/abpframework/abp/issues/5606)).
+* Made some API revisions & startup template changes for the Blazor UI.
-
+## Upgraded to .NET 5.0
+
+ABP Framework has been moved to .NET 5.0. So, if you want to upgrade to the ABP Framework 4.0, you also need to upgrade to .NET 5.0.
-If it is hard to change it for your application, you can continue to use the version 3.x route strategy, by following one of the approaches;
+See the [Migrate from ASP.NET Core 3.1 to 5.0](https://docs.microsoft.com/en-us/aspnet/core/migration/31-to-50) document to learn how to upgrade your solution to .NET 5.0.
-* Set `UseV3UrlStyle` to `true` in the options of the `options.ConventionalControllers.Create(...)` method. Example:
+## Moved to System.Text.Json
+
+ABP Framework 4.0 uses the System.Text.Json by default as the JSON serialization library. It, actually, using a hybrid approach: Continues to use the Newtonsoft.Json when it needs to use features not supported by the System.Text.Json.
+
+### Unsupported Types
+
+If you want to use the Newtonsoft.Json to serialize/deserialize for some specific types, you can configure the `AbpSystemTextJsonSerializerOptions` in your module's `ConfigureServices` method.
+
+**Example: Use Newtonsoft.Json for `MySpecialClass`**
````csharp
-options.ConventionalControllers
- .Create(typeof(BookStoreApplicationModule).Assembly, opts =>
- {
- opts.UseV3UrlStyle = true;
- });
+Configure(options =>
+{
+ options.UnsupportedTypes.AddIfNotContains(typeof(MySpecialClass));
+});
````
-This approach effects only the controllers for the `BookStoreApplicationModule`.
+### Always Use the Newtonsoft.Json
-* Set `UseV3UrlStyle` to `true` for the `AbpConventionalControllerOptions` to set it globally. Example:
+If you want to continue to use the Newtonsoft.Json library for all the types, you can set `UseHybridSerializer` to false in the `PreConfigureServices` method of your module class:
-```csharp
-Configure(options =>
+````csharp
+PreConfigure(options =>
{
- options.UseV3UrlStyle = true;
+ options.UseHybridSerializer = false;
});
-```
-
-Setting it globally effects all the modules in a modular application.
+````
-## Identity Server Changes
+## Upgraded to Identity Server 4.1.1
-ABP Framework upgrades the [IdentityServer4](https://www.nuget.org/packages/IdentityServer4) library from 3.x to 4.x with the ABP Framework version 4.0. IdentityServer 4.x has a lot of changes, some of them are **breaking changes in the data structure**.
+ABP Framework upgrades the [IdentityServer4](https://www.nuget.org/packages/IdentityServer4) library from 3.x to 4.1.1 with the ABP Framework version 4.0. IdentityServer 4.x has a lot of changes. Some of them are **breaking changes in the data structure**.
### Entity Changes
-Entity changed don't directly affect your application, however it is good to know.
+Entity changes don't directly affect your application; however, it is good to know.
#### ApiScope
-As the **most important breaking change**, Identity Server 4.x places the `ApiScope` as an independent aggregate root. Previously it was a part of the to `ApiResource` aggregate. This requires manual operation. See the *Database Changes* section.
+As the **most critical breaking change**; Identity Server 4.x defines the `ApiScope` as an independent aggregate root. Previously, it was the child entity of the `ApiResource`. This change requires manual operation. See the _Database Changes_ section.
Also, added `Enabled(string)` and `Description(bool,true)` properties.
#### ApiResource
-* Added `AllowedAccessTokenSigningAlgorithms (string)` and `ShowInDiscoveryDocument(bool, default: true)` properties
+- Added `AllowedAccessTokenSigningAlgorithms (string)` and `ShowInDiscoveryDocument(bool, default: true)` properties
#### Client
-* Added `RequireRequestObject (bool)` and `AllowedIdentityTokenSigningAlgorithms (string)` properties.
-* Changed default value of `RequireConsent` from `true` to `false`.
-* Changed default value of `RequirePkce` from `false` to `true`.
+- Added `RequireRequestObject ` and `AllowedIdentityTokenSigningAlgorithms ` properties.
+- Changed the default value of `RequireConsent` from `true` to `false`.
+- Changed the default value of `RequirePkce` from `false` to `true`.
#### DeviceFlowCodes
-* Added `SessionId (string)` and `Description (string)` properties.
+- Added `SessionId ` and `Description ` properties.
#### PersistedGrant
-* Added `SessionId (string)` and `Description(string)` and `ConsumedTime (DateTime?)` properties
+- Added `SessionId `, `Description ` and `ConsumedTime ` properties
### Database Changes
> Attention: **Please backup your database** before the migration!
-**If you are upgrading from 3.x, then there are some change should be done in your database.**
+**If you are upgrading from 3.x, then there are some steps should be done in your database.**
#### Database Schema Migration
-If you are using **Entity Framework Core**, you need to add a new database migration, using the `Add-Migration` command, and apply changes to the database. Please **review the migration** script and read the sections below to understand if it affects your existing data. Otherwise, you may **loose some of your configuration**, which may not be easy to remember and re-configure.
+If you are using **Entity Framework Core**, you need to add a new database migration, using the `Add-Migration` command, and apply changes to the database. Please **review the migration** script and read the sections below to understand if it affects your existing data. Otherwise, you may **lose some of your configuration**, which may not be easy to remember and re-configure.
#### Seed Code
-If you haven't customize the `IdentityServerDataSeedContributor` and haven't customized the initial data inside the `IdentityServer*` tables;
+If you haven't customized the `IdentityServerDataSeedContributor` and haven't customized the initial data inside the `IdentityServer*` tables;
1. Update `IdentityServerDataSeedContributor` class by comparing to [the latest code](https://github.com/abpframework/abp/blob/dev/templates/app/aspnet-core/src/MyCompanyName.MyProjectName.Domain/IdentityServer/IdentityServerDataSeedContributor.cs). You probably only need to add the `CreateApiScopesAsync` method and the code related to it.
-2. Then you can simply clear all the **table data** in these tables then execute the `DbMigrator` application again to fill it with the new configuration.
+2. Then you can simply clear all the **data** in these tables then execute the `DbMigrator` application to fill it with the new configuration.
#### Migrating the Configuration Data
-If you've customize your IdentityServer configuration in the database or in the seed data, you should understand the changes and upgrade your code/data accordingly. Especially, the following changes will affect your application:
+If you've customized your IdentityServer configuration in the database or in the seed data, you should understand the changes and upgrade your code/data accordingly. Especially, the following changes will affect your application:
-- `IdentityServerApiScopes` table's `Enabled` field is dropped and re-created. So, you need to manually enable the api scopes again.
+- `IdentityServerApiScopes` table's `Enabled` field is dropped and re-created. So, you need to enable the API scopes again manually.
- `IdentityServerApiResourceScopes` table is dropped and recreated. So, you need to backup and move your current data to the new table.
- `IdentityServerIdentityResourceClaims` table is dropped and recreated. So, you need to backup and move your current data to the new table.
@@ -98,11 +115,11 @@ You may need to perform additional steps based on how much you made custom confi
### Other IdentityServer Changes
-IdentityServer has removed the [public origin option](https://github.com/IdentityServer/IdentityServer4/pull/4335). It was resolving http/https conversion problems, but they decided to leave this to the developer. This is especially needed if you use a reverse proxy where your external protocol is HTTPS but internal protocol is HTTP.
+IdentityServer has removed the [public origin option](https://github.com/IdentityServer/IdentityServer4/pull/4335). It was resolving HTTP/HTTPS conversion issues, but they decided to leave this to the developer. This is especially needed if you use a reverse proxy where your external protocol is HTTPS but internal protocol is HTTP.
-One simple solution is to add such a middleware into your ASP.NET Core pipeline, at the beginning.
+One simple solution is to add such a middleware at the begingning of your ASP.NET Core pipeline.
-```
+```csharp
app.Use((httpContext, next) =>
{
httpContext.Request.Scheme = "https";
@@ -113,5 +130,105 @@ app.Use((httpContext, next) =>
> This sample is obtained from the [ASP.NET Core documentation](https://docs.microsoft.com/en-us/aspnet/core/host-and-deploy/proxy-load-balancer#scenarios-and-use-cases). You can use it if you always use HTTPS in all environments.
### Related Resources
-* https://leastprivilege.com/2020/06/19/announcing-identityserver4-v4-0/
-* https://github.com/IdentityServer/IdentityServer4/issues/4592
+
+- https://leastprivilege.com/2020/06/19/announcing-identityserver4-v4-0/
+- https://github.com/IdentityServer/IdentityServer4/issues/4592
+
+## Auto API Controller Route Changes
+
+The route calculation for the [Auto API Controllers](https://docs.abp.io/en/abp/latest/API/Auto-API-Controllers) is changing with the ABP Framework version 4.0 ([#5325](https://github.com/abpframework/abp/issues/5325)). Before v4.0 the route paths were **camelCase**. After version 4.0, it's changed to **kebab-case** route paths where it is possible.
+
+**A typical auto API before v4.0**
+
+
+
+**camelCase route parts become kebab-case with 4.0**
+
+
+
+### How to Fix?
+
+You may not take any action for the MVC & Blazor UI projects.
+
+For the Angular UI, this change may effect your client UI. If you have used the [ABP CLI Service Proxy Generation](../UI/Angular/Service-Proxies.md), you can run the server side and re-generate the service proxies. If you haven't used this tool, you should manually update the related URLs in your application.
+
+If there are other type of clients (e.g. 3rd-party companies) using your APIs, they also need to update the URLs.
+
+### Use the v3.x style URLs
+
+If it is hard to change it in your application, you can still to use the version 3.x route strategy, by following one of the approaches;
+
+- Set `UseV3UrlStyle` to `true` in the options of the `options.ConventionalControllers.Create(...)` method. Example:
+
+```csharp
+options.ConventionalControllers
+ .Create(typeof(BookStoreApplicationModule).Assembly, opts =>
+ {
+ opts.UseV3UrlStyle = true;
+ });
+```
+
+This approach affects only the controllers for the `BookStoreApplicationModule`.
+
+- Set `UseV3UrlStyle` to `true` for the `AbpConventionalControllerOptions` to set it globally. Example:
+
+```csharp
+Configure(options =>
+{
+ options.UseV3UrlStyle = true;
+});
+```
+
+Setting it globally affects all the modules in a modular application.
+
+## Removed Retry for the Dynamic HTTP Client Proxies
+
+[Dynamic C# HTTP Client Proxies](../API/Dynamic-CSharp-API-Clients.md) were trying up to 3 times if a request fails using the [Polly](https://github.com/App-vNext/Polly) library. Starting from the version 4.0, this logic has been removed. If you need it, you should configure it in your own application, by configuring the `AbpHttpClientBuilderOptions` in the `PreConfigureServices` method of your module.
+
+**Example: Retry 3 times on failure by incremental waiting between tries**
+
+````csharp
+public override void PreConfigureServices(ServiceConfigurationContext context)
+{
+ PreConfigure(options =>
+ {
+ options.ProxyClientBuildActions.Add((remoteServiceName, clientBuilder) =>
+ {
+ clientBuilder.AddTransientHttpErrorPolicy(
+ policyBuilder => policyBuilder
+ .WaitAndRetryAsync(3, i => TimeSpan.FromSeconds(Math.Pow(2, i)))
+ );
+ });
+ });
+}
+````
+
+This example uses the Microsoft.Extensions.Http.Polly NuGet package.
+
+If you create a new solution, you can find the same configuration in the `.HttpApi.Client.ConsoleTestApp` project's module class, as an example.
+
+## Creation Audit Properties Made Read-Only
+
+Removed setters from the `IHasCreationTime.CreationTime`, ` IMustHaveCreator.CreatorId` and `IMayHaveCreator.CreatorId` properties to accidently set the creation properties while updating an existing entity.
+
+Since the ABP Framework automatically sets these properties, you normally don't need to directly set them. If you want to set them, as a best practice, it is suggested to make it in the constructor to not provide a way to change it later.
+
+These properties implemented with `protected set` in the `Entity` and `AggregateRoot` base classes. That means you can still set in a derived class, if you need it. Alternatively, you can use reflection to set them (Or use `ObjectHelper.TrySetProperty` which internally uses reflection) out of the class if you have to do.
+
+## Changed type of the IHasExtraProperties.ExtraProperties
+
+`IHasExtraProperties.ExtraProperties` was a regular `Dictionary`. With the version 4.0, it is replaced with `ExtraPropertyDictionary` class which inherits the `Dictionary`.
+
+Most of the applications don't be affected by this change. If you've directly implemented this interface, replace the standard dictionary the the `ExtraPropertyDictionary`.
+
+## ASP.NET Core MVC / Razor Pages UI
+
+See the [ASP.NET Core MVC / Razor Pages UI Migration Guide](Abp-4_0-MVC-Razor-Pages.md).
+
+## Angular UI
+
+See the [Angular UI Migration Guide](Abp-4_0-Angular.md).
+
+## Blazor UI
+
+See the [Blazor UI Migration Guide](Abp-4_0-Blazor.md).
\ No newline at end of file
diff --git a/docs/en/MongoDB.md b/docs/en/MongoDB.md
index 475eb94f1b..98769c13b8 100644
--- a/docs/en/MongoDB.md
+++ b/docs/en/MongoDB.md
@@ -40,7 +40,7 @@ public class MyDbContext : AbpMongoDbContext
protected override void CreateModel(IMongoModelBuilder modelBuilder)
{
base.CreateModel(modelBuilder);
-
+
//Customize the configuration for your collections.
}
}
@@ -62,7 +62,7 @@ So, most of times you don't need to explicitly configure registration for your e
protected override void CreateModel(IMongoModelBuilder modelBuilder)
{
base.CreateModel(modelBuilder);
-
+
modelBuilder.Entity(b =>
{
b.CollectionName = "MyQuestions"; //Sets the collection name
@@ -88,7 +88,7 @@ If you have multiple databases in your application, you can configure the connec
[ConnectionStringName("MySecondConnString")]
public class MyDbContext : AbpMongoDbContext
{
-
+
}
````
@@ -202,7 +202,7 @@ You generally want to derive from the `IRepository` to inherit standard reposito
Example implementation of the `IBookRepository` interface:
```csharp
-public class BookRepository :
+public class BookRepository :
MongoDbRepository,
IBookRepository
{
@@ -242,9 +242,9 @@ context.Services.AddMongoDbContext(options =>
This is especially important when you want to **override a base repository method** to customize it. For instance, you may want to override `DeleteAsync` method to delete an entity in a more efficient way:
```csharp
-public override async Task DeleteAsync(
- Guid id,
- bool autoSave = false,
+public async override Task DeleteAsync(
+ Guid id,
+ bool autoSave = false,
CancellationToken cancellationToken = default)
{
//TODO: Custom implementation of the delete method
@@ -381,4 +381,4 @@ context.Services.AddMongoDbContext(options =>
});
```
-In this example, `OtherMongoDbContext` implements `IBookStoreMongoDbContext`. This feature allows you to have multiple MongoDbContext (one per module) on development, but single MongoDbContext (implements all interfaces of all MongoDbContexts) on runtime.
\ No newline at end of file
+In this example, `OtherMongoDbContext` implements `IBookStoreMongoDbContext`. This feature allows you to have multiple MongoDbContext (one per module) on development, but single MongoDbContext (implements all interfaces of all MongoDbContexts) on runtime.
diff --git a/docs/en/Repositories.md b/docs/en/Repositories.md
index 0fc80984e7..5ff8372af8 100644
--- a/docs/en/Repositories.md
+++ b/docs/en/Repositories.md
@@ -79,6 +79,14 @@ If your entity does not have an Id primary key (it may have a composite primary
> `IRepository` has a few missing methods those normally works with the `Id` property of an entity. Because of the entity has no `Id` property in that case, these methods are not available. One example is the `Get` method that gets an id and returns the entity with given id. However, you can still use `IQueryable` features to query entities by standard LINQ methods.
+### Soft / Hard Delete
+
+`DeleteAsync` method of the repository doesn't delete the entity if the entity is a **soft-delete** entity (that implements `ISoftDelete`). Soft-delete entities are marked as "deleted" in the database. Data Filter system ensures that the soft deleted entities are not retrieved from database normally.
+
+If your entity is a soft-delete entity, you can use the `HardDeleteAsync` method to really delete the entity from database in case of you need it.
+
+See the [Data Filtering](Data-Filtering.md) documentation for more about soft-delete.
+
## Custom Repositories
Default generic repositories will be sufficient for most cases. However, you may need to create a custom repository class for your entity.
diff --git a/docs/en/Road-Map.md b/docs/en/Road-Map.md
index 8670ba080b..3a45ca1767 100644
--- a/docs/en/Road-Map.md
+++ b/docs/en/Road-Map.md
@@ -1,17 +1,12 @@
# ABP Framework Road Map
-You can always check the milestone planning and the prioritized backlog issues on [the GitHub repository](https://github.com/abpframework/abp/milestones) for a detailed road map.
-
-While we will **continue to add other exciting features**, we will work on the following major items in the **middle term**:
-
-* **Blazor UI** for the framework and all the pre-built modules (in progress).
-* **.NET 5.0**! As Microsoft has announced that the .NET 5.0 will be released in November 2020, we will prepare for this change before and move to the .NET 5.0 just after Microsoft releases it. We hope a smooth transition.
-
-Beside this middle term goals, there are many features in the [backlog](https://github.com/abpframework/abp/milestone/2). Here, a list of some major items in the backlog;
+You can always check the milestone planning and the prioritized backlog issues on [the GitHub repository](https://github.com/abpframework/abp/milestones) for a detailed road map. Here, a list of some major items in the backlog;
* [#2882](https://github.com/abpframework/abp/issues/2882) / Providing a gRPC integration infrastructure (while it is [already possible](https://github.com/abpframework/abp-samples/tree/master/GrpcDemo) to create or consume gRPC endpoints for your application, we plan to create endpoints for the [standard application modules](https://docs.abp.io/en/abp/latest/Modules/Index))
* [#236](https://github.com/abpframework/abp/issues/236) Resource based authorization system
+* [#6132](https://github.com/abpframework/abp/issues/6132) A New Theme alternative to the Basic Theme
* [#1754](https://github.com/abpframework/abp/issues/1754) / Multi-lingual entities
+* [#497](https://github.com/abpframework/abp/issues/497) API Versioning system finalize & document
* [#633](https://github.com/abpframework/abp/issues/633) / Realtime notification system
* [#57](https://github.com/abpframework/abp/issues/57) / Built-in CQRS infrastructure
* [#336](https://github.com/abpframework/abp/issues/336) / Health Check abstraction
diff --git a/docs/en/Specifications.md b/docs/en/Specifications.md
index ea4f140e95..036baa749e 100644
--- a/docs/en/Specifications.md
+++ b/docs/en/Specifications.md
@@ -1,3 +1,255 @@
# Specifications
-TODO!
\ No newline at end of file
+Specification Pattern is used to define **named, reusable, combinable and testable filters** for entities and other business objects.
+
+> A Specification is a part of the Domain Layer.
+
+## Installation
+
+> This package is **already installed** when you use the startup templates. So, most of the times you don't need to manually install it.
+
+Install the [Volo.Abp.Specifications](https://abp.io/package-detail/Volo.Abp.Specifications) package to your project. You can use the [ABP CLI](CLI.md) *add-package* command in a command line terminal when the current folder is the root folder of your project (`.csproj`):
+
+````bash
+abp add-package Volo.Abp.Specifications
+````
+
+## Defining the Specifications
+
+Assume that you've a Customer entity as defined below:
+
+````csharp
+using System;
+using Volo.Abp.Domain.Entities;
+
+namespace MyProject
+{
+ public class Customer : AggregateRoot
+ {
+ public string Name { get; set; }
+
+ public byte Age { get; set; }
+
+ public long Balance { get; set; }
+
+ public string Location { get; set; }
+ }
+}
+````
+
+You can create a new Specification class derived from the `Specification`.
+
+**Example: A specification to select the customers with 18+ age:**
+
+````csharp
+using System;
+using System.Linq.Expressions;
+using Volo.Abp.Specifications;
+
+namespace MyProject
+{
+ public class Age18PlusCustomerSpecification : Specification
+ {
+ public override Expression> ToExpression()
+ {
+ return c => c.Age >= 18;
+ }
+ }
+}
+````
+
+You simply define a lambda [Expression](https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/operators/lambda-expressions) to define a specification.
+
+> Instead, you can directly implement the `ISpecification` interface, but the `Specification` base class much simplifies it.
+
+## Using the Specifications
+
+There are two common use cases of the specifications.
+
+### IsSatisfiedBy
+
+`IsSatisfiedBy` method can be used to check if a single object satisfies the specification.
+
+**Example: Throw exception if the customer doesn't satisfy the age specification**
+
+````csharp
+using System;
+using System.Threading.Tasks;
+using Volo.Abp.DependencyInjection;
+
+namespace MyProject
+{
+ public class CustomerService : ITransientDependency
+ {
+ public async Task BuyAlcohol(Customer customer)
+ {
+ if (!new Age18PlusCustomerSpecification().IsSatisfiedBy(customer))
+ {
+ throw new Exception(
+ "This customer doesn't satisfy the Age specification!"
+ );
+ }
+
+ //TODO...
+ }
+ }
+}
+````
+
+### ToExpression & Repositories
+
+`ToExpression()` method can be used to use the specification as Expression. In this way, you can use a specification to **filter entities while querying from the database**.
+
+````csharp
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Threading.Tasks;
+using Volo.Abp.DependencyInjection;
+using Volo.Abp.Domain.Repositories;
+using Volo.Abp.Domain.Services;
+
+namespace MyProject
+{
+ public class CustomerManager : DomainService, ITransientDependency
+ {
+ private readonly IRepository _customerRepository;
+
+ public CustomerManager(IRepository customerRepository)
+ {
+ _customerRepository = customerRepository;
+ }
+
+ public async Task> GetCustomersCanBuyAlcohol()
+ {
+ var query = _customerRepository.Where(
+ new Age18PlusCustomerSpecification().ToExpression()
+ );
+
+ return await AsyncExecuter.ToListAsync(query);
+ }
+ }
+}
+````
+
+> Specifications are correctly translated to SQL/Database queries and executed efficiently in the DBMS side. While it is not related to the Specifications, see the [Repositories](Repositories.md) document if you want to know more about the `AsyncExecuter`.
+
+Actually, using the `ToExpression()` method is not necessary since the specifications are automatically casted to Expressions. This would also work:
+
+````csharp
+var query = _customerRepository.Where(
+ new Age18PlusCustomerSpecification()
+);
+````
+
+## Composing the Specifications
+
+One powerful feature of the specifications is that they are composable with `And`, `Or`, `Not` and `AndNot` extension methods.
+
+Assume that you have another specification as defined below:
+
+```csharp
+using System;
+using System.Linq.Expressions;
+using Volo.Abp.Specifications;
+
+namespace MyProject
+{
+ public class PremiumCustomerSpecification : Specification
+ {
+ public override Expression> ToExpression()
+ {
+ return (customer) => (customer.Balance >= 100000);
+ }
+ }
+}
+```
+
+You can combine the `PremiumCustomerSpecification` with the `Age18PlusCustomerSpecification` to query the count of premium adult customers as shown below:
+
+````csharp
+using System;
+using System.Threading.Tasks;
+using Volo.Abp.DependencyInjection;
+using Volo.Abp.Domain.Repositories;
+using Volo.Abp.Domain.Services;
+using Volo.Abp.Specifications;
+
+namespace MyProject
+{
+ public class CustomerManager : DomainService, ITransientDependency
+ {
+ private readonly IRepository _customerRepository;
+
+ public CustomerManager(IRepository customerRepository)
+ {
+ _customerRepository = customerRepository;
+ }
+
+ public async Task GetAdultPremiumCustomerCountAsync()
+ {
+ return await _customerRepository.CountAsync(
+ new Age18PlusCustomerSpecification()
+ .And(new PremiumCustomerSpecification()).ToExpression()
+ );
+ }
+ }
+}
+````
+
+If you want to make this combination another reusable specification, you can create such a combination specification class deriving from the `AndSpecification`:
+
+````csharp
+using Volo.Abp.Specifications;
+
+namespace MyProject
+{
+ public class AdultPremiumCustomerSpecification : AndSpecification
+ {
+ public AdultPremiumCustomerSpecification()
+ : base(new Age18PlusCustomerSpecification(),
+ new PremiumCustomerSpecification())
+ {
+ }
+ }
+}
+````
+
+Now, you can re-write the `GetAdultPremiumCustomerCountAsync` method as shown below:
+
+````csharp
+public async Task GetAdultPremiumCustomerCountAsync()
+{
+ return await _customerRepository.CountAsync(
+ new AdultPremiumCustomerSpecification()
+ );
+}
+````
+
+> You see the power of the specifications with these samples. If you change the `PremiumCustomerSpecification` later, say change the balance from `100.000` to `200.000`, all the queries and combined specifications will be effected by the change. This is a good way to reduce code duplication!
+
+## Discussions
+
+While the specification pattern is older than C# lambda expressions, it's generally compared to expressions. Some developers may think it's not needed anymore and we can directly pass expressions to a repository or to a domain service as shown below:
+
+````csharp
+var count = await _customerRepository.CountAsync(c => c.Balance > 100000 && c.Age => 18);
+````
+
+Since ABP's [Repository](Repositories.md) supports Expressions, this is a completely valid use. You don't have to define or use any specification in your application and you can go with expressions.
+
+So, what's the point of a specification? Why and when should we consider to use them?
+
+### When To Use?
+
+Some benefits of using specifications:
+
+- **Reusabe**: Imagine that you need the Premium Customer filter in many places in your code base. If you go with expressions and do not create a specification, what happens if you later change the "Premium Customer" definition? Say you want to change the minimum balance from $100,000 to $250,000 and add another condition to be a customer older than 3 years. If you'd used a specification, you just change a single class. If you repeated (copy/pasted) the same expression everywhere, you need to change all of them.
+- **Composable**: You can combine multiple specifications to create new specifications. This is another type of reusability.
+- **Named**: `PremiumCustomerSpecification` better explains the intent rather than a complex expression. So, if you have an expression that is meaningful in your business, consider using specifications.
+- **Testable**: A specification is a separately (and easily) testable object.
+
+### When To Not Use?
+
+- **Non business expressions**: Do not use specifications for non business-related expressions and operations.
+- **Reporting**: If you are just creating a report, do not create specifications, but directly use `IQueryable` & LINQ expressions. You can even use plain SQL, views or another tool for reporting. DDD does not necessarily care about reporting, so the way you query the underlying data store can be important from a performance perspective.
\ No newline at end of file
diff --git a/docs/en/Testing.md b/docs/en/Testing.md
index 5dafef3daf..0a617174ed 100644
--- a/docs/en/Testing.md
+++ b/docs/en/Testing.md
@@ -1,3 +1,673 @@
-# Testing
+# Automated Testing
-TODO!
\ No newline at end of file
+## Introduction
+
+ABP Framework has been designed with testability in mind. There are some different levels of automated testing;
+
+* **Unit Tests**: You typically test a single class (or a very few classes together). These tests will be fast. However, you generally need to deal with mocking for the dependencies of your service(s).
+* **Integration Tests**: You typically test a service, but this time you don't mock the fundamental infrastructure and services to see if they properly working together.
+* **UI Tests**: You test the UI of the application, just like the users interact with your application.
+
+### Unit Tests vs Integration Tests
+
+Integration tests have some significant **advantages** compared to unit tests;
+
+* **Easier to write** since you don't work to establish mocking and dealing with the dependencies.
+* Your test code runs with all the real services and infrastructure (including database mapping and queries), so it is much closer to the **real application test**.
+
+While they have some drawbacks;
+
+* They are **slower** compared to unit tests since all the infrastructure is prepared for each test case.
+* A bug in a service may make multiple test cases broken, so it may be **harder to find the real problem** in some cases.
+
+We suggest to go mixed: Write unit or integration test where it is necessary and you find effective to write and maintain it.
+
+## The Application Startup Template
+
+The [Application Startup Template](Startup-Templates/Application.md) comes with the test infrastructure properly installed and configured for you.
+
+### The Test Projects
+
+See the following solution structure in the Visual Studio:
+
+
+
+There are more than one test project, organized by the layers;
+
+* `Domain.Tests` is used to test your Domain Layer objects (like [Domain Services](Domain-Services.md) and [Entities](Entities.md)).
+* `Application.Tests` is used to test your Application Layer (like [Application Services](Application-Services.md)).
+* `EntityFrameworkCore.Tests` is used to test your custom repository implementations or EF Core mappings (this project will be different if you use another [Database Provider](Data-Access.md)).
+* `Web.Tests` is used to test the UI Layer (like Pages, Controllers and View Components). This project does exists only for MVC / Razor Page applications.
+* `TestBase` contains some classes those are shared/used by the other projects.
+
+> `HttpApi.Client.ConsoleTestApp` is not an automated test application. It is an example Console Application that shows how to consume your HTTP APIs from a .NET Console Application.
+
+The following sections will introduce the base classes and other infrastructure included in these projects.
+
+### The Test Infrastructure
+
+The startup solution has the following libraries already installed;
+
+* [xUnit](https://xunit.net/) as the test framework.
+* [NSubstitute](https://nsubstitute.github.io/) as the mocking library.
+* [Shouldly](https://github.com/shouldly/shouldly) as the assertion library.
+
+While you are free to replace them with your favorite tools, this document and examples will be base on these tooling.
+
+## The Test Explorer
+
+You can use the Test Explorer to view and run the tests in Visual Studio. For other IDEs, see their own documentation.
+
+### Open the Test Explorer
+
+Open the *Test Explorer*, under the *Tests* menu, if it is not already open:
+
+
+
+### Run the Tests
+
+Then you can click to the Run All or Run buttons to run the tests. The initial startup template has some sample tests for you:
+
+
+
+### Run Tests In Parallel
+
+The test infrastructure is compatible to run the tests in parallel. It is **strongly suggested** to run all the tests in parallel, which is pretty faster then running them one by one.
+
+To enable it, click to the caret icon near to the settings (gear) button and select the *Run Tests In Parallel*.
+
+
+
+## Unit Tests
+
+For Unit Tests, you don't need to much infrastructure. You typically instantiate your class and provide some pre-configured mocked objects to prepare your object to test.
+
+### Classes Without Dependencies
+
+In this simplest case, the class you want to test has no dependencies. In this case, you can directly instantiate your class, call its methods and make your assertions.
+
+#### Example: Testing an Entity
+
+Assume that you've an `Issue` [entity](Entities.md) as shown below:
+
+````csharp
+using System;
+using Volo.Abp.Domain.Entities;
+
+namespace MyProject.Issues
+{
+ public class Issue : AggregateRoot
+ {
+ public string Title { get; set; }
+ public string Description { get; set; }
+ public bool IsLocked { get; set; }
+ public bool IsClosed { get; private set; }
+ public DateTime? CloseDate { get; private set; }
+
+ public void Close()
+ {
+ IsClosed = true;
+ CloseDate = DateTime.UtcNow;
+ }
+
+ public void Open()
+ {
+ if (!IsClosed)
+ {
+ return;
+ }
+
+ if (IsLocked)
+ {
+ throw new IssueStateException("You can not open a locked issue!");
+ }
+
+ IsClosed = true;
+ CloseDate = null;
+ }
+ }
+}
+
+````
+
+Notice that the `IsClosed` and `CloseDate` properties have private setters to force some business rules by using the `Open()` and `Close()` methods;
+
+* Whenever you close an issue, the `CloseDate` should be set to the [current time](Timing.md).
+* An issue can not be re-opened if it is locked. And if it is re-opened, the `CloseDate` should be set to `null`.
+
+Since the `Issue` entity is a part of the Domain Layer, we should test it in the `Domain.Tests` project. Create an `Issue_Tests` class inside the `Domain.Tests` project:
+
+````csharp
+using Shouldly;
+using Xunit;
+
+namespace MyProject.Issues
+{
+ public class Issue_Tests
+ {
+ [Fact]
+ public void Should_Set_The_CloseDate_Whenever_Close_An_Issue()
+ {
+ // Arrange
+
+ var issue = new Issue();
+ issue.CloseDate.ShouldBeNull(); // null at the beginning
+
+ // Act
+
+ issue.Close();
+
+ // Assert
+
+ issue.IsClosed.ShouldBeTrue();
+ issue.CloseDate.ShouldNotBeNull();
+ }
+ }
+}
+````
+
+This test follows the AAA (Arrange-Act-Assert) pattern;
+
+* **Arrange** part creates an `Issue` entity and ensures the `CloseDate` is `null` at the beginning.
+* **Act** part executes the method we want to test for this case.
+* **Assert** part checks if the `Issue` properties are same as we expect to be.
+
+`[Fact]` attribute is defined by the [xUnit](https://xunit.net/) library and marks a method as a test method. `Should...` extension methods are provided by the [Shouldly](https://github.com/shouldly/shouldly) library. You can directly use the `Assert` class of the xUnit, but Shouldly makes it much comfortable and straightforward.
+
+When you execute the tests, you will see that is passes successfully:
+
+
+
+Let's add two more test methods:
+
+````csharp
+[Fact]
+public void Should_Allow_To_ReOpen_An_Issue()
+{
+ // Arrange
+
+ var issue = new Issue();
+ issue.Close();
+
+ // Act
+
+ issue.Open();
+
+ // Assert
+
+ issue.IsClosed.ShouldBeFalse();
+ issue.CloseDate.ShouldBeNull();
+}
+
+[Fact]
+public void Should_Not_Allow_To_ReOpen_A_Locked_Issue()
+{
+ // Arrange
+
+ var issue = new Issue();
+ issue.Close();
+ issue.IsLocked = true;
+
+ // Act & Assert
+
+ Assert.Throws(() =>
+ {
+ issue.Open();
+ });
+}
+````
+
+`Assert.Throws` checks if the executed code throws a matching exception.
+
+> See the xUnit & Shoudly documentations to learn more about these libraries.
+
+### Classes With Dependencies
+
+If your service has dependencies and you want to unit test this service, you need to mock the dependencies.
+
+#### Example: Testing a Domain Service
+
+Assume that you've an `IssueManager` [Domain Service](Domain-Services.md) that is defined as below:
+
+````csharp
+using System;
+using System.Threading.Tasks;
+using Volo.Abp;
+using Volo.Abp.Domain.Services;
+
+namespace MyProject.Issues
+{
+ public class IssueManager : DomainService
+ {
+ public const int MaxAllowedOpenIssueCountForAUser = 3;
+
+ private readonly IIssueRepository _issueRepository;
+
+ public IssueManager(IIssueRepository issueRepository)
+ {
+ _issueRepository = issueRepository;
+ }
+
+ public async Task AssignToUserAsync(Issue issue, Guid userId)
+ {
+ var issueCount = await _issueRepository.GetIssueCountOfUserAsync(userId);
+
+ if (issueCount >= MaxAllowedOpenIssueCountForAUser)
+ {
+ throw new BusinessException(
+ code: "IM:00392",
+ message: $"You can not assign more" +
+ $"than {MaxAllowedOpenIssueCountForAUser} issues to a user!"
+ );
+ }
+
+ issue.AssignedUserId = userId;
+ }
+ }
+}
+````
+
+`IssueManager` depends on the `IssueRepository` service, that will be mocked in this example.
+
+**Business Rule**: The example `AssignToUserAsync` doesn't allow to assign more than 3 (`MaxAllowedOpenIssueCountForAUser` constant) issues to a user. If you want to assign an issue in this case, you first need to unassign an existing issue.
+
+The test case below tries to make a valid assignment:
+
+````csharp
+using System;
+using System.Threading.Tasks;
+using NSubstitute;
+using Shouldly;
+using Volo.Abp;
+using Xunit;
+
+namespace MyProject.Issues
+{
+ public class IssueManager_Tests
+ {
+ [Fact]
+ public async Task Should_Assign_An_Issue_To_A_User()
+ {
+ // Arrange
+
+ var userId = Guid.NewGuid();
+
+ var fakeRepo = Substitute.For();
+ fakeRepo.GetIssueCountOfUserAsync(userId).Returns(1);
+
+ var issueManager = new IssueManager(fakeRepo);
+
+ var issue = new Issue();
+
+ // Act
+
+ await issueManager.AssignToUserAsync(issue, userId);
+
+ //Assert
+
+ issue.AssignedUserId.ShouldBe(userId);
+ await fakeRepo.Received(1).GetIssueCountOfUserAsync(userId);
+ }
+ }
+}
+````
+
+* `Substitute.For` creates a mock (fake) object that is passed into the `IssueManager` constructor.
+* `fakeRepo.GetIssueCountOfUserAsync(userId).Returns(1)` ensures that the `GetIssueCountOfUserAsync` method of the repository returns `1`.
+* `issueManager.AssignToUserAsync` doesn't throw any exception since the repository returns `1` for the currently assigned issue count.
+* `issue.AssignedUserId.ShouldBe(userId);` line checks if the `AssignedUserId` has the correct value.
+* `await fakeRepo.Received(1).GetIssueCountOfUserAsync(userId);` checks if the `IssueManager` called the `GetIssueCountOfUserAsync` method exactly one time.
+
+Let's add a second test to see if it prevents to assign issues to a user more than the allowed count:
+
+````csharp
+[Fact]
+public async Task Should_Not_Allow_To_Assign_Issues_Over_The_Limit()
+{
+ // Arrange
+
+ var userId = Guid.NewGuid();
+
+ var fakeRepo = Substitute.For();
+ fakeRepo
+ .GetIssueCountOfUserAsync(userId)
+ .Returns(IssueManager.MaxAllowedOpenIssueCountForAUser);
+
+ var issueManager = new IssueManager(fakeRepo);
+
+ // Act & Assert
+
+ var issue = new Issue();
+
+ await Assert.ThrowsAsync(async () =>
+ {
+ await issueManager.AssignToUserAsync(issue, userId);
+ });
+
+ issue.AssignedUserId.ShouldBeNull();
+ await fakeRepo.Received(1).GetIssueCountOfUserAsync(userId);
+}
+````
+
+> For more information on the mocking, see the [NSubstitute](https://nsubstitute.github.io/) documentation.
+
+It is relatively easy to mock a single dependency. But, when your dependencies grow, it gets harder to setup the test objects and mock all the dependencies. See the *Integration Tests* section that doesn't require mocking the dependencies.
+
+### Tip: Share the Test Class Constructor
+
+[xUnit](https://xunit.net/) creates a **new test class instance** (`IssueManager_Tests` for this example) for each test method. So, you can move some *Arrange* code into the constructor to reduce the code duplication. The constructor will be executed for each test case and doesn't affect each other, even if they work in parallel.
+
+**Example: Refactor the `IssueManager_Tests` to reduce the code duplication**
+
+````csharp
+using System;
+using System.Threading.Tasks;
+using NSubstitute;
+using Shouldly;
+using Volo.Abp;
+using Xunit;
+
+namespace MyProject.Issues
+{
+ public class IssueManager_Tests
+ {
+ private readonly Guid _userId;
+ private readonly IIssueRepository _fakeRepo;
+ private readonly IssueManager _issueManager;
+ private readonly Issue _issue;
+
+ public IssueManager_Tests()
+ {
+ _userId = Guid.NewGuid();
+ _fakeRepo = Substitute.For();
+ _issueManager = new IssueManager(_fakeRepo);
+ _issue = new Issue();
+ }
+
+ [Fact]
+ public async Task Should_Assign_An_Issue_To_A_User()
+ {
+ // Arrange
+ _fakeRepo.GetIssueCountOfUserAsync(_userId).Returns(1);
+
+ // Act
+ await _issueManager.AssignToUserAsync(_issue, _userId);
+
+ //Assert
+ _issue.AssignedUserId.ShouldBe(_userId);
+ await _fakeRepo.Received(1).GetIssueCountOfUserAsync(_userId);
+ }
+
+ [Fact]
+ public async Task Should_Not_Allow_To_Assign_Issues_Over_The_Limit()
+ {
+ // Arrange
+ _fakeRepo
+ .GetIssueCountOfUserAsync(_userId)
+ .Returns(IssueManager.MaxAllowedOpenIssueCountForAUser);
+
+ // Act & Assert
+ await Assert.ThrowsAsync(async () =>
+ {
+ await _issueManager.AssignToUserAsync(_issue, _userId);
+ });
+
+ _issue.AssignedUserId.ShouldBeNull();
+ await _fakeRepo.Received(1).GetIssueCountOfUserAsync(_userId);
+ }
+ }
+}
+````
+
+> Keep your test code clean to create a maintainable test suite.
+
+## Integration Tests
+
+> You can follow the [web application development tutorial](Tutorials/Part-1.md) to learn developing a full stack application, including the integration tests.
+
+### The Integration Test Infrastructure
+
+ABP Provides a complete infrastructure to write integration tests. All the ABP infrastructure and services will perform in your tests. The application startup template comes with the necessary infrastructure pre-configured for you;
+
+#### The Database
+
+The startup template is configured to use **in-memory SQLite** database for the EF Core (for MongoDB, it uses [Mongo2Go](https://github.com/Mongo2Go/Mongo2Go) library). So, all the configuration and queries are performed against a real database and you can even test database transactions.
+
+Using in-memory SQLite database has two main advantages;
+
+* It is faster compared to an external DBMS.
+* It create a **new fresh database** for each test case, so tests doesn't affect each other.
+
+> **Tip**: Do not use EF Core's In-Memory database for advanced integration tests. It is not a real DBMS and has many differences in details. For example, it doesn't support transaction and rollback scenarios, so you can't truly test the failing scenarios. On the other hand, In-Memory SQLite is a real DBMS and supports the fundamental SQL database features.
+
+### The Seed Data
+
+Writing tests against an empty database is not practical. In most cases, you need to some initial data in the database. For example, if you write a test class that query, update and delete the Products, it would be helpful to have a few products in the database before executing the test case.
+
+ABP's [Data Seeding](Data-Seeding.md) system is a powerful way to seed the initial data. The application startup template has a *YourProject*TestDataSeedContributor class in the `.TestBase` project. You can fill it to have an initial data that you can use for each test method.
+
+**Example: Create some Issues as the seed data**
+
+````csharp
+using System.Threading.Tasks;
+using MyProject.Issues;
+using Volo.Abp.Data;
+using Volo.Abp.DependencyInjection;
+
+namespace MyProject
+{
+ public class MyProjectTestDataSeedContributor
+ : IDataSeedContributor, ITransientDependency
+ {
+ private readonly IIssueRepository _issueRepository;
+
+ public MyProjectTestDataSeedContributor(IIssueRepository issueRepository)
+ {
+ _issueRepository = issueRepository;
+ }
+
+ public async Task SeedAsync(DataSeedContext context)
+ {
+ await _issueRepository.InsertAsync(
+ new Issue
+ {
+ Title = "Test issue one",
+ Description = "Test issue one description",
+ AssignedUserId = TestData.User1Id
+ });
+
+ await _issueRepository.InsertAsync(
+ new Issue
+ {
+ Title = "Test issue two",
+ Description = "Test issue two description",
+ AssignedUserId = TestData.User1Id
+ });
+
+ await _issueRepository.InsertAsync(
+ new Issue
+ {
+ Title = "Test issue three",
+ Description = "Test issue three description",
+ AssignedUserId = TestData.User1Id
+ });
+
+ await _issueRepository.InsertAsync(
+ new Issue
+ {
+ Title = "Test issue four",
+ Description = "Test issue four description",
+ AssignedUserId = TestData.User2Id
+ });
+ }
+ }
+}
+````
+
+Also created a static class to store the User `Ids`:
+
+````csharp
+using System;
+
+namespace MyProject
+{
+ public static class TestData
+ {
+ public static Guid User1Id = Guid.Parse("41951813-5CF9-4204-8B18-CD765DBCBC9B");
+ public static Guid User2Id = Guid.Parse("2DAB4460-C21B-4925-BF41-A52750A9B999");
+ }
+}
+````
+
+In this way, we can use these known Issues and the User `Id`s to perform the tests.
+
+### Example: Testing a Domain Service
+
+`AbpIntegratedTest` class (defined in the [Volo.Abp.TestBase](https://www.nuget.org/packages/Volo.Abp.TestBase) package) is used to write tests integrated to the ABP Framework. `T` is the Type of the root module to setup and initialize the application.
+
+The application startup template has base classes in each test project, so you can derive from these base classes to make it easier.
+
+See the `IssueManager` tests are re-written as integration tests
+
+````csharp
+using System.Threading.Tasks;
+using Shouldly;
+using Volo.Abp;
+using Xunit;
+
+namespace MyProject.Issues
+{
+ public class IssueManager_Integration_Tests : MyProjectDomainTestBase
+ {
+ private readonly IssueManager _issueManager;
+ private readonly Issue _issue;
+
+ public IssueManager_Integration_Tests()
+ {
+ _issueManager = GetRequiredService();
+ _issue = new Issue
+ {
+ Title = "Test title",
+ Description = "Test description"
+ };
+ }
+
+ [Fact]
+ public async Task Should_Not_Allow_To_Assign_Issues_Over_The_Limit()
+ {
+ // Act & Assert
+ await Assert.ThrowsAsync(async () =>
+ {
+ await _issueManager.AssignToUserAsync(_issue, TestData.User1Id);
+ });
+
+ _issue.AssignedUserId.ShouldBeNull();
+ }
+
+ [Fact]
+ public async Task Should_Assign_An_Issue_To_A_User()
+ {
+ // Act
+ await _issueManager.AssignToUserAsync(_issue, TestData.User2Id);
+
+ //Assert
+ _issue.AssignedUserId.ShouldBe(TestData.User2Id);
+ }
+ }
+}
+````
+
+* First test method assigns the issue to the User 1, which has already assigned to 3 issues in the Data Seed code. So, it throws a `BusinessException`.
+* Second test method assigns the issue to User 2, which has only 1 issue assigned. So, the method succeeds.
+
+This class typically locates in the `.Domain.Tests` project since it tests a class located in the `.Domain` project. It is derived from the `MyProjectDomainTestBase` which is already configured to properly run the tests.
+
+Writing such an integration test class is very straightforward. Another benefit is that you won't need to change the test class later when you add another dependency to the `IssueManager` class.
+
+### Example: Testing an Application Service
+
+Testing an [Application Service](Application-Services.md) is not so different. Assume that you've created an `IssueAppService` as defined below:
+
+````csharp
+using System.Collections.Generic;
+using System.Threading.Tasks;
+using Volo.Abp.Application.Services;
+
+namespace MyProject.Issues
+{
+ public class IssueAppService : ApplicationService, IIssueAppService
+ {
+ private readonly IIssueRepository _issueRepository;
+
+ public IssueAppService(IIssueRepository issueRepository)
+ {
+ _issueRepository = issueRepository;
+ }
+
+ public async Task> GetListAsync()
+ {
+ var issues = await _issueRepository.GetListAsync();
+
+ return ObjectMapper.Map, List>(issues);
+ }
+ }
+}
+````
+
+*(assuming you've also defined the `IIssueAppService` and `IssueDto` and created the [object mapping](Object-To-Object-Mapping.md) between `Issue` and the `IssueDto`)*
+
+Now, you can write a test class inside the `.Application.Tests` project:
+
+````csharp
+using System.Threading.Tasks;
+using Shouldly;
+using Xunit;
+
+namespace MyProject.Issues
+{
+ public class IssueAppService_Tests : MyProjectApplicationTestBase
+ {
+ private readonly IIssueAppService _issueAppService;
+
+ public IssueAppService_Tests()
+ {
+ _issueAppService = GetRequiredService();
+ }
+
+ [Fact]
+ public async Task Should_Get_All_Issues()
+ {
+ //Act
+ var issueDtos = await _issueAppService.GetListAsync();
+
+ //Assert
+ issueDtos.Count.ShouldBeGreaterThan(0);
+ }
+ }
+}
+````
+
+It's that simple. This test method tests everything, including the application service, EF Core mapping, object to object mapping and the repository implementation. In this way, you can fully test the Application Later and the Domain Layer of your solution.
+
+## UI Tests
+
+In general, there are two types of UI Tests;
+
+### Non Visual Tests
+
+Such tests completely depends on your UI Framework choice;
+
+* For an MVC / Razor Pages UI, you typically make request to the server, get some HTML and test if some expected DOM elements exist in the returned result.
+* Angular has its own infrastructure and practices to test the components, views and services.
+
+See the following documents to learn Non Visual UI Testing;
+
+* [Testing in ASP.NET Core MVC / Razor Pages](UI/AspNetCore/Testing.md)
+* [Testing in Angular](UI/Angular/Testing.md)
+* [Testing in Blazor](UI/Blazor/Testing.md)
+
+### Visual Tests
+
+Visual Tests are used to interact with the application UI just like a real user does. It fully tests the application, including the visual appearance of the pages and components.
+
+Visual UI Testing is out of the scope for the ABP Framework. There are a lot of tooling in the industry (like [Selenium](https://www.selenium.dev/)) that you can use to test your application's UI.
\ No newline at end of file
diff --git a/docs/en/Text-Templating.md b/docs/en/Text-Templating.md
index 07f3b141ab..b080087ba8 100644
--- a/docs/en/Text-Templating.md
+++ b/docs/en/Text-Templating.md
@@ -187,9 +187,9 @@ var result = await _templateRenderer.RenderAsync(
In this case, we haven't created a model class, but created an anonymous object as the model.
-### PascalCase vs camelCase
+### PascalCase vs snake_case
-PascalCase property names (like `UserName`) is used as camelCase (like `userName`) in the templates.
+PascalCase property names (like `UserName`) is used as snake_case (like `user_name`) in the templates.
## Localization
diff --git a/docs/en/Tutorials/Part-10.md b/docs/en/Tutorials/Part-10.md
index 53f19d93dd..a9de3a062e 100644
--- a/docs/en/Tutorials/Part-10.md
+++ b/docs/en/Tutorials/Part-10.md
@@ -10,7 +10,7 @@
In this tutorial series, you will build an ABP based web application named `Acme.BookStore`. This application is used to manage a list of books and their authors. It is developed using the following technologies:
-* **{{DB_Value}}** as the ORM provider.
+* **{{DB_Value}}** as the ORM provider.
* **{{UI_Value}}** as the UI Framework.
This tutorial is organized as the following parts;
@@ -50,7 +50,7 @@ public Guid AuthorId { get; set; }
{{if DB=="EF"}}
-> In this tutorial, we preferred to not add a **navigation property** to the `Author` entity from the `Book` class (like `public Author Author { get; set; }`). This is due to follow the DDD best practices (rule: refer to other aggregates only by id). However, you can add such a navigation property and configure it for the EF Core. In this way, you don't need to write join queries while getting books with their authors (just like we will done below) which makes your application code simpler.
+> In this tutorial, we preferred to not add a **navigation property** to the `Author` entity from the `Book` class (like `public Author Author { get; set; }`). This is due to follow the DDD best practices (rule: refer to other aggregates only by id). However, you can add such a navigation property and configure it for the EF Core. In this way, you don't need to write join queries while getting books with their authors (like we will done below) which makes your application code simpler.
{{end}}
@@ -78,7 +78,7 @@ builder.Entity(b =>
b.ToTable(BookStoreConsts.DbTablePrefix + "Books", BookStoreConsts.DbSchema);
b.ConfigureByConvention(); //auto configure for the base class props
b.Property(x => x.Name).IsRequired().HasMaxLength(128);
-
+
// ADD THE MAPPING FOR THE RELATION
b.HasOne().WithMany().HasForeignKey(x => x.AuthorId).IsRequired();
});
@@ -384,8 +384,8 @@ namespace Acme.BookStore.Books
return bookDto;
}
- public override async Task>
- GetListAsync(PagedAndSortedResultRequestDto input)
+ public override async Task> GetListAsync(
+ PagedAndSortedResultRequestDto input)
{
await CheckGetListPolicyAsync();
@@ -485,7 +485,7 @@ namespace Acme.BookStore.Books
DeletePolicyName = BookStorePermissions.Books.Create;
}
- public override async Task GetAsync(Guid id)
+ public async override Task GetAsync(Guid id)
{
await CheckGetPolicyAsync();
@@ -498,7 +498,7 @@ namespace Acme.BookStore.Books
return bookDto;
}
- public override async Task>
+ public async override Task>
GetListAsync(PagedAndSortedResultRequestDto input)
{
await CheckGetListPolicyAsync();
@@ -524,7 +524,7 @@ namespace Acme.BookStore.Books
var authorDictionary = await GetAuthorDictionaryAsync(books);
//Set AuthorName for the DTOs
- bookDtos.ForEach(bookDto => bookDto.AuthorName =
+ bookDtos.ForEach(bookDto => bookDto.AuthorName =
authorDictionary[bookDto.AuthorId].Name);
//Get the total count with another query (required for the paging)
@@ -622,7 +622,7 @@ namespace Acme.BookStore.Books
result.Items.ShouldContain(b => b.Name == "1984" &&
b.AuthorName == "George Orwell");
}
-
+
[Fact]
public async Task Should_Create_A_Valid_Book()
{
@@ -645,7 +645,7 @@ namespace Acme.BookStore.Books
result.Id.ShouldNotBe(Guid.Empty);
result.Name.ShouldBe("New test book 42");
}
-
+
[Fact]
public async Task Should_Not_Create_A_Book_Without_Name()
{
@@ -911,6 +911,17 @@ You can run the application and try to create a new book or update an existing b
{{else if UI=="NG"}}
+### Service Proxy Generation
+
+Since the HTTP APIs have been changed, you need to update Angular client side [service proxies](../UI/Angular/Service-Proxies.md). Before running `generate-proxy` command, your host must be up and running.
+
+Run the following command in the `angular` folder (you may need to stop the angular application):
+
+```bash
+abp generate-proxy
+```
+This command will update the service proxy files under the `/src/app/proxy/` folder.
+
### The Book List
Book list page change is trivial. Open the `/src/app/book/book.component.html` and add the following column definition between the `Name` and `Type` columns:
@@ -1082,36 +1093,37 @@ Add the following field to the `@code` section of the `Books.razor` file:
IReadOnlyList authorList = Array.Empty();
````
-And fill it in the `OnInitializedAsync` method, by adding the following code to the end of the method:
+Override the `OnInitializedAsync` method and adding the following code:
````csharp
-authorList = (await AppService.GetAuthorLookupAsync()).Items;
+protected override async Task OnInitializedAsync()
+{
+ await base.OnInitializedAsync();
+ authorList = (await AppService.GetAuthorLookupAsync()).Items;
+}
````
+* It is essential to call the `base.OnInitializedAsync()` since `AbpCrudPageBase` has some initialization code to be executed.
+
The final `@code` block should be the following:
````csharp
@code
{
- bool canCreateBook;
- bool canEditBook;
- bool canDeleteBook;
-
//ADDED A NEW FIELD
IReadOnlyList authorList = Array.Empty();
+ public Books() // Constructor
+ {
+ CreatePolicyName = BookStorePermissions.Books.Create;
+ UpdatePolicyName = BookStorePermissions.Books.Edit;
+ DeletePolicyName = BookStorePermissions.Books.Delete;
+ }
+
+ //GET AUTHORS ON INITIALIZATION
protected override async Task OnInitializedAsync()
{
await base.OnInitializedAsync();
-
- canCreateBook = await
- AuthorizationService.IsGrantedAsync(BookStorePermissions.Books.Create);
- canEditBook = await
- AuthorizationService.IsGrantedAsync(BookStorePermissions.Books.Edit);
- canDeleteBook = await
- AuthorizationService.IsGrantedAsync(BookStorePermissions.Books.Delete);
-
- //GET AUTHORS
authorList = (await AppService.GetAuthorLookupAsync()).Items;
}
}
@@ -1164,4 +1176,4 @@ Add the following `Field` definition into the `ModalBody` of the *Edit* modal, a
That's all. We are reusing the `authorList` defined for the *Create* modal.
-{{end}}
\ No newline at end of file
+{{end}}
diff --git a/docs/en/Tutorials/Part-2.md b/docs/en/Tutorials/Part-2.md
index f4fcf61e25..7bc42b1fc9 100644
--- a/docs/en/Tutorials/Part-2.md
+++ b/docs/en/Tutorials/Part-2.md
@@ -590,7 +590,6 @@ Open the `Books.razor` and replace the content as the following:
````xml
@page "/books"
@using Volo.Abp.Application.Dtos
-@using Volo.Abp.BlazoriseUI
@using Acme.BookStore.Books
@using Acme.BookStore.Localization
@using Microsoft.Extensions.Localization
@@ -620,14 +619,14 @@ Open the `Books.razor` and replace the content as the following:
@context.PublishDate.ToShortDateString()
` section with the following
````xml
-
-
+
+
@L["Books"]
-
-
-
-
+
+
@@ -1196,48 +1194,67 @@ Now, we can add a modal that will be opened when we click to the button.
Open the `Books.razor` and add the following code to the end of the page:
````xml
-
+
-
- @L["NewBook"]
-
-
-
-
- @L["Name"]
-
-
-
- @L["Type"]
-
-
-
- @L["PublishDate"]
-
-
-
- @L["Price"]
-
-
-
-
-
-
-
+
````
+This code requires a service; Inject the `AbpBlazorMessageLocalizerHelper` at the top of the file, just before the `@inherits...` line:
+
+````csharp
+@inject AbpBlazorMessageLocalizerHelper LH
+````
+
+* The form implements validation and the `AbpBlazorMessageLocalizerHelper` is used to simply localize the validation messages.
* `CreateModal` object, `CloseCreateModalAsync` and `CreateEntityAsync` method are defined by the base class. See the [Blazorise documentation](https://blazorise.com/docs/) if you want to understand the `Modal` and the other components.
That's all. Run the application and try to add a new book:
@@ -1250,78 +1267,81 @@ Editing a books is similar to the creating a new book.
### Actions Dropdown
-Open the `Books.razor` and add the following `DataGridColumn` section inside the `DataGridColumns` as the first item:
+Open the `Books.razor` and add the following `DataGridEntityActionsColumn` section inside the `DataGridColumns` as the first item:
````xml
-
+
-
-
- @L["Actions"]
-
-
-
- @L["Edit"]
-
-
-
+
+
+
-
+
````
-* `OpenEditModalAsync` is defined in the base class which takes the `Id` of the entity (book) to edit.
+* `OpenEditModalAsync` is defined in the base class which takes the entity (book) to edit.
-This adds an "Actions" dropdown to all the books inside the `DataGrid` with an `Edit` action:
+`DataGridEntityActionsColumn` component is used to show an "Actions" dropdown for each row in the `DataGrid`. `DataGridEntityActionsColumn` shows a **single button** instead of a dropdown if there is only one available action inside it:
-
+
### Edit Modal
We can now define a modal to edit the book. Add the following code to the end of the `Books.razor` page:
````xml
-
+
-
- @EditingEntity.Name
-
-
-
-
- @L["Name"]
-
-
-
- @L["Type"]
-
-
-
- @L["PublishDate"]
-
-
-
- @L["Price"]
-
-
-
-
-
-
-
+
````
@@ -1356,17 +1376,26 @@ You can now run the application and try to edit a book.

+> Tip: Try to leave the *Name* field empty and submit the form to show the validation error message.
+
## Deleting a Book
-Open the `Books.razor` page and add the following `DropdownItem` under the "Edit" action inside the "Actions" `DropdownMenu`:
+Open the `Books.razor` page and add the following `EntityAction` under the "Edit" action inside the `EntityActions`:
````xml
-
- @L["Delete"]
-
+
````
-* `DeleteEntityAsync` is defined in the base class.
+* `DeleteEntityAsync` is defined in the base class that deletes the entity by performing a call to the server.
+* `ConfirmationMessage` is a callback to show a confirmation message before executing the action.
+* `GetDeleteConfirmationMessage` is defined in the base class. You can override this method (or pass another value to the `ConfirmationMessage` parameter) to customize the localization message.
+
+The "Actions" button becomes a dropdown since it has two actions now:
+
+
Run the application and try to delete a book.
@@ -1377,26 +1406,22 @@ Here the complete code to create the book management CRUD page, that has been de
````xml
@page "/books"
@using Volo.Abp.Application.Dtos
-@using Volo.Abp.BlazoriseUI
@using Acme.BookStore.Books
@using Acme.BookStore.Localization
@using Microsoft.Extensions.Localization
@inject IStringLocalizer L
+@inject AbpBlazorMessageLocalizerHelper LH
@inherits AbpCrudPageBase
-
-
+
+
@L["Books"]
-
-
+
-
+ Clicked="OpenCreateModalAsync">@L["NewBook"]
@@ -1408,27 +1433,19 @@ Here the complete code to create the book management CRUD page, that has been de
ShowPager="true"
PageSize="PageSize">
-
+
-
-
- @L["Actions"]
-
-
-
- @L["Edit"]
-
-
- @L["Delete"]
-
-
-
+
+
+
+
-
+
@@ -1436,7 +1453,7 @@ Here the complete code to create the book management CRUD page, that has been de
Field="@nameof(BookDto.Type)"
Caption="@L["Type"]">
- @L[$"Enum:BookType:{(int)context.Type}"]
+ @L[$"Enum:BookType:{(int) context.Type}"]
-
+
-
- @L["NewBook"]
-
-
-
-
- @L["Name"]
-
-
-
- @L["Type"]
-
-
-
- @L["PublishDate"]
-
-
-
- @L["Price"]
-
-
-
-
-
-
-
+
-
+
-
- @EditingEntity.Name
-
-
-
-
- @L["Name"]
-
-
-
- @L["Type"]
-
-
-
- @L["PublishDate"]
-
-
-
- @L["Price"]
-
-
-
-
-
-
-
+
````
diff --git a/docs/en/Tutorials/Part-4.md b/docs/en/Tutorials/Part-4.md
index d9c000be5c..8dab28a8db 100644
--- a/docs/en/Tutorials/Part-4.md
+++ b/docs/en/Tutorials/Part-4.md
@@ -126,7 +126,7 @@ public async Task Should_Create_A_Valid_Book()
{
Name = "New test book 42",
Price = 10,
- PublishDate = System.DateTime.Now,
+ PublishDate = DateTime.Now,
Type = BookType.ScienceFiction
}
);
@@ -208,7 +208,7 @@ namespace Acme.BookStore.Books
{
Name = "New test book 42",
Price = 10,
- PublishDate = System.DateTime.Now,
+ PublishDate = DateTime.Now,
Type = BookType.ScienceFiction
}
);
diff --git a/docs/en/Tutorials/Part-5.md b/docs/en/Tutorials/Part-5.md
index a293ec586e..29374419cb 100644
--- a/docs/en/Tutorials/Part-5.md
+++ b/docs/en/Tutorials/Part-5.md
@@ -70,7 +70,7 @@ namespace Acme.BookStore.Permissions
}
````
-This is a hierarchical way of defining permission names. For example, "create book" permission name was defined as `BookStore.Books.Create`.
+This is a hierarchical way of defining permission names. For example, "create book" permission name was defined as `BookStore.Books.Create`. ABP doesn't force you to a structure, but we find this way useful.
### Permission Definitions
@@ -423,13 +423,13 @@ Open the `/src/app/book/book.component.html` file and replace the create button
````html
-
diff --git a/framework/src/Volo.Abp.AspNetCore.Authentication.OpenIdConnect/Volo.Abp.AspNetCore.Authentication.OpenIdConnect.csproj b/framework/src/Volo.Abp.AspNetCore.Authentication.OpenIdConnect/Volo.Abp.AspNetCore.Authentication.OpenIdConnect.csproj
index bac9441f00..b0aa11f206 100644
--- a/framework/src/Volo.Abp.AspNetCore.Authentication.OpenIdConnect/Volo.Abp.AspNetCore.Authentication.OpenIdConnect.csproj
+++ b/framework/src/Volo.Abp.AspNetCore.Authentication.OpenIdConnect/Volo.Abp.AspNetCore.Authentication.OpenIdConnect.csproj
@@ -9,7 +9,7 @@
-
+
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/BasicThemeBundleContributor.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/BasicThemeBundleContributor.cs
new file mode 100644
index 0000000000..edef97eb52
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/BasicThemeBundleContributor.cs
@@ -0,0 +1,17 @@
+using Volo.Abp.Bundling;
+
+namespace Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme
+{
+ public class BasicThemeBundleContributor : IBundleContributor
+ {
+ public void AddScripts(BundleContext context)
+ {
+
+ }
+
+ public void AddStyles(BundleContext context)
+ {
+ context.Add("_content/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/libs/abp/css/theme.css");
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/Branding.razor b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/Branding.razor
new file mode 100644
index 0000000000..4810320664
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/Branding.razor
@@ -0,0 +1,3 @@
+@using Volo.Abp.Ui.Branding
+@inject IBrandingProvider BrandingProvider
+@BrandingProvider.AppName
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/FirstLevelNavMenuItem.razor b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/FirstLevelNavMenuItem.razor
index 574a684213..31090ac092 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/FirstLevelNavMenuItem.razor
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/FirstLevelNavMenuItem.razor
@@ -15,7 +15,7 @@
{
if (MenuItem.Icon.StartsWith("fa"))
{
-
+
}
}
@MenuItem.DisplayName
@@ -32,7 +32,7 @@ else
{
if (MenuItem.Icon.StartsWith("fa"))
{
-
+
}
}
@MenuItem.DisplayName
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LanguageSwitch.razor b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LanguageSwitch.razor
index 8c7b348f7d..6b93618f44 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LanguageSwitch.razor
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LanguageSwitch.razor
@@ -5,23 +5,23 @@
@inject IJSRuntime JsRuntime
@if (_otherLanguages != null && _otherLanguages.Any())
{
-
-
+
+
@_currentLanguage.DisplayName
-
-
+
+
@foreach (var language in _otherLanguages)
{
- @language.DisplayName
+ @language.DisplayName
}
-
-
+
+
}
@code {
private IReadOnlyList _otherLanguages;
private LanguageInfo _currentLanguage;
- protected override async Task OnInitializedAsync()
+ protected async override Task OnInitializedAsync()
{
var selectedLanguageName = await JsRuntime.InvokeAsync(
"localStorage.getItem",
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor
index e4767cc031..58d4d5753e 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor
@@ -26,9 +26,6 @@
@menuItem.DisplayName
}
}
-
- @UiLocalizer["ManageYourAccount"]
- Logout
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor.cs
index 17718dbcfb..494eca34ab 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/LoginDisplay.razor.cs
@@ -1,50 +1,33 @@
using System;
using System.Threading.Tasks;
-using Localization.Resources.AbpUi;
using Microsoft.AspNetCore.Components;
using Microsoft.AspNetCore.Components.Routing;
-using Microsoft.Extensions.Localization;
-using Microsoft.Extensions.Options;
-using Volo.Abp.Http.Client;
using Volo.Abp.UI.Navigation;
namespace Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme.Themes.Basic
{
public partial class LoginDisplay : IDisposable
{
- [Inject]
- protected IMenuManager MenuManager { get; set; }
-
- [Inject]
- protected IStringLocalizer UiLocalizer { get; set; }
-
[Inject]
- protected IOptions RemoteServiceOptions { get; set; }
-
- protected ApplicationMenu Menu { get; set; }
+ protected IMenuManager MenuManager { get; set; }
- protected string ServerUrl { get; set; }
- protected string ServerAccountUrl { get; set; }
+ protected ApplicationMenu Menu { get; set; }
- protected override async Task OnInitializedAsync()
+ protected async override Task OnInitializedAsync()
{
Menu = await MenuManager.GetAsync(StandardMenus.User);
-
- ServerUrl = RemoteServiceOptions.Value.RemoteServices.Default?.BaseUrl?.TrimEnd('/');
- ServerAccountUrl = ServerUrl + "/Account/Manage?returnUrl=" + Navigation.Uri;
-
+
Navigation.LocationChanged += OnLocationChanged;
}
protected virtual void OnLocationChanged(object sender, LocationChangedEventArgs e)
{
- ServerAccountUrl = ServerUrl + "/Account/Manage?returnUrl=" + Navigation.Uri;
StateHasChanged();
}
-
+
public void Dispose()
{
Navigation.LocationChanged -= OnLocationChanged;
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/MainLayout.razor b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/MainLayout.razor
index 45496a7984..e85e899123 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/MainLayout.razor
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly.BasicTheme/Themes/Basic/MainLayout.razor
@@ -1,9 +1,7 @@
@inherits LayoutComponentBase
-@using Volo.Abp.Ui.Branding
-@inject IBrandingProvider BrandingProvider
-
-
+
+
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/AbpBlazorMessageLocalizerHelper.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/AbpBlazorMessageLocalizerHelper.cs
new file mode 100644
index 0000000000..2a6fb19a11
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/AbpBlazorMessageLocalizerHelper.cs
@@ -0,0 +1,48 @@
+using System.Collections.Generic;
+using System.Linq;
+using JetBrains.Annotations;
+using Microsoft.Extensions.Localization;
+
+namespace Volo.Abp.AspNetCore.Components.WebAssembly
+{
+ public class AbpBlazorMessageLocalizerHelper
+ {
+ private readonly IStringLocalizer stringLocalizer;
+
+ public AbpBlazorMessageLocalizerHelper(IStringLocalizer stringLocalizer)
+ {
+ this.stringLocalizer = stringLocalizer;
+ }
+
+ public string Localize(string message, [CanBeNull] IEnumerable arguments)
+ {
+ try
+ {
+ return arguments?.Count() > 0
+ ? stringLocalizer[message, LocalizeMessageArguments(arguments)?.ToArray()]
+ : stringLocalizer[message];
+ }
+ catch
+ {
+ return stringLocalizer[message];
+ }
+ }
+
+ private IEnumerable LocalizeMessageArguments(IEnumerable arguments)
+ {
+ foreach (var argument in arguments)
+ {
+ // first try to localize with "DisplayName:{Name}"
+ var localization = stringLocalizer[$"DisplayName:{argument}"];
+
+ if (localization.ResourceNotFound)
+ {
+ // then try to localize with just "{Name}"
+ localization = stringLocalizer[argument];
+ }
+
+ yield return localization;
+ }
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertManager.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertManager.cs
index 290c36848a..02c1251f16 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertManager.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertManager.cs
@@ -1,4 +1,5 @@
-using Volo.Abp.DependencyInjection;
+using Volo.Abp.AspNetCore.Components.Alerts;
+using Volo.Abp.DependencyInjection;
namespace Volo.Abp.AspNetCore.Components.WebAssembly.Alerts
{
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/IAlertManager.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/IAlertManager.cs
deleted file mode 100644
index f30fe4c262..0000000000
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/IAlertManager.cs
+++ /dev/null
@@ -1,7 +0,0 @@
-namespace Volo.Abp.AspNetCore.Components.WebAssembly.Alerts
-{
- public interface IAlertManager
- {
- AlertList Alerts { get; }
- }
-}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/ComponentsWebAssemblyBundleContributor.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/ComponentsWebAssemblyBundleContributor.cs
new file mode 100644
index 0000000000..9f05ff24d1
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/ComponentsWebAssemblyBundleContributor.cs
@@ -0,0 +1,17 @@
+using Volo.Abp.Bundling;
+
+namespace Volo.Abp.AspNetCore.Components.WebAssembly
+{
+ public class ComponentsWebAssemblyBundleContributor : IBundleContributor
+ {
+ public void AddScripts(BundleContext context)
+ {
+ context.Add("_content/Volo.Abp.AspNetCore.Components.WebAssembly/libs/abp/js/abp.js");
+ }
+
+ public void AddStyles(BundleContext context)
+ {
+
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/ExceptionHandling/UserExceptionInformer.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/ExceptionHandling/UserExceptionInformer.cs
index c409c38319..ca3a2be9c5 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/ExceptionHandling/UserExceptionInformer.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/ExceptionHandling/UserExceptionInformer.cs
@@ -1,7 +1,7 @@
using System;
-using System.Threading.Tasks;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Logging.Abstractions;
+using Volo.Abp.AspNetCore.Components.Messages;
using Volo.Abp.AspNetCore.ExceptionHandling;
using Volo.Abp.DependencyInjection;
using Volo.Abp.Http;
@@ -31,11 +31,11 @@ namespace Volo.Abp.AspNetCore.Components.WebAssembly.ExceptionHandling
if (errorInfo.Details.IsNullOrEmpty())
{
//TODO: Should we introduce MessageService.Error (sync) method instead of such a usage (without await)..?
- MessageService.ErrorAsync(errorInfo.Message);
+ MessageService.Error(errorInfo.Message);
}
else
{
- MessageService.ErrorAsync(errorInfo.Details, errorInfo.Message);
+ MessageService.Error(errorInfo.Details, errorInfo.Message);
}
}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/IUiMessageService.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/IUiMessageService.cs
deleted file mode 100644
index fd70cc40e1..0000000000
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/IUiMessageService.cs
+++ /dev/null
@@ -1,18 +0,0 @@
-using System;
-using System.Threading.Tasks;
-
-namespace Volo.Abp.AspNetCore.Components.WebAssembly
-{
- public interface IUiMessageService
- {
- Task InfoAsync(string message, string title = null, Action options = null);
-
- Task SuccessAsync(string message, string title = null, Action options = null);
-
- Task WarnAsync(string message, string title = null, Action options = null);
-
- Task ErrorAsync(string message, string title = null, Action options = null);
-
- Task ConfirmAsync(string message, string title = null, Action options = null);
- }
-}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/IUiNotificationService.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/IUiNotificationService.cs
deleted file mode 100644
index a9816dc325..0000000000
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/IUiNotificationService.cs
+++ /dev/null
@@ -1,12 +0,0 @@
-using System.Threading.Tasks;
-
-namespace Volo.Abp.AspNetCore.Components.WebAssembly
-{
- public interface IUiNotificationService
- {
- Task Info(string message);
- Task Success(string message);
- Task Warn(string message);
- Task Error(string message);
- }
-}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/SimpleUiMessageService.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Messages/SimpleUiMessageService.cs
similarity index 54%
rename from framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/SimpleUiMessageService.cs
rename to framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Messages/SimpleUiMessageService.cs
index 8f43b3fdcd..3218912ad0 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/SimpleUiMessageService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Messages/SimpleUiMessageService.cs
@@ -1,9 +1,10 @@
using System;
using System.Threading.Tasks;
using Microsoft.JSInterop;
+using Volo.Abp.AspNetCore.Components.Messages;
using Volo.Abp.DependencyInjection;
-namespace Volo.Abp.AspNetCore.Components.WebAssembly
+namespace Volo.Abp.AspNetCore.Components.WebAssembly.Messages
{
public class SimpleUiMessageService : IUiMessageService, ITransientDependency
{
@@ -14,27 +15,27 @@ namespace Volo.Abp.AspNetCore.Components.WebAssembly
JsRuntime = jsRuntime;
}
- public async Task InfoAsync(string message, string title = null, Action options = null)
+ public async Task Info(string message, string title = null, Action options = null)
{
await JsRuntime.InvokeVoidAsync("alert", message);
}
- public async Task SuccessAsync(string message, string title = null, Action options = null)
+ public async Task Success(string message, string title = null, Action options = null)
{
await JsRuntime.InvokeVoidAsync("alert", message);
}
- public async Task WarnAsync(string message, string title = null, Action options = null)
+ public async Task Warn(string message, string title = null, Action options = null)
{
await JsRuntime.InvokeVoidAsync("alert", message);
}
- public async Task ErrorAsync(string message, string title = null, Action options = null)
+ public async Task Error(string message, string title = null, Action options = null)
{
await JsRuntime.InvokeVoidAsync("alert", message);
}
- public async Task ConfirmAsync(string message, string title = null, Action options = null)
+ public async Task Confirm(string message, string title = null, Action options = null)
{
return await JsRuntime.InvokeAsync("confirm", message);
}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/NullUiNotificationService.cs b/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/NullUiNotificationService.cs
deleted file mode 100644
index 85ee10287c..0000000000
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/NullUiNotificationService.cs
+++ /dev/null
@@ -1,27 +0,0 @@
-using System.Threading.Tasks;
-using Volo.Abp.DependencyInjection;
-
-namespace Volo.Abp.AspNetCore.Components.WebAssembly
-{
- public class NullUiNotificationService : IUiNotificationService, ITransientDependency
- {
- public Task Info(string message)
- {
- return Task.CompletedTask;
- }
-
- public Task Success(string message)
- {
- return Task.CompletedTask;
- }
-
- public Task Warn(string message)
- {
- return Task.CompletedTask;
- }
- public Task Error(string message)
- {
- return Task.CompletedTask;
- }
- }
-}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo.Abp.AspNetCore.Components.csproj b/framework/src/Volo.Abp.AspNetCore.Components/Volo.Abp.AspNetCore.Components.csproj
index 973f49118c..2023908431 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components/Volo.Abp.AspNetCore.Components.csproj
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo.Abp.AspNetCore.Components.csproj
@@ -21,7 +21,7 @@
-
+
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/AbpComponentBase.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/AbpComponentBase.cs
index a4b4c7740e..0c78a3c784 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/AbpComponentBase.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/AbpComponentBase.cs
@@ -5,6 +5,9 @@ using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Localization;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Logging.Abstractions;
+using Volo.Abp.AspNetCore.Components.Alerts;
+using Volo.Abp.AspNetCore.Components.Messages;
+using Volo.Abp.AspNetCore.Components.Notifications;
using Volo.Abp.Localization;
using Volo.Abp.ObjectMapping;
using Volo.Abp.Users;
@@ -53,6 +56,17 @@ namespace Volo.Abp.AspNetCore.Components
protected ICurrentUser CurrentUser => LazyGetRequiredService(ref _currentUser);
private ICurrentUser _currentUser;
+ protected IUiMessageService Message => LazyGetNonScopedRequiredService(ref _message);
+ private IUiMessageService _message;
+
+ protected IUiNotificationService Notify => LazyGetNonScopedRequiredService(ref _notify);
+ private IUiNotificationService _notify;
+
+ protected IAlertManager AlertManager => LazyGetNonScopedRequiredService(ref _alertManager);
+ private IAlertManager _alertManager;
+
+ protected AlertList Alerts => AlertManager.Alerts;
+
protected IObjectMapper ObjectMapper
{
get
@@ -90,6 +104,21 @@ namespace Volo.Abp.AspNetCore.Components
return reference;
}
+ protected TService LazyGetNonScopedRequiredService(ref TService reference) => LazyGetNonScopedRequiredService(typeof(TService), ref reference);
+
+ protected TRef LazyGetNonScopedRequiredService(Type serviceType, ref TRef reference)
+ {
+ if (reference == null)
+ {
+ reference = (TRef)NonScopedServices.GetRequiredService(serviceType);
+ }
+
+ return reference;
+ }
+
+ [Inject]
+ protected IServiceProvider NonScopedServices { get; set; }
+
protected virtual IStringLocalizer CreateLocalizer()
{
if (LocalizationResource != null)
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertList.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertList.cs
similarity index 82%
rename from framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertList.cs
rename to framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertList.cs
index 47a26ad9ea..866095a9b9 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertList.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertList.cs
@@ -1,6 +1,11 @@
-using System.Collections.ObjectModel;
+using System;
+using System.Collections.Generic;
+using System.Collections.ObjectModel;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
-namespace Volo.Abp.AspNetCore.Components.WebAssembly.Alerts
+namespace Volo.Abp.AspNetCore.Components.Alerts
{
public class AlertList : ObservableCollection
{
@@ -29,4 +34,4 @@ namespace Volo.Abp.AspNetCore.Components.WebAssembly.Alerts
Add(new AlertMessage(AlertType.Success, text, title, dismissible));
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertMessage.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertMessage.cs
similarity index 92%
rename from framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertMessage.cs
rename to framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertMessage.cs
index e8a8f3529a..1b6a7dd21d 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertMessage.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertMessage.cs
@@ -1,6 +1,6 @@
using JetBrains.Annotations;
-namespace Volo.Abp.AspNetCore.Components.WebAssembly.Alerts
+namespace Volo.Abp.AspNetCore.Components.Alerts
{
public class AlertMessage
{
@@ -27,4 +27,4 @@ namespace Volo.Abp.AspNetCore.Components.WebAssembly.Alerts
Dismissible = dismissible;
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertType.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertType.cs
similarity index 52%
rename from framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertType.cs
rename to framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertType.cs
index 6a61a9995f..4703158b12 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/Alerts/AlertType.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/AlertType.cs
@@ -1,4 +1,10 @@
-namespace Volo.Abp.AspNetCore.Components.WebAssembly.Alerts
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace Volo.Abp.AspNetCore.Components.Alerts
{
public enum AlertType
{
@@ -12,4 +18,4 @@
Light,
Dark
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/IAlertManager.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/IAlertManager.cs
new file mode 100644
index 0000000000..542b576b01
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Alerts/IAlertManager.cs
@@ -0,0 +1,13 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace Volo.Abp.AspNetCore.Components.Alerts
+{
+ public interface IAlertManager
+ {
+ AlertList Alerts { get; }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/IUiMessageService.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/IUiMessageService.cs
new file mode 100644
index 0000000000..4f43cb90d6
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/IUiMessageService.cs
@@ -0,0 +1,18 @@
+using System;
+using System.Threading.Tasks;
+
+namespace Volo.Abp.AspNetCore.Components.Messages
+{
+ public interface IUiMessageService
+ {
+ Task Info(string message, string title = null, Action options = null);
+
+ Task Success(string message, string title = null, Action options = null);
+
+ Task Warn(string message, string title = null, Action options = null);
+
+ Task Error(string message, string title = null, Action options = null);
+
+ Task Confirm(string message, string title = null, Action options = null);
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageEventArgs.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageEventArgs.cs
similarity index 94%
rename from framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageEventArgs.cs
rename to framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageEventArgs.cs
index 4431d3c819..51ef8ed558 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageEventArgs.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageEventArgs.cs
@@ -1,7 +1,7 @@
using System;
using System.Threading.Tasks;
-namespace Volo.Abp.AspNetCore.Components.WebAssembly
+namespace Volo.Abp.AspNetCore.Components.Messages
{
public class UiMessageEventArgs : EventArgs
{
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageOptions.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageOptions.cs
similarity index 96%
rename from framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageOptions.cs
rename to framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageOptions.cs
index 59b2e8f400..9da4bcad60 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageOptions.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageOptions.cs
@@ -1,4 +1,4 @@
-namespace Volo.Abp.AspNetCore.Components.WebAssembly
+namespace Volo.Abp.AspNetCore.Components.Messages
{
///
/// Options to override message dialog appearance.
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageType.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageType.cs
similarity index 81%
rename from framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageType.cs
rename to framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageType.cs
index 5e3a77a3c5..677297055e 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.WebAssembly/Volo/Abp/AspNetCore/Components/WebAssembly/UiMessageType.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Messages/UiMessageType.cs
@@ -1,4 +1,4 @@
-namespace Volo.Abp.AspNetCore.Components.WebAssembly
+namespace Volo.Abp.AspNetCore.Components.Messages
{
///
/// Defines the possible ui message types with predefined actions.
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/IUiNotificationService.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/IUiNotificationService.cs
new file mode 100644
index 0000000000..dfbde9b1eb
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/IUiNotificationService.cs
@@ -0,0 +1,16 @@
+using System;
+using System.Threading.Tasks;
+
+namespace Volo.Abp.AspNetCore.Components.Notifications
+{
+ public interface IUiNotificationService
+ {
+ Task Info(string message, string title = null, Action options = null);
+
+ Task Success(string message, string title = null, Action options = null);
+
+ Task Warn(string message, string title = null, Action options = null);
+
+ Task Error(string message, string title = null, Action options = null);
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/NullUiNotificationService.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/NullUiNotificationService.cs
new file mode 100644
index 0000000000..ed8217348e
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/NullUiNotificationService.cs
@@ -0,0 +1,28 @@
+using System;
+using System.Threading.Tasks;
+using Volo.Abp.DependencyInjection;
+
+namespace Volo.Abp.AspNetCore.Components.Notifications
+{
+ public class NullUiNotificationService : IUiNotificationService, ITransientDependency
+ {
+ public Task Info(string message, string title = null, Action options = null)
+ {
+ return Task.CompletedTask;
+ }
+
+ public Task Success(string message, string title = null, Action options = null)
+ {
+ return Task.CompletedTask;
+ }
+
+ public Task Warn(string message, string title = null, Action options = null)
+ {
+ return Task.CompletedTask;
+ }
+ public Task Error(string message, string title = null, Action options = null)
+ {
+ return Task.CompletedTask;
+ }
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationEventArgs.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationEventArgs.cs
new file mode 100644
index 0000000000..30d497d2ab
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationEventArgs.cs
@@ -0,0 +1,23 @@
+using System;
+
+namespace Volo.Abp.AspNetCore.Components.Notifications
+{
+ public class UiNotificationEventArgs : EventArgs
+ {
+ public UiNotificationEventArgs(UiNotificationType notificationType, string message, string title, UiNotificationOptions options)
+ {
+ NotificationType = notificationType;
+ Message = message;
+ Title = title;
+ Options = options;
+ }
+
+ public UiNotificationType NotificationType { get; set; }
+
+ public string Message { get; }
+
+ public string Title { get; }
+
+ public UiNotificationOptions Options { get; }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationOptions.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationOptions.cs
new file mode 100644
index 0000000000..eb1c5d7102
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationOptions.cs
@@ -0,0 +1,20 @@
+using Volo.Abp.Localization;
+
+namespace Volo.Abp.AspNetCore.Components.Notifications
+{
+ ///
+ /// Options to override notification appearance.
+ ///
+ public class UiNotificationOptions
+ {
+ ///
+ /// Custom text for the Ok button.
+ ///
+ public ILocalizableString OkButtonText { get; set; }
+
+ ///
+ /// Custom icon for the Ok button.
+ ///
+ public object OkButtonIcon { get; set; }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationType.cs b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationType.cs
new file mode 100644
index 0000000000..3547d451e9
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components/Volo/Abp/AspNetCore/Components/Notifications/UiNotificationType.cs
@@ -0,0 +1,10 @@
+namespace Volo.Abp.AspNetCore.Components.Notifications
+{
+ public enum UiNotificationType
+ {
+ Info,
+ Success,
+ Warning,
+ Error,
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteFeatureChecker.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteFeatureChecker.cs
index f769f512a5..4cba01bea2 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteFeatureChecker.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteFeatureChecker.cs
@@ -13,7 +13,7 @@ namespace Volo.Abp.AspNetCore.Mvc.Client
ConfigurationClient = configurationClient;
}
- public override async Task GetOrNullAsync(string name)
+ public async override Task GetOrNullAsync(string name)
{
var configuration = await ConfigurationClient.GetAsync();
return configuration.Features.Values.GetOrDefault(name);
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemotePermissionChecker.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemotePermissionChecker.cs
index 18f5272e7b..5ce1ea9edc 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemotePermissionChecker.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemotePermissionChecker.cs
@@ -1,4 +1,4 @@
-using System.Security.Claims;
+using System.Security.Claims;
using System.Threading.Tasks;
using Volo.Abp.Authorization.Permissions;
using Volo.Abp.DependencyInjection;
@@ -26,5 +26,25 @@ namespace Volo.Abp.AspNetCore.Mvc.Client
/* This provider always works for the current principal. */
return await IsGrantedAsync(name);
}
+
+ public async Task IsGrantedAsync(string[] names)
+ {
+ var result = new MultiplePermissionGrantResult();
+ var configuration = await ConfigurationClient.GetAsync();
+ foreach (var name in names)
+ {
+ result.Result.Add(name, configuration.Auth.GrantedPolicies.ContainsKey(name) ?
+ PermissionGrantResult.Granted :
+ PermissionGrantResult.Undefined);
+ }
+
+ return result;
+ }
+
+ public async Task IsGrantedAsync(ClaimsPrincipal claimsPrincipal, string[] names)
+ {
+ /* This provider always works for the current principal. */
+ return await IsGrantedAsync(names);
+ }
}
}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteSettingProvider.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteSettingProvider.cs
index db3ea6e9a6..89db44b43d 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteSettingProvider.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.Client.Common/Volo/Abp/AspNetCore/Mvc/Client/RemoteSettingProvider.cs
@@ -14,13 +14,19 @@ namespace Volo.Abp.AspNetCore.Mvc.Client
{
ConfigurationClient = configurationClient;
}
-
+
public async Task GetOrNullAsync(string name)
{
var configuration = await ConfigurationClient.GetAsync();
return configuration.Setting.Values.GetOrDefault(name);
}
+ public async Task> GetAllAsync(string[] names)
+ {
+ var configuration = await ConfigurationClient.GetAsync();
+ return names.Select(x => new SettingValue(x, configuration.Setting.Values.GetOrDefault(x))).ToList();
+ }
+
public async Task> GetAllAsync()
{
var configuration = await ConfigurationClient.GetAsync();
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Contracts/Volo/Abp/AspNetCore/Mvc/ApplicationConfigurations/ObjectExtending/ExtensionPropertyUiDto.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Contracts/Volo/Abp/AspNetCore/Mvc/ApplicationConfigurations/ObjectExtending/ExtensionPropertyUiDto.cs
index 2fb841ec5a..caa5d67957 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Contracts/Volo/Abp/AspNetCore/Mvc/ApplicationConfigurations/ObjectExtending/ExtensionPropertyUiDto.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.Contracts/Volo/Abp/AspNetCore/Mvc/ApplicationConfigurations/ObjectExtending/ExtensionPropertyUiDto.cs
@@ -8,5 +8,6 @@ namespace Volo.Abp.AspNetCore.Mvc.ApplicationConfigurations.ObjectExtending
public ExtensionPropertyUiTableDto OnTable { get; set; }
public ExtensionPropertyUiFormDto OnCreateForm { get; set; }
public ExtensionPropertyUiFormDto OnEditForm { get; set; }
+ public ExtensionPropertyUiLookupDto Lookup { get; set; }
}
}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Contracts/Volo/Abp/AspNetCore/Mvc/ApplicationConfigurations/ObjectExtending/ExtensionPropertyUiLookupDto.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Contracts/Volo/Abp/AspNetCore/Mvc/ApplicationConfigurations/ObjectExtending/ExtensionPropertyUiLookupDto.cs
new file mode 100644
index 0000000000..035178df1e
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.Contracts/Volo/Abp/AspNetCore/Mvc/ApplicationConfigurations/ObjectExtending/ExtensionPropertyUiLookupDto.cs
@@ -0,0 +1,14 @@
+using System;
+
+namespace Volo.Abp.AspNetCore.Mvc.ApplicationConfigurations.ObjectExtending
+{
+ [Serializable]
+ public class ExtensionPropertyUiLookupDto
+ {
+ public string Url { get; set; }
+ public string ResultListPropertyName { get; set; }
+ public string DisplayPropertyName { get; set; }
+ public string ValuePropertyName { get; set; }
+ public string FilterParamName { get; set; }
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Breadcrumb/AbpBreadcrumbTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Breadcrumb/AbpBreadcrumbTagHelperService.cs
index 4b433ac495..e1b4469f8c 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Breadcrumb/AbpBreadcrumbTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Breadcrumb/AbpBreadcrumbTagHelperService.cs
@@ -8,7 +8,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Breadcrumb
{
public class AbpBreadcrumbTagHelperService : AbpTagHelperService
{
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
output.TagName = "nav";
output.Attributes.Add("aria-label", "breadcrumb");
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Carousel/AbpCarouselTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Carousel/AbpCarouselTagHelperService.cs
index c3672db7f5..e4a277ba66 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Carousel/AbpCarouselTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Carousel/AbpCarouselTagHelperService.cs
@@ -19,7 +19,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Carousel
L = localizer;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
output.TagName = "div";
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionItemTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionItemTagHelperService.cs
index 6e532cb71d..0cd400d43c 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionItemTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionItemTagHelperService.cs
@@ -9,7 +9,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Collapse
{
public class AbpAccordionItemTagHelperService : AbpTagHelperService
{
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
SetRandomIdIfNotProvided();
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionTagHelperService.cs
index c7f93e87cc..7651309e1a 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpAccordionTagHelperService.cs
@@ -17,7 +17,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Collapse
HtmlGenerator = htmlGenerator;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
SetRandomIdIfNotProvided();
@@ -25,7 +25,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Collapse
output.TagMode = TagMode.StartTagAndEndTag;
output.Attributes.AddClass("accordion");
output.Attributes.Add("id",TagHelper.Id);
-
+
var items = InitilizeFormGroupContentsContext(context, output);
await output.GetChildContentAsync();
@@ -62,4 +62,4 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Collapse
}
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpCollapseBodyTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpCollapseBodyTagHelperService.cs
index 10e49b92a7..198c5b91e6 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpCollapseBodyTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Collapse/AbpCollapseBodyTagHelperService.cs
@@ -6,7 +6,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Collapse
{
public class AbpCollapseBodyTagHelperService : AbpTagHelperService
{
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
output.TagName = "div";
output.Attributes.AddClass("collapse");
@@ -27,4 +27,4 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Collapse
output.Content.SetHtmlContent(innerContent);
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Dropdown/AbpDropdownButtonTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Dropdown/AbpDropdownButtonTagHelperService.cs
index 673204b302..52ea6ace85 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Dropdown/AbpDropdownButtonTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Dropdown/AbpDropdownButtonTagHelperService.cs
@@ -24,7 +24,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Dropdown
_serviceProvider = serviceProvider;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
var content = await output.GetChildContentAsync();
@@ -60,7 +60,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Dropdown
protected virtual async Task GetMainButtonAsync(TagHelperContext context, TagHelperOutput output, TagHelperContent content)
{
var abpButtonTagHelper = _serviceProvider.GetRequiredService();
-
+
abpButtonTagHelper.Icon = TagHelper.Icon;
abpButtonTagHelper.Text = TagHelper.Text;
abpButtonTagHelper.IconType = TagHelper.IconType;
@@ -121,7 +121,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Dropdown
{"aria-haspopup", "true"},
{"aria-expanded", "false"},
};
-
+
attributes.AddClass("dropdown-toggle");
attributes.AddClass("dropdown-toggle-split");
@@ -142,4 +142,4 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Dropdown
return buttonTag;
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpDynamicformTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpDynamicformTagHelperService.cs
index 4504906566..cd8c7e6f22 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpDynamicformTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpDynamicformTagHelperService.cs
@@ -31,7 +31,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Form
_serviceProvider = serviceProvider;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
var list = InitilizeFormGroupContentsContext(context, output);
@@ -144,7 +144,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Form
}
}
}
-
+
protected virtual void RemoveFormGroupItemsNotInModel(TagHelperContext context, TagHelperOutput output, List items)
{
var models = GetModels(context, output);
@@ -303,4 +303,4 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Form
return explorer.GetAttribute() != null;
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelperService.cs
index ff5b085df5..adb50d7382 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelperService.cs
@@ -26,7 +26,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Form
_tagHelperLocalizer = tagHelperLocalizer;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
var (innerHtml, isCheckBox) = await GetFormInputGroupAsHtmlAsync(context, output);
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelper.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelper.cs
index 4df9733d83..2f4440b904 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelper.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelper.cs
@@ -21,6 +21,20 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Form
[HtmlAttributeName("required-symbol")]
public bool DisplayRequiredSymbol { get; set; } = true;
+ public string AutocompleteApiUrl { get; set; }
+
+ public string AutocompleteItemsPropertyName { get; set; }
+
+ public string AutocompleteDisplayPropertyName { get; set; }
+
+ public string AutocompleteValuePropertyName { get; set; }
+
+ public string AutocompleteFilterParamName { get; set; }
+
+ public string AutocompleteSelectedItemName { get; set; }
+
+ public string AutocompleteSelectedItemValue { get; set; }
+
public AbpSelectTagHelper(AbpSelectTagHelperService tagHelperService)
: base(tagHelperService)
{
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelperService.cs
index 017d58dc17..7741807aae 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpSelectTagHelperService.cs
@@ -1,4 +1,5 @@
-using Microsoft.AspNetCore.Mvc.Rendering;
+using Microsoft.AspNetCore.Mvc.Diagnostics;
+using Microsoft.AspNetCore.Mvc.Rendering;
using Microsoft.AspNetCore.Mvc.TagHelpers;
using Microsoft.AspNetCore.Mvc.ViewFeatures;
using Microsoft.AspNetCore.Razor.TagHelpers;
@@ -35,7 +36,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Form
_stringLocalizerFactory = stringLocalizerFactory;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
var innerHtml = await GetFormInputGroupAsHtmlAsync(context, output);
@@ -78,20 +79,47 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Form
var selectTagHelper = new SelectTagHelper(_generator)
{
For = TagHelper.AspFor,
- Items = GetSelectItems(context, output),
ViewContext = TagHelper.ViewContext
};
+ if (TagHelper.AutocompleteApiUrl.IsNullOrEmpty())
+ {
+ selectTagHelper.Items = GetSelectItems(context, output);
+ }
+ else
+ {
+ selectTagHelper.Items = new SelectListItem[]
+ {
+ new SelectListItem(TagHelper.AutocompleteSelectedItemName,TagHelper.AutocompleteSelectedItemValue,true)
+ };
+ }
+
var selectTagHelperOutput = await selectTagHelper.ProcessAndGetOutputAsync(GetInputAttributes(context, output), context, "select", TagMode.StartTagAndEndTag);
selectTagHelperOutput.Attributes.AddClass("form-control");
selectTagHelperOutput.Attributes.AddClass(GetSize(context, output));
AddDisabledAttribute(selectTagHelperOutput);
AddInfoTextId(selectTagHelperOutput);
+ AddAutocompleteAttributes(selectTagHelperOutput);
return selectTagHelperOutput;
}
+ protected virtual void AddAutocompleteAttributes(TagHelperOutput output)
+ {
+ if (!TagHelper.AutocompleteApiUrl.IsNullOrEmpty())
+ {
+ output.Attributes.AddClass("auto-complete-select");
+ output.Attributes.Add("data-autocomplete-api-url", TagHelper.AutocompleteApiUrl);
+ output.Attributes.Add("data-autocomplete-items-property", TagHelper.AutocompleteItemsPropertyName);
+ output.Attributes.Add("data-autocomplete-display-property", TagHelper.AutocompleteDisplayPropertyName);
+ output.Attributes.Add("data-autocomplete-value-property", TagHelper.AutocompleteValuePropertyName);
+ output.Attributes.Add("data-autocomplete-filter-param-name", TagHelper.AutocompleteFilterParamName);
+ output.Attributes.Add("data-autocomplete-selected-item-name", TagHelper.AutocompleteSelectedItemName);
+ output.Attributes.Add("data-autocomplete-selected-item-value", TagHelper.AutocompleteSelectedItemValue);
+ }
+ }
+
protected virtual void AddDisabledAttribute(TagHelperOutput inputTagHelperOutput)
{
var disabledAttribute = TagHelper.AspFor.ModelExplorer.GetAttribute();
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Modal/AbpModalTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Modal/AbpModalTagHelperService.cs
index 78d664b562..9086d8def4 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Modal/AbpModalTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Modal/AbpModalTagHelperService.cs
@@ -1,22 +1,22 @@
using Microsoft.AspNetCore.Html;
using Microsoft.AspNetCore.Mvc.Rendering;
-using Microsoft.AspNetCore.Razor.TagHelpers;
-using System.Text;
+using Microsoft.AspNetCore.Razor.TagHelpers;
+using System.Text;
using System.Threading.Tasks;
-namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Modal
-{
- public class AbpModalTagHelperService : AbpTagHelperService
- {
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
- {
- output.TagName = null;
-
- var childContent = await output.GetChildContentAsync();
-
- SetContent(context, output, childContent);
- }
-
+namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Modal
+{
+ public class AbpModalTagHelperService : AbpTagHelperService
+ {
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ {
+ output.TagName = null;
+
+ var childContent = await output.GetChildContentAsync();
+
+ SetContent(context, output, childContent);
+ }
+
protected virtual void SetContent(TagHelperContext context, TagHelperOutput output, TagHelperContent childContent)
{
var modalContent = GetModalContentElement(context, output, childContent);
@@ -24,16 +24,16 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Modal
var modal = GetModal(context, output, modalDialog);
output.Content.SetHtmlContent(modal);
- }
-
+ }
+
protected virtual TagBuilder GetModalContentElement(TagHelperContext context, TagHelperOutput output, TagHelperContent childContent)
{
var element = new TagBuilder("div");
element.AddCssClass(GetModalContentClasses());
element.InnerHtml.SetHtmlContent(childContent);
return element;
- }
-
+ }
+
protected virtual TagBuilder GetModalDialogElement(TagHelperContext context, TagHelperOutput output, IHtmlContent innerHtml)
{
var element = new TagBuilder("div");
@@ -41,8 +41,8 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Modal
element.Attributes.Add("role", "document");
element.InnerHtml.SetHtmlContent(innerHtml);
return element;
- }
-
+ }
+
protected virtual TagBuilder GetModal(TagHelperContext context, TagHelperOutput output, IHtmlContent innerHtml)
{
var element = new TagBuilder("div");
@@ -61,37 +61,37 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Modal
element.InnerHtml.SetHtmlContent(innerHtml);
return element;
- }
-
- protected virtual string GetModalClasses()
- {
- return "modal fade";
- }
-
- protected virtual string GetModalDialogClasses()
- {
- var classNames = new StringBuilder("modal-dialog");
-
- if (TagHelper.Centered ?? false)
- {
- classNames.Append(" ");
- classNames.Append("modal-dialog-centered");
- }
-
- if (TagHelper.Size != AbpModalSize.Default)
- {
- classNames.Append(" ");
- classNames.Append(TagHelper.Size.ToClassName());
- }
-
- return classNames.ToString();
- }
-
- protected virtual string GetModalContentClasses()
- {
- return "modal-content";
- }
-
+ }
+
+ protected virtual string GetModalClasses()
+ {
+ return "modal fade";
+ }
+
+ protected virtual string GetModalDialogClasses()
+ {
+ var classNames = new StringBuilder("modal-dialog");
+
+ if (TagHelper.Centered ?? false)
+ {
+ classNames.Append(" ");
+ classNames.Append("modal-dialog-centered");
+ }
+
+ if (TagHelper.Size != AbpModalSize.Default)
+ {
+ classNames.Append(" ");
+ classNames.Append(TagHelper.Size.ToClassName());
+ }
+
+ return classNames.ToString();
+ }
+
+ protected virtual string GetModalContentClasses()
+ {
+ return "modal-content";
+ }
+
protected virtual string GetDataAttributes()
{
if (TagHelper.Static == true)
@@ -107,6 +107,6 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Modal
{
builder.Attributes.Add("data-backdrop", "static");
}
- }
- }
-}
\ No newline at end of file
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Pagination/AbpPaginationTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Pagination/AbpPaginationTagHelperService.cs
index 71f3bfe247..148f196aad 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Pagination/AbpPaginationTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Pagination/AbpPaginationTagHelperService.cs
@@ -21,9 +21,9 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Pagination
private readonly IStringLocalizerFactory _stringLocalizerFactory;
public AbpPaginationTagHelperService(
- IHtmlGenerator generator,
- HtmlEncoder encoder,
- IAbpTagHelperLocalizer tagHelperLocalizer,
+ IHtmlGenerator generator,
+ HtmlEncoder encoder,
+ IAbpTagHelperLocalizer tagHelperLocalizer,
IStringLocalizerFactory stringLocalizerFactory)
{
_generator = generator;
@@ -32,7 +32,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Pagination
_stringLocalizerFactory = stringLocalizerFactory;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
if (TagHelper.Model.ShownItemsCount <= 0)
{
@@ -199,4 +199,4 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Pagination
}
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabDropdownTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabDropdownTagHelperService.cs
index 8746e4a48b..cdaf55d87c 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabDropdownTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabDropdownTagHelperService.cs
@@ -9,7 +9,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Tab
{
public class AbpTabDropdownTagHelperService : AbpTagHelperService
{
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
if (string.IsNullOrWhiteSpace(TagHelper.Name))
{
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabTagHelperService.cs
index 9080e49d83..c4d4b654a9 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabTagHelperService.cs
@@ -9,7 +9,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Tab
{
public class AbpTabTagHelperService : AbpTagHelperService
{
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
SetPlaceholderForNameIfNotProvided();
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabsTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabsTagHelperService.cs
index 92e50990a4..8046af8ccd 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabsTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Tab/AbpTabsTagHelperService.cs
@@ -20,7 +20,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Tab
HtmlGenerator = htmlGenerator;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
SetRandomNameIfNotProvided();
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/BundlerBase.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/BundlerBase.cs
index 299e8f834e..41ff63468e 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/BundlerBase.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/BundlerBase.cs
@@ -11,6 +11,8 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling
{
public abstract class BundlerBase : IBundler, ITransientDependency
{
+ private static string[] _minFileSuffixes = {"min", "prod"};
+
public ILogger Logger { get; set; }
protected IWebContentFileProvider WebContentFileProvider { get; }
@@ -82,7 +84,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling
{
var fileContent = GetFileInfo(context, fileName).ReadAsString();
var nonMinifiedSize = fileContent.Length;
-
+
Logger.LogDebug($"- {fileName} ({nonMinifiedSize} bytes) - non minified, minifying...");
fileContent = Minifier.Minify(
@@ -110,14 +112,32 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling
protected virtual bool IsMinFile(string fileName)
{
- return fileName.EndsWith($".min.{FileExtension}", StringComparison.InvariantCultureIgnoreCase);
+ foreach (var suffix in _minFileSuffixes)
+ {
+ if (fileName.EndsWith($".{suffix}.{FileExtension}", StringComparison.InvariantCultureIgnoreCase))
+ {
+ return true;
+ }
+ }
+
+ return false;
}
protected virtual IFileInfo GetMinFileInfoOrNull(string file)
{
- var fileInfo = WebContentFileProvider.GetFileInfo($"{file.RemovePostFix($".{FileExtension}")}.min.{FileExtension}");
+ foreach (var suffix in _minFileSuffixes)
+ {
+ var fileInfo = WebContentFileProvider.GetFileInfo(
+ $"{file.RemovePostFix($".{FileExtension}")}.{suffix}.{FileExtension}"
+ );
+
+ if (fileInfo.Exists)
+ {
+ return fileInfo;
+ }
+ }
- return fileInfo.Exists ? fileInfo : null;
+ return null;
}
protected virtual string ProcessBeforeAddingToTheBundle(IBundlerContext context, string filePath, string fileContent)
@@ -125,4 +145,4 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling
return fileContent;
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleItemTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleItemTagHelperService.cs
index eca80c5fcf..0ffb5b9049 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleItemTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleItemTagHelperService.cs
@@ -16,7 +16,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling.TagHelpers
ResourceService = resourceService;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
var tagHelperItems = context.Items.GetOrDefault(AbpTagHelperConsts.ContextBundleItemListKey) as List;
if (tagHelperItems != null)
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleTagHelperService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleTagHelperService.cs
index f0c76d5cc4..9158f33b90 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleTagHelperService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpBundleTagHelperService.cs
@@ -16,7 +16,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling.TagHelpers
ResourceService = resourceService;
}
- public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
+ public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
await ResourceService.ProcessAsync(
TagHelper.ViewContext,
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperScriptService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperScriptService.cs
index 6efbb793e7..9e4e77a6dd 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperScriptService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperScriptService.cs
@@ -38,7 +38,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling.TagHelpers
);
}
- protected override async Task> GetBundleFilesAsync(string bundleName)
+ protected async override Task> GetBundleFilesAsync(string bundleName)
{
return await BundleManager.GetScriptBundleFilesAsync(bundleName);
}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperStyleService.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperStyleService.cs
index 8612e4f8b4..a9d9b54fca 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperStyleService.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bundling/Volo/Abp/AspNetCore/Mvc/UI/Bundling/TagHelpers/AbpTagHelperStyleService.cs
@@ -38,7 +38,7 @@ namespace Volo.Abp.AspNetCore.Mvc.UI.Bundling.TagHelpers
);
}
- protected override async Task> GetBundleFilesAsync(string bundleName)
+ protected async override Task> GetBundleFilesAsync(string bundleName)
{
return await BundleManager.GetStyleBundleFilesAsync(bundleName);
}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.MultiTenancy/Volo/Abp/AspNetCore/Mvc/UI/MultiTenancy/Localization/de-DE.json b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.MultiTenancy/Volo/Abp/AspNetCore/Mvc/UI/MultiTenancy/Localization/de-DE.json
new file mode 100644
index 0000000000..3b1f12c29d
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.MultiTenancy/Volo/Abp/AspNetCore/Mvc/UI/MultiTenancy/Localization/de-DE.json
@@ -0,0 +1,12 @@
+{
+ "culture": "de-DE",
+ "texts": {
+ "GivenTenantIsNotAvailable": "Der angegebene Mandant ist nicht verfügbar: {0}",
+ "Tenant": "Mandant",
+ "Switch": "wechseln",
+ "Name": "Name",
+ "SwitchTenantHint": "Lassen Sie das Namensfeld leer, um zur Host-Seite zu wechseln.",
+ "SwitchTenant": "Mandant wechseln",
+ "NotSelected": "Nicht ausgewählt"
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Packages/Volo/Abp/AspNetCore/Mvc/UI/Packages/VeeValidate/VeeValidateScriptContributor.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Packages/Volo/Abp/AspNetCore/Mvc/UI/Packages/VeeValidate/VeeValidateScriptContributor.cs
new file mode 100644
index 0000000000..7bc67110d2
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Packages/Volo/Abp/AspNetCore/Mvc/UI/Packages/VeeValidate/VeeValidateScriptContributor.cs
@@ -0,0 +1,15 @@
+using Volo.Abp.AspNetCore.Mvc.UI.Bundling;
+using Volo.Abp.AspNetCore.Mvc.UI.Packages.Vue;
+using Volo.Abp.Modularity;
+
+namespace Volo.Abp.AspNetCore.Mvc.UI.Packages.VeeValidate
+{
+ [DependsOn(typeof(VueScriptContributor))]
+ public class VeeValidateScriptContributor : BundleContributor
+ {
+ public override void ConfigureBundle(BundleConfigurationContext context)
+ {
+ context.Files.Add("/libs/vee-validate/vee-validate.full.min.js");
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Packages/Volo/Abp/AspNetCore/Mvc/UI/Packages/Vue/VueScriptContributor.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Packages/Volo/Abp/AspNetCore/Mvc/UI/Packages/Vue/VueScriptContributor.cs
new file mode 100644
index 0000000000..c5740cd94b
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Packages/Volo/Abp/AspNetCore/Mvc/UI/Packages/Vue/VueScriptContributor.cs
@@ -0,0 +1,12 @@
+using Volo.Abp.AspNetCore.Mvc.UI.Bundling;
+
+namespace Volo.Abp.AspNetCore.Mvc.UI.Packages.Vue
+{
+ public class VueScriptContributor : BundleContributor
+ {
+ public override void ConfigureBundle(BundleConfigurationContext context)
+ {
+ context.Files.Add("/libs/vue/vue.min.js");
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic/Themes/Basic/Components/Menu/Default.cshtml b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic/Themes/Basic/Components/Menu/Default.cshtml
index 064a3756c4..6c5d22877e 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic/Themes/Basic/Components/Menu/Default.cshtml
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic/Themes/Basic/Components/Menu/Default.cshtml
@@ -11,7 +11,7 @@
if (menuItem.Url != null)
{