mirror of https://github.com/abpframework/abp.git
923 changed files with 23011 additions and 5768 deletions
@ -1,24 +1,24 @@ |
|||
name: Merge branch rel-4.2 with rel-4.1 |
|||
name: Merge branch dev with rel-4.2 |
|||
on: |
|||
push: |
|||
branches: |
|||
- rel-4.1 |
|||
- rel-4.2 |
|||
jobs: |
|||
merge-rel-4-2-with-rel-4-1: |
|||
merge-dev-with-rel-4-2: |
|||
runs-on: ubuntu-latest |
|||
steps: |
|||
- uses: actions/checkout@v2 |
|||
with: |
|||
ref: rel-4.2 |
|||
ref: dev |
|||
- name: Reset promotion branch |
|||
run: | |
|||
git fetch origin rel-4.1:rel-4.1 |
|||
git reset --hard rel-4.1 |
|||
git fetch origin rel-4.2:rel-4.2 |
|||
git reset --hard rel-4.2 |
|||
- name: Create Pull Request |
|||
uses: peter-evans/create-pull-request@v3 |
|||
with: |
|||
branch: auto-merge/rel-4-1/${{github.run_number}} |
|||
title: Merge branch rel-4.2 with rel-4.1 |
|||
body: This PR generated automatically to merge rel-4.2 with rel-4.1. Please review the changed files before merging to prevent any errors that may occur. |
|||
branch: auto-merge/rel-4-2/${{github.run_number}} |
|||
title: Merge branch dev with rel-4.2 |
|||
body: This PR generated automatically to merge dev with rel-4.2. Please review the changed files before merging to prevent any errors that may occur. |
|||
reviewers: ${{github.actor}} |
|||
token: ${{ github.token }} |
|||
|
|||
@ -0,0 +1,14 @@ |
|||
{ |
|||
"culture": "en-GB", |
|||
"texts": { |
|||
"Account": "ABP Account - Login & Register | ABP.IO", |
|||
"Welcome": "Welcome", |
|||
"UseOneOfTheFollowingLinksToContinue": "Use one of the following links to continue", |
|||
"FrameworkHomePage": "Framework home page", |
|||
"FrameworkDocumentation": "Framework documentation", |
|||
"OfficialBlog": "Official blog", |
|||
"CommercialHomePage": "Commercial home page", |
|||
"CommercialSupportWebSite": "Commercial support web site", |
|||
"CommunityWebSite": "ABP community web site" |
|||
} |
|||
} |
|||
@ -0,0 +1,202 @@ |
|||
{ |
|||
"culture": "en-GB", |
|||
"texts": { |
|||
"Permission:Organizations": "Organisations", |
|||
"Permission:Manage": "Manage Organisations", |
|||
"Permission:DiscountRequests": "Discount Requests", |
|||
"Permission:DiscountManage": "Manage Discount Requests", |
|||
"Permission:Disable": "Disable", |
|||
"Permission:Enable": "Enable", |
|||
"Permission:EnableSendEmail": "Enable Send Email", |
|||
"Permission:SendEmail": "Send Email", |
|||
"Permission:NpmPackages": "NPM Packages", |
|||
"Permission:NugetPackages": "Nuget Packages", |
|||
"Permission:Maintenance": "Maintenance", |
|||
"Permission:Maintain": "Maintain", |
|||
"Permission:ClearCaches": "Clear caches", |
|||
"Permission:Modules": "Modules", |
|||
"Permission:Packages": "Packages", |
|||
"Permission:Edit": "Edit", |
|||
"Permission:Delete": "Delete", |
|||
"Permission:Create": "Create", |
|||
"Permission:Accounting": "Accounting", |
|||
"Permission:Accounting:Quotation": "Quotation", |
|||
"Permission:Accounting:Invoice": "Invoice", |
|||
"Menu:Organizations": "Organisations", |
|||
"Menu:Accounting": "Accounting", |
|||
"Menu:Packages": "Packages", |
|||
"Menu:DiscountRequests": "Discount Requests", |
|||
"NpmPackageDeletionWarningMessage": "This NPM Package will be deleted. Please confirm this?", |
|||
"NugetPackageDeletionWarningMessage": "This Nuget Package will be deleted. Please confirm this?", |
|||
"ModuleDeletionWarningMessage": "This Module will be deleted. Please confirm this?", |
|||
"Name": "Name", |
|||
"DisplayName": "Display name", |
|||
"ShortDescription": "Short description", |
|||
"NameFilter": "Name", |
|||
"CreationTime": "Creation time", |
|||
"IsPro": "Is pro", |
|||
"ShowOnModuleList": "Show in module list", |
|||
"EfCoreConfigureMethodName": "Configure method name", |
|||
"IsProFilter": "Is pro", |
|||
"ApplicationType": "Application type", |
|||
"Target": "Target", |
|||
"TargetFilter": "Target", |
|||
"ModuleClass": "Module class", |
|||
"NugetPackageTarget.DomainShared": "Domain Shared", |
|||
"NugetPackageTarget.Domain": "Domain", |
|||
"NugetPackageTarget.Application": "Application", |
|||
"NugetPackageTarget.ApplicationContracts": "Application Contracts", |
|||
"NugetPackageTarget.HttpApi": "Http Api", |
|||
"NugetPackageTarget.HttpApiClient": "Http Api Client", |
|||
"NugetPackageTarget.Web": "Web", |
|||
"NugetPackageTarget.EntityFrameworkCore": "Delete Entity Framework Core", |
|||
"NugetPackageTarget.MongoDB": "MongoDB", |
|||
"Edit": "Edit", |
|||
"Delete": "Delete", |
|||
"Refresh": "Refresh", |
|||
"NpmPackages": "NPM Packages", |
|||
"NugetPackages": "Nuget Packages", |
|||
"NpmPackageCount": "NPM Package Count", |
|||
"NugetPackageCount": "Nuget Package Count", |
|||
"Module": "Modules", |
|||
"ModuleInfo": "Module info", |
|||
"CreateANpmPackage": "Create a NPM package", |
|||
"CreateAModule": "Create a module", |
|||
"CreateANugetPackage": "Create a Nuget package", |
|||
"AddNew": "Add new", |
|||
"PackageAlreadyExist{0}": "\"{0}\" package has already been added.", |
|||
"ModuleAlreadyExist{0}": "\"{0}\" module has already been added.", |
|||
"ClearCache": "Clear cache", |
|||
"SuccessfullyCleared": "Successfully cleared", |
|||
"Menu:NpmPackages": "NPM Packages", |
|||
"Menu:Modules": "Modules", |
|||
"Menu:Maintenance": "Maintenance", |
|||
"Menu:NugetPackages": "Nuget Packages", |
|||
"CreateAnOrganization": "Create an organisation", |
|||
"Organizations": "Organisations", |
|||
"LongName": "Long name", |
|||
"LicenseType": "License type", |
|||
"MissingLicenseTypeField": "The license type field is required!", |
|||
"LicenseStartTime": "License start time", |
|||
"LicenseEndTime": "License end time", |
|||
"AllowedDeveloperCount": "Allowed developer count", |
|||
"UserNameOrEmailAddress": "Username or email address", |
|||
"AddOwner": "Add owner", |
|||
"UserName": "Username", |
|||
"Email": "Email", |
|||
"Developers": "Developers", |
|||
"AddDeveloper": "Add developer", |
|||
"Create": "Create", |
|||
"UserNotFound": "User not found", |
|||
"{0}WillBeRemovedFromDevelopers": "{0} Will be removed from developers, please confirm?", |
|||
"{0}WillBeRemovedFromOwners": "{0} Will be removed from owners, please confirm?", |
|||
"Computers": "Computers", |
|||
"UniqueComputerId": "Unique computer id", |
|||
"LastSeenDate": "Last seen date", |
|||
"{0}Computer{1}WillBeRemovedFromRecords": "Computer of {0} ({1}) will be removed from records", |
|||
"OrganizationDeletionWarningMessage": "Organisation will be deleted", |
|||
"DeletingLastOwnerWarningMessage": "An organisation must have at least one owner, therefore you cannot remove this owner", |
|||
"This{0}AlreadyExistInThisOrganization": "{0} already exist in this organisation", |
|||
"AreYouSureYouWantToDeleteAllComputers": "Are you sure you want to delete all computers?", |
|||
"DeleteAll": "Delete all", |
|||
"DoYouWantToCreateNewUser": "Do you want to create new user?", |
|||
"MasterModules": "Master Modules", |
|||
"OrganizationName": "Organisation name", |
|||
"CreationDate": "Creation date", |
|||
"LicenseStartDate": "License start date", |
|||
"LicenseEndDate": "License end date", |
|||
"OrganizationNamePlaceholder": "Organisation name...", |
|||
"TotalQuestionCountPlaceholder": "Total question count...", |
|||
"RemainingQuestionCountPlaceholder": "Remaining question count...", |
|||
"LicenseTypePlaceholder": "License type...", |
|||
"CreationDatePlaceholder": "Creation date...", |
|||
"LicenseStartDatePlaceholder": "License start date...", |
|||
"LicenseEndDatePlaceholder": "License end date...", |
|||
"UsernameOrEmail": "Username or email", |
|||
"UsernameOrEmailPlaceholder": "Username or email...", |
|||
"Member": "Member", |
|||
"PurchaseOrderNo": "Purchase Order No.", |
|||
"QuotationDate": "Quotation date", |
|||
"CompanyName": "Company name", |
|||
"CompanyAddress": "Company address", |
|||
"Price": "Price", |
|||
"DiscountText": "Discount text", |
|||
"DiscountQuantity": "Discount quantity", |
|||
"DiscountPrice": "Discount price", |
|||
"Quotation": "Quotation", |
|||
"ExtraText": "Extra text", |
|||
"ExtraAmount": "Extra Amount", |
|||
"DownloadQuotation": "Download Quotation", |
|||
"Invoice": "Invoice", |
|||
"TaxNumber": "Tax No.", |
|||
"InvoiceNumber": "Invoice No.", |
|||
"InvoiceDate": "Invoice Date", |
|||
"InvoiceNote": "Invoice Note", |
|||
"Quantity": "Quantity", |
|||
"AddProduct": "Add Product", |
|||
"AddProductWarning": "You need to add a Product!", |
|||
"TotalPrice": "Total Price", |
|||
"Generate": "Generate", |
|||
"MissingQuantityField": "The Quantity field is required!", |
|||
"MissingPriceField": "The Price field is required!", |
|||
"CodeUsageStatus": "Status", |
|||
"Country": "Country", |
|||
"DeveloperCount": "Developer Count", |
|||
"RequestCode": "Request Code", |
|||
"WebSite": "Website", |
|||
"GithubUsername": "Github Username", |
|||
"PhoneNumber": "Phone Number", |
|||
"ProjectDescription": "Project Description", |
|||
"Referrer": "Referrer", |
|||
"DiscountRequests": "Discount Request", |
|||
"Copylink": "Copy Link", |
|||
"Disable": "Disable", |
|||
"Enable": "Enable", |
|||
"EnableSendEmail": "Enable Send Email", |
|||
"SendEmail": "Send Email", |
|||
"SuccessfullyDisabled": "Successfully disabled", |
|||
"SuccessfullyEnabled": "Successfully enabled", |
|||
"EmailSent": "Email sent", |
|||
"SuccessfullySent": "Successfully sent", |
|||
"SuccessfullyDeleted": "Successfully deleted", |
|||
"DiscountRequestDeletionWarningMessage": "Discount request will be deleted", |
|||
"BusinessType": "Business Type", |
|||
"TotalQuestionCount": "Total question count", |
|||
"RemainingQuestionCount": "Remaining question count", |
|||
"TotalQuestionMustBeGreaterWarningMessage": "TotalQuestionCount must be greater than RemainingQuestionCount!", |
|||
"QuestionCountsMustBeGreaterThanZero": "TotalQuestionCount and RemainingQuestionCount must be zero or greater than zero!", |
|||
"UnlimitedQuestionCount": "Unlimited question count", |
|||
"Notes": "Notes", |
|||
"Menu:Community": "Community", |
|||
"Menu:Articles": "Articles", |
|||
"Wait": "Wait", |
|||
"Approve": "Approve", |
|||
"Reject": "Reject", |
|||
"Details": "Details", |
|||
"Url": "Url", |
|||
"Title": "Title", |
|||
"ContentSource": "Content source", |
|||
"Status": "Status", |
|||
"ReadArticle": "Read article", |
|||
"ArticleHasBeenWaiting": "Article has been waiting", |
|||
"ArticleHasBeenApproved": "Article has been approved", |
|||
"ArticleHasBeenRejected": "Article has been rejected", |
|||
"Permission:Community": "Community", |
|||
"Permission:CommunityArticle": "Article", |
|||
"Link": "Link", |
|||
"Enum:ContentSource:0": "Github", |
|||
"Enum:ContentSource:1": "External", |
|||
"Enum:Status:0": "Waiting", |
|||
"Enum:Status:1": "Author name", |
|||
"Enum:Status:2": "Approved", |
|||
"Summary": "Summary", |
|||
"AuthorName": "Author", |
|||
"CoverImage": "Cover Image", |
|||
"RemoveCacheConfirmationMessage": "Are you sure you want remove the cache for \"{0}\" article?", |
|||
"SuccessfullyRemoved": "Successfully cleared", |
|||
"RemoveCache": "Remove Cache", |
|||
"Language": "Language", |
|||
"Optional": "Optional", |
|||
"CreateArticleLanguageInfo": "The language in which the article is written" |
|||
} |
|||
} |
|||
@ -0,0 +1,35 @@ |
|||
{ |
|||
"culture": "en-GB", |
|||
"texts": { |
|||
"Volo.AbpIo.Domain:010004": "Maximum member count reached!", |
|||
"Volo.AbpIo.Domain:010005": "Maximum owner count reached!", |
|||
"Volo.AbpIo.Domain:010006": "This user is already an owner in this organisation!", |
|||
"Volo.AbpIo.Domain:010007": "This user is already a developer in this organisation!", |
|||
"Volo.AbpIo.Domain:010008": "Allowed Developer Count can not be less then current developer count!", |
|||
"Volo.AbpIo.Domain:010009": "Allowed Developer Count can not be less then 0!", |
|||
"Volo.AbpIo.Domain:010010": "Maximum mac address count has been exceeded!", |
|||
"Volo.AbpIo.Domain:010011": "Personal license can't have more than 1 developer!", |
|||
"Volo.AbpIo.Domain:010012": "License can't be extended one month after license expires!", |
|||
"Volo.AbpIo.Domain:020001": "Couldn't delete this NPM Package because \"{NugetPackages}\" Nuget Packages are dependent to this package.", |
|||
"Volo.AbpIo.Domain:020002": "Couldn't delete this NPM Package because \"{Modules}\" Modules are using this package.", |
|||
"Volo.AbpIo.Domain:020003": "Couldn't delete this NPM Package because \"{Modules}\" Modules are using this package and \"{NugetPackages}\" Nuget Packages are dependent to this package.", |
|||
"Volo.AbpIo.Domain:020004": "Couldn't delete this Nuget Package because \"{Modules}\" Modules are using this package.", |
|||
"WantToLearn?": "Want to learn?", |
|||
"ReadyToGetStarted?": "Ready to get started?", |
|||
"JoinOurCommunity": "Join our community", |
|||
"GetStartedUpper": "GET STARTED", |
|||
"ForkMeOnGitHub": "Fork me on GitHub", |
|||
"Features": "Features", |
|||
"GetStarted": "Get Started", |
|||
"Documents": "Documents", |
|||
"Community": "Community", |
|||
"ContributionGuide": "Contribution Guide", |
|||
"Blog": "Blog", |
|||
"Commercial": "Commercial", |
|||
"MyAccount": "My account", |
|||
"Permission:License": "License", |
|||
"Permission:UserInfo": "User info", |
|||
"SeeDocuments": "See Documents", |
|||
"Samples": "Samples" |
|||
} |
|||
} |
|||
@ -0,0 +1,39 @@ |
|||
{ |
|||
"culture": "en-GB", |
|||
"texts": { |
|||
"OrganizationManagement": "Organisation Management", |
|||
"OrganizationList": "Organisation list", |
|||
"Volo.AbpIo.Commercial:010003": "You are not owner of this organisation!", |
|||
"OrganizationNotFoundMessage": "Organisation not found!", |
|||
"DeveloperCount": "Allocated / total developers", |
|||
"QuestionCount": "Remaining / total questions", |
|||
"Unlimited": "Unlimited", |
|||
"Owners": "Owners", |
|||
"AddMember": "Add member", |
|||
"AddOwner": "Add owner", |
|||
"AddDeveloper": "Add developer", |
|||
"UserName": "Username", |
|||
"Name": "Name", |
|||
"EmailAddress": "Email address", |
|||
"Developers": "Developers", |
|||
"LicenseType": "License type", |
|||
"Manage": "Manage", |
|||
"StartDate": "Start date", |
|||
"EndDate": "End date", |
|||
"Modules": "Modules", |
|||
"LicenseExtendMessage": "Your license end date is extended to {0}", |
|||
"LicenseUpgradeMessage": "Your license is upgraded to {0}", |
|||
"LicenseAddDeveloperMessage": "{0} developers added to your license", |
|||
"Volo.AbpIo.Commercial:010004": "Cannot find the specified user! The user must have already registered.", |
|||
"MyOrganizations": "My organisations", |
|||
"ApiKey": "API key", |
|||
"UserNameNotFound": "There is no user with username {0}", |
|||
"SuccessfullyAddedToNewsletter": "Thank you for subscribing to our newsletter!", |
|||
"MyProfile": "My Profile", |
|||
"EmailNotValid": "Please enter a valid email address.", |
|||
"JoinOurMarketingNewsletter": "Join our marketing newsletter", |
|||
"WouldLikeToReceiveMarketingMaterials": "I would like to receive marketing materials e.g. product deals & special offers.", |
|||
"StartUsingYourLicenseNow": "Start using your license now!", |
|||
"WelcomePage": "Welcome Page" |
|||
} |
|||
} |
|||
@ -0,0 +1,106 @@ |
|||
{ |
|||
"culture": "en-GB", |
|||
"texts": { |
|||
"Permission:CommunityArticle": "Community Article", |
|||
"Permission:Edit": "Edit", |
|||
"Waiting": "Waiting", |
|||
"Approved": "Approved", |
|||
"Rejected": "Rejected", |
|||
"Wait": "Wait", |
|||
"Approve": "Approve", |
|||
"Reject": "Reject", |
|||
"ReadArticle": "Read Article", |
|||
"Status": "Status", |
|||
"ContentSource": "Content Source", |
|||
"Details": "Details", |
|||
"Url": "Url", |
|||
"Title": "Title", |
|||
"CreationTime": "Creation time", |
|||
"Save": "Save", |
|||
"SameUrlAlreadyExist": "The Url already exists. If you want to add this article, you should change the url!", |
|||
"UrlIsNotValid": "Url is not valid.", |
|||
"UrlNotFound": "Url not found.", |
|||
"UrlContentNotFound": "Url content not found.", |
|||
"Summary": "Summary", |
|||
"MostRead": "Most Read", |
|||
"Latest": "Latest", |
|||
"ContributeAbpCommunity": "Contribute to the ABP Community", |
|||
"SubmitYourArticle": "Submit Your Article", |
|||
"ContributionGuide": "Contribution Guide", |
|||
"BugReport": "Bug Report", |
|||
"SeeAllArticles": "See All Articles", |
|||
"WelcomeToABPCommunity!": "Welcome to the ABP Community!", |
|||
"MyProfile": "My Profile", |
|||
"MyOrganizations": "My Organisations", |
|||
"EmailNotValid": "Please enter a valid email address.", |
|||
"FeatureRequest": "Feature Request", |
|||
"CreateArticleTitleInfo": "Title of the article to be shown on the article list.", |
|||
"CreateArticleUrlInfo": "Original GitHub/External URL of the article.", |
|||
"CreateArticleSummaryInfo": "A short summary of the article to be shown on the article list.", |
|||
"CreateArticleCoverInfo": "To create an effective article, add a cover photo and an upload 16:9 aspect ratio pictures for the best view (Maximum file size: 1MB)", |
|||
"ThisExtensionIsNotAllowed": "The extension is not allowed.", |
|||
"TheFileIsTooLarge": "The file is too large.", |
|||
"GoToTheArticle": "Go to the Article", |
|||
"Contribute": "Contribute", |
|||
"OverallProgress": "Overall Progress", |
|||
"Done": "Done", |
|||
"Open": "Open", |
|||
"Closed": "Closed", |
|||
"LatestQuestionOnThe": "Latest Question on the", |
|||
"Stackoverflow": "Stackoverflow", |
|||
"Votes": "votes", |
|||
"Answer": "Answer", |
|||
"Views": "views", |
|||
"Answered": "Answered", |
|||
"WaitingForYourAnswer": "Waiting for your answer", |
|||
"Asked": "asked", |
|||
"AllQuestions": "All Questions", |
|||
"NextVersion": "Next Version", |
|||
"MilestoneErrorMessage": "Couldn't get the current milestone details from Github.", |
|||
"QuestionItemErrorMessage": "Couldn't get the latest question details from Stackoverflow.", |
|||
"Oops": "Oops!", |
|||
"CreateArticleSuccessMessage": "The Article has been successfully submitted. It will be published once it has been reviewed by the site admin.", |
|||
"ChooseCoverImage": "Choose a cover image...", |
|||
"CoverImage": "Cover Image", |
|||
"ShareYourExperiencesWithTheABPFramework": "Share your experiences with the ABP Framework!", |
|||
"Optional": "Optional", |
|||
"UpdateUserWebSiteInfo": "Example: https://johndoe.com", |
|||
"UpdateUserTwitterInfo": "Example: johndoe", |
|||
"UpdateUserGithubInfo": "Example: johndoe", |
|||
"UpdateUserLinkedinInfo": "Example: https://www.linkedin.com/...", |
|||
"UpdateUserCompanyInfo": "Example: Volosoft", |
|||
"UpdateUserJobTitleInfo": "Example: Software Developer", |
|||
"UserName": "UserName", |
|||
"Company": "Company", |
|||
"PersonalWebsite": "Personal website", |
|||
"RegistrationDate": "Registration date", |
|||
"Social": "Social", |
|||
"Biography": "Biography", |
|||
"HasNoPublishedArticlesYet": "has no published articles yet", |
|||
"Author": "Author", |
|||
"LatestGithubAnnouncements": "Latest GitHub Announcements", |
|||
"SeeAllAnnouncements": "View all Announcements", |
|||
"LatestBlogPost": "Latest Blog Post", |
|||
"Edit": "Edit", |
|||
"ProfileImageChange": "Change the profile image", |
|||
"BlogItemErrorMessage": "Couldn't get the latest blog post details from ABP.", |
|||
"PlannedReleaseDate": "Planned release date", |
|||
"CommunityArticleRequestErrorMessage": "Couldn't get the latest article request from GitHub.", |
|||
"ArticleRequestFromGithubIssue": "There aren't any article requests at the moment.", |
|||
"LatestArticles": "Latest Articles", |
|||
"ArticleRequests": "Article Requests", |
|||
"AllArticleRequests": "See all Article requests", |
|||
"SubscribeToTheNewsletter": "Subscribe to the Newsletter", |
|||
"NewsletterEmailDefinition": "Receive information about what's happening in ABP; 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 <a href=\"https://commercial.abp.io/Privacy\">Privacy Policy</a>.", |
|||
"ArticleRequestMessageTitle": "<a href=\"https://github.com/abpframework/abp/issues/new\">Open an issue</a> on the GitHub to request an article/tutorial you want to see on this web site.", |
|||
"ArticleRequestMessageBody": "Here is the list of the requested articles by the Community. Do you want to write a requested article? Please click on the request and join the discussion.", |
|||
"Language": "Language", |
|||
"CreateArticleLanguageInfo": "The language in which the article is written" |
|||
} |
|||
} |
|||
@ -0,0 +1,198 @@ |
|||
{ |
|||
"culture": "en-GB", |
|||
"texts": { |
|||
"GetStarted": "Get Started - Startup Templates", |
|||
"Create": "Create", |
|||
"NewProject": "New Project", |
|||
"DirectDownload": "Direct Download", |
|||
"ProjectName": "Project name", |
|||
"ProjectType": "Project type", |
|||
"DatabaseProvider": "Database provider", |
|||
"NTier": "N-Tier", |
|||
"IncludeUserInterface": "Include user interface", |
|||
"CreateNow": "Create now", |
|||
"TheStartupProject": "The startup project", |
|||
"Tutorial": "Tutorial", |
|||
"UsingCLI": "Using CLI", |
|||
"SeeDetails": "See Details", |
|||
"AbpShortDescription": "ABP Framework is a complete infrastructure to create modern web applications by following the software development best practices and conventions.", |
|||
"SourceCodeUpper": "SOURCE CODE", |
|||
"LatestReleaseLogs": "Latest release logs", |
|||
"Infrastructure": "Infrastructure", |
|||
"Architecture": "Architecture", |
|||
"Modular": "Modular", |
|||
"DontRepeatYourself": "Don’t Repeat Yourself", |
|||
"DeveloperFocused": "Developer Focused", |
|||
"FullStackApplicationInfrastructure": "Full stack application infrastructure.", |
|||
"DomainDrivenDesign": "Domain Driven Design", |
|||
"DomainDrivenDesignExplanation": "Designed and developed based on DDD patterns and principles. Provides a layered model for your application.", |
|||
"Authorization": "Authorisation", |
|||
"AuthorizationExplanation": "Advanced authorisation with user, role and fine-grained permission system. Built on the Microsoft Identity library.", |
|||
"MultiTenancy": "Multi-Tenancy", |
|||
"MultiTenancyExplanationShort": "SaaS applications made easy! Integrated multi-tenancy from database to UI.", |
|||
"CrossCuttingConcerns": "Cross Cutting Concerns", |
|||
"CrossCuttingConcernsExplanationShort": "Complete infrastructure for authorization, validation, exception handling, caching, audit logging, transaction management and more.", |
|||
"BuiltInBundlingMinification": "Built-In Bundling & Minification", |
|||
"BuiltInBundlingMinificationExplanation": "No need to use external tools for bundling & minification. ABP offers a simpler, dynamic, powerful, modular and built-in way!", |
|||
"VirtualFileSystem": "Virtual File System", |
|||
"VirtualFileSystemExplanation": "Embed views, scripts, styles, images... into packages/libraries and reuse them in different applications.", |
|||
"Theming": "Theming", |
|||
"ThemingExplanationShort": "Use and customise the bootstrap-based standard UI theme or create your own.", |
|||
"BootstrapTagHelpersDynamicForms": "Bootstrap Tag Helpers & Dynamic Forms", |
|||
"BootstrapTagHelpersDynamicFormsExplanation": "Instead of manually writing bootstrap components, Use ABP's tag helpers to simplify and take advantage of intellisense. Quickly build UI forms based on a C# model using the dynamic form tag helper.", |
|||
"HTTPAPIsDynamicProxies": "HTTP APIs & Dynamic Proxies", |
|||
"HTTPAPIsDynamicProxiesExplanation": "Automatically expose application services as REST style HTTP APIs, and consume them with dynamic JavaScript and C# proxies.", |
|||
"CompleteArchitectureInfo": "Modern architecture to create maintainable software solutions.", |
|||
"DomainDrivenDesignBasedLayeringModelExplanation": "Helps you to implement a DDD based layered architecture and build a maintainable code base.", |
|||
"DomainDrivenDesignBasedLayeringModelExplanationCont": "Provides startup templates, abstractions, base classes, services, documentation and guides to help you to develop your application based on DDD patterns & principles.", |
|||
"MicroserviceCompatibleModelExplanation": "The core framework & pre-build modules are designed with microservice architecture in mind.", |
|||
"MicroserviceCompatibleModelExplanationCont": "Provides infrastructure, integrations, samples and documentation to implement microservice solutions easier, while it doesn’t bring additional complexity if you want a monolithic application.", |
|||
"ModularInfo": "ABP provides a module system that allows you to develop reusable application modules, tie into application lifecycle events, and express dependencies between core parts of your system.", |
|||
"PreBuiltModulesThemes": "Pre-Built Modules & Themes", |
|||
"PreBuiltModulesThemesExplanation": "Open source and commercial modules & themes are ready to use in your business application.", |
|||
"NuGetNPMPackages": "NuGet & NPM Packages", |
|||
"NuGetNPMPackagesExplanation": "Distributed as NuGet & NPM packages. Easy to install and upgrade.", |
|||
"ExtensibleReplaceable": "Extensible/Replaceable", |
|||
"ExtensibleReplaceableExplanation": "All services & modules are designed extensibility in mind. You can replace services, pages, styles and components.", |
|||
"CrossCuttingConcernsExplanation2": "Keep your codebase smaller so you can maintain focus on the code that’s specific to your business.", |
|||
"CrossCuttingConcernsExplanation3": "Don’t spend time implementing common application requirements on multiple projects.", |
|||
"AuthenticationAuthorization": "Authentication & Authorization", |
|||
"ExceptionHandling": "Exception Handling", |
|||
"Validation": "Validation", |
|||
"DatabaseConnection": "Database Connection", |
|||
"TransactionManagement": "Transaction management", |
|||
"AuditLogging": "Audit Logging", |
|||
"Caching": "Caching", |
|||
"Multitenancy": "Multitenancy", |
|||
"DataFiltering": "Data filtering", |
|||
"ConventionOverConfiguration": "Convention Over Configuration", |
|||
"ConventionOverConfigurationExplanation": "ABP implements common application conventions by default with a minimal or zero configuration.", |
|||
"ConventionOverConfigurationExplanationList1": "Auto registers known services to dependency injection.", |
|||
"ConventionOverConfigurationExplanationList2": "Exposes application services as HTTP APIs by naming conventions.", |
|||
"ConventionOverConfigurationExplanationList3": "Creates dynamic HTTP client proxies for C# and JavaScript.", |
|||
"ConventionOverConfigurationExplanationList4": "Provides default repositories for your entities.", |
|||
"ConventionOverConfigurationExplanationList5": "Manages Unit of Work per web request or application service method.", |
|||
"ConventionOverConfigurationExplanationList6": "Publishes create, update & delete events for your entities.", |
|||
"BaseClasses": "Base Classes", |
|||
"BaseClassesExplanation": "Pre-built base classes for common application patterns.", |
|||
"DeveloperFocusedExplanation": "ABP is for developers.", |
|||
"DeveloperFocusedExplanationCont": "It aims to simplify your daily software development while not restricting you from writing low level code.", |
|||
"SeeAllFeatures": "See All Features", |
|||
"CLI_CommandLineInterface": "CLI (Command Line Interface)", |
|||
"CLI_CommandLineInterfaceExplanation": "Includes a CLI to help you automate the creation of new projects and the addition of new modules.", |
|||
"StartupTemplates": "Startup Templates", |
|||
"StartupTemplatesExplanation": "Various startup templates provide a fully configured solution to jump start your development.", |
|||
"BasedOnFamiliarTools": "Based on Familiar Tools", |
|||
"BasedOnFamiliarToolsExplanation": "Built and integrated with popular tools you already know. Low learning curve, easy adaptation, comfortable development.", |
|||
"ORMIndependent": "ORM Independent", |
|||
"ORMIndependentExplanation": "The core framework is ORM/database independent and can work with any data source. Entity Framework Core and MongoDB providers are already available.", |
|||
"Features": "Explore the ABP Framework Features", |
|||
"ABPCLI": "ABP CLI", |
|||
"Modularity": "Modularity", |
|||
"BootstrapTagHelpers": "Bootstrap Tag Helpers", |
|||
"DynamicForms": "Dynamic Forms", |
|||
"BundlingMinification": "Bundling & Minification", |
|||
"BackgroundJobs": "Background Jobs", |
|||
"BackgroundJobsExplanation": "Define simple classes to execute jobs in the background as queued. Use the built-in job manager or integrate your own. <a href=\"{0}\">Hangfire</a> & <a href=\"{1}\">RabbitMQ</a> integrations are already available.", |
|||
"DDDInfrastructure": "DDD Infrastructure", |
|||
"DomainDrivenDesignInfrastructure": "Domain Driven Design Infrastructure", |
|||
"AutoRESTAPIs": "Auto REST APIs", |
|||
"DynamicClientProxies": "Dynamic Client Proxies", |
|||
"DistributedEventBus": "Distributed Event Bus", |
|||
"DistributedEventBusWithRabbitMQIntegration": "Distributed Event Bus with RabbitMQ Integration", |
|||
"TestInfrastructure": "Test Infrastructure", |
|||
"AuditLoggingEntityHistories": "Audit Logging & Entity Histories", |
|||
"ObjectToObjectMapping": "Object to Object Mapping", |
|||
"ObjectToObjectMappingExplanation": "<a href=\"{0}\">Object to object mapping</a> abstraction with AutoMapper integration.", |
|||
"EmailSMSAbstractions": "Email & SMS Abstractions", |
|||
"EmailSMSAbstractionsWithTemplatingSupport": "Email & SMS Abstractions with Templating Support", |
|||
"Localization": "Localization", |
|||
"SettingManagement": "Setting Management", |
|||
"ExtensionMethods": "Extension Methods", |
|||
"ExtensionMethodsHelpers": "Extension Methods & Helpers", |
|||
"AspectOrientedProgramming": "Aspect Oriented Programming", |
|||
"DependencyInjection": "Dependency Injection", |
|||
"DependencyInjectionByConventions": "Dependency Injection by Conventions", |
|||
"ABPCLIExplanation": "The ABP CLI (Command Line Interface) is a command line tool to perform common operations for ABP based solutions.", |
|||
"ModularityExplanation": "ABP provides a complete infrastructure to build your own application modules those may have entities, services, database integration, APIs, UI components and so on.", |
|||
"MultiTenancyExplanation": "ABP framework not only supports to develop multi-tenant applications, but also makes your code mostly unaware of the multi-tenancy.", |
|||
"MultiTenancyExplanation2": "Can automatically determine the current Tenant and isolate each Tenant's data from each other.", |
|||
"MultiTenancyExplanation3": "Supports single database, database per tenant and hybrid approaches.", |
|||
"MultiTenancyExplanation4": "You focus on your business code and let the framework to handle multi-tenancy on behalf of you.", |
|||
"BootstrapTagHelpersExplanation": "Instead of manually writing the repeating details of bootstrap components, use ABP's tag helpers to simplify and take advantage of intellisense. You can use Bootstrap whenever you need it.", |
|||
"DynamicFormsExplanation": "Dynamic form & input tag helpers can create the complete form from a C# class as the model.", |
|||
"AuthenticationAuthorizationExplanation": "Rich authentication & authorization options integrated to ASP.NET Core Identity & IdentityServer4. Provides an extensible & detailed permission system.", |
|||
"CrossCuttingConcernsExplanation": "Don't repeat yourself when implementing common features. Instead, focus on your business logic and let ABP automate by convention.", |
|||
"DatabaseConnectionTransactionManagement": "Database Connection & Transaction Management", |
|||
"CorrelationIdTracking": "Correlation-Id Tracking", |
|||
"BundlingMinificationExplanation": "ABP offers a simple, dynamic, powerful, modular and built-in bundling & minification system.", |
|||
"VirtualFileSystemnExplanation": "The Virtual File System makes it possible to manage files those do not physically exist on the file system (disk). It's mainly used to embed (js, css, image, cshtml...) files into assemblies and use them like physical files on runtime.", |
|||
"ThemingExplanation": "Theming system allows to develop your application & modules theme independent by defining a set of common base libraries and layouts, based on the latest Bootstrap framework.", |
|||
"DomainDrivenDesignInfrastructureExplanation": "A complete infrastructure to build layered applications based on the Domain Driven Design patterns & principles.", |
|||
"Specification": "Specification", |
|||
"Repository": "Repository", |
|||
"DomainService": "Domain Service", |
|||
"ValueObject": "Value Object", |
|||
"ApplicationService": "Application Service", |
|||
"DataTransferObject": "Data Transfer Object", |
|||
"AggregateRootEntity": "Aggregate Root, Entity", |
|||
"AutoRESTAPIsExplanation": "ABP can automagically configure your application services as API Controllers by convention.", |
|||
"DynamicClientProxiesExplanation": "Easily consume your APIs from JavaScript and C# clients.", |
|||
"DistributedEventBusWithRabbitMQIntegrationExplanation": "Easily publish & consume distributed events using built-in Distributed Event Bus with RabbitMQ integration available.", |
|||
"TestInfrastructureExplanation": "The framework has been developed unit & integration testing in mind. Provides you base classes to make it easier. Startup templates come with pre-configured for testing.", |
|||
"AuditLoggingEntityHistoriesExplanation": "Built-in audit logging for business-critical applications. Request, service, method level audit logging and entity histories with property-level details.", |
|||
"EmailSMSAbstractionsWithTemplatingSupportExplanation": "IEmailSender and ISmsSender abstractions decouples your application logic from the infrastructure. Advanced email template system allows to create & localize email templates and easily use whenever needed.", |
|||
"LocalizationExplanation": "Localization system allows to create resources in plain JSON files and use them to localize your UI. It supports advanced scenarios like inheritance, extensions and JavaScript integration while it is fully compatible with AspNet Core's localization system.", |
|||
"SettingManagementExplanation": "Define settings for your application and get values on runtime based on the current configuration, tenant and user.", |
|||
"ExtensionMethodsHelpersExplanation": "Don't repeat yourself even for trivial code parts. Extensions & helpers for standard types makes your code much cleaner and easy to write.", |
|||
"AspectOrientedProgrammingExplanation": "Provides a comfortable infrastructure to create dynamic proxies and implement Aspect Oriented Programming. Intercept any class and execute your code before & after every method execution.", |
|||
"DependencyInjectionByConventionsExplanation": "No need to register your classes to dependency injection manually. Automatically registers common service types by convention. For other type of services, you can use interfaces and attributes to make it easier and in-place.", |
|||
"DataFilteringExplanation": "Define and use data filters those are automatically applied when you query entities from database. Soft Delete & Multi-Tenant filters are provided out of the box when you implement simple interfaces.", |
|||
"PublishEvents": "Publish Events", |
|||
"HandleEvents": "Handle Events", |
|||
"AndMore": "and more...", |
|||
"Code": "Code", |
|||
"Result": "Result", |
|||
"SeeTheDocumentForMoreInformation": "See the <a href=\"{1}\">{0} document</a> for more information", |
|||
"IndexPageHeroSection": "<span class=\"first-line shine\"><strong>open source</strong></span><span class=\"second-line text-uppercase\">Web Application<br />Framework </span><span class=\"third-line shine2\"><strong>for asp.net core</strong></span>", |
|||
"UiFramework": "UI Framework", |
|||
"EmailAddress": "Email address", |
|||
"Mobile": "Mobile", |
|||
"ReactNative": "React Native", |
|||
"Strong": "Strong", |
|||
"Complete": "Complete", |
|||
"BasedLayeringModel": "Based Layering Model", |
|||
"Microservice": "Microservice", |
|||
"Compatible": "Compatible", |
|||
"MeeTTheABPCommunityInfo": "Our mission is to create an environment where developers can help each other with articles, tutorials, case studies, etc. and meet like-minded people.", |
|||
"JoinTheABPCommunityInfo": "Get involved with a vibrant community and become a contributor to the ABP Framework!", |
|||
"AllArticles": "All Articles", |
|||
"SubmitYourArticle": "Submit Your Article", |
|||
"DynamicClientProxyDocument": "See the dynamic client proxy documentations for <a href=\"{0}\">JavaScript</a> & <a href=\"{1}\">C#</a>.", |
|||
"EmailSMSAbstractionsDocument": "See the <a href=\"{0}\">emailing</a> and <a href=\"{1}\">SMS sending</a> documents for more information.", |
|||
"CreateProjectWizard": "This wizard creates a new project from the startup template which is properly configured to jump start to your project.", |
|||
"TieredOption": "Creates a tiered solution where Web and Http API layers are physically separated. If not checked, creates a layered solution which is less complex and suitable for most scenarios.", |
|||
"SeparateIdentityServerOption": "Separates server side into two applications: First one is for the identity server and the second one is for your server side HTTP API.", |
|||
"UseslatestPreVersion": "Uses latest pre-release version", |
|||
"ReadTheDocumentation": "<span class=\"text-primary\">Read</span><span class=\"text-success\">The Documentation</span>", |
|||
"Documentation": "Documentation", |
|||
"GettingStartedTutorial": "Getting Started Tutorial", |
|||
"ApplicationDevelopmentTutorial": "Application Development Tutorial", |
|||
"TheStartupTemplate": "The Startup Template", |
|||
"InstallABPCLIInfo": "ABP CLI is the fastest way to start a new solution with the ABP framework. Install the ABP CLI using a command line window:", |
|||
"DifferentLevelOfNamespaces": "You can use different level of namespaces; e.g. BookStore, Acme.BookStore or Acme.Retail.BookStore.", |
|||
"ABPCLIExamplesInfo": "<strong>new</strong> command creates a <strong>layered MVC application</strong> with <strong>Entity Framework Core</strong> as the database provider. However, it has additional options. Examples:", |
|||
"SeeCliDocumentForMoreInformation": "See the <a href=\"{0}\">ABP CLI document</a> for more options or select the \"Direct Download\" tab above.", |
|||
"Optional": "Optional", |
|||
"LocalFrameworkRef": "Keep local project reference for the framework packages.", |
|||
"BlobStoring": "BLOB Storing", |
|||
"BlobStoringExplanation": "BLOB Storing system provides an abstraction to work with BLOBs. ABP provides some pre-built storage provider integrations (Azure, AWS, File System, Database, etc.) that you can easily use in your applications.", |
|||
"TextTemplating": "Text Templating", |
|||
"TextTemplatingExplanation": "Text templating is used to dynamically render contents based on a template and a model (a data object). For example, you can use it to create dynamic email contents with a pre-built template.", |
|||
"MultipleUIOptions": "Multiple UI Options", |
|||
"MultipleDBOptions": "Multiple Database Providers", |
|||
"MultipleUIOptionsExplanation": "The core framework is designed as UI independent and can work with any type of UI system, while there are multiple pre-built and integrated options are provided out of the box.", |
|||
"MultipleDBOptionsExplanation": "The framework can work with any data source, while the following providers are officially developed and supported:", |
|||
"SelectLanguage": "Select language" |
|||
} |
|||
} |
|||
@ -0,0 +1,51 @@ |
|||
# ABP.IO Platform 4.1 Final Has Been Released! |
|||
|
|||
[ABP Framework](https://abp.io/) and [ABP Commercial](https://commercial.abp.io/) 4.1 versions have been released today. |
|||
|
|||
## What's New With 4.1? |
|||
|
|||
Since all the new features are already explained in details with the [4.1 RC Announcement Post](https://blog.abp.io/abp/ABP.IO-Platform-v4.1-RC-Has-Been-Released), I will not repeat all the details again. See the [RC Blog Post](https://blog.abp.io/abp/ABP.IO-Platform-v4.1-RC-Has-Been-Released) for all the features and enhancements. |
|||
|
|||
## Creating New Solutions |
|||
|
|||
You can create a new solution with the ABP Framework version 4.1 by either using the `abp new` command or using the **direct download** tab on the [get started page](https://abp.io/get-started). |
|||
|
|||
> See the [getting started document](https://docs.abp.io/en/abp/latest/Getting-Started) for details. |
|||
|
|||
## How to Upgrade an Existing Solution |
|||
|
|||
### Install/Update the ABP CLI |
|||
|
|||
First of all, install the ABP CLI or upgrade to the latest version. |
|||
|
|||
If you haven't installed yet: |
|||
|
|||
```bash |
|||
dotnet tool install -g Volo.Abp.Cli |
|||
``` |
|||
|
|||
To update an existing installation: |
|||
|
|||
```bash |
|||
dotnet tool update -g Volo.Abp.Cli |
|||
``` |
|||
|
|||
### ABP UPDATE Command |
|||
|
|||
[ABP CLI](https://docs.abp.io/en/abp/latest/CLI) provides a handy command to update all the ABP related NuGet and NPM packages in your solution with a single command: |
|||
|
|||
```bash |
|||
abp update |
|||
``` |
|||
|
|||
Run this command in the root folder of your solution. |
|||
|
|||
## ABP Community |
|||
|
|||
We started to get more contributions by the community for the [ABP Community](https://community.abp.io/) contents. Thank you all! |
|||
|
|||
We will be adding **Video Content** sharing system in a short time. We are planning to create short video contents, especially to explore the new features in every release. Again, we will be waiting video contributions by the community :) |
|||
|
|||
## About the Next Versions |
|||
|
|||
Planned preview date for the version **4.2 is January 14, 2021**. See the [Road Map](https://docs.abp.io/en/abp/latest/Road-Map) document and [GitHub Milestones](https://github.com/abpframework/abp/milestones) to learn what's planned for the next versions. We are trying to be clear about the coming features and the next release dates. |
|||
@ -0,0 +1,101 @@ |
|||
# ABP version 4.2 Migration Guide |
|||
|
|||
This version has no breaking changes but there is an important change on the repositories that should be applied for your application for an important performance and scalability gain. |
|||
|
|||
## IRepository.GetQueryableAsync |
|||
|
|||
`IRepository` interface inherits `IQueryable`, so you can directly use the standard LINQ extension methods, like `Where`, `OrderBy`, `First`, `Sum`... etc. |
|||
|
|||
**Example: Using LINQ directly over the repository object** |
|||
|
|||
````csharp |
|||
public class BookAppService : ApplicationService, IBookAppService |
|||
{ |
|||
private readonly IRepository<Book, Guid> _bookRepository; |
|||
|
|||
public BookAppService(IRepository<Book, Guid> bookRepository) |
|||
{ |
|||
_bookRepository = bookRepository; |
|||
} |
|||
|
|||
public async Task DoItInOldWayAsync() |
|||
{ |
|||
//Apply any standard LINQ extension method |
|||
var query = _bookRepository |
|||
.Where(x => x.Price > 10) |
|||
.OrderBy(x => x.Name); |
|||
|
|||
//Execute the query asynchronously |
|||
var books = await AsyncExecuter.ToListAsync(query); |
|||
} |
|||
} |
|||
```` |
|||
|
|||
*See [the documentation](https://docs.abp.io/en/abp/4.2/Repositories#iqueryable-async-operations) if you wonder what is the `AsyncExecuter`.* |
|||
|
|||
**Beginning from the version 4.2, the recommended way is using `IRepository.GetQueryableAsync()` to obtain an `IQueryable`, then use the LINQ extension methods over it.** |
|||
|
|||
**Example: Using the new GetQueryableAsync method** |
|||
|
|||
````csharp |
|||
public async Task DoItInNewWayAsync() |
|||
{ |
|||
//Use GetQueryableAsync to obtain the IQueryable<Book> first |
|||
var queryable = await _bookRepository.GetQueryableAsync(); |
|||
|
|||
//Then apply any standard LINQ extension method |
|||
var query = queryable |
|||
.Where(x => x.Price > 10) |
|||
.OrderBy(x => x.Name); |
|||
|
|||
//Finally, execute the query asynchronously |
|||
var books = await AsyncExecuter.ToListAsync(query); |
|||
} |
|||
```` |
|||
|
|||
ABP may start a database transaction when you get an `IQueryable` (If current [Unit Of Work](https://docs.abp.io/en/abp/latest/Unit-Of-Work) is transactional). In this new way, it is possible to **start the database transaction in an asynchronous way**. Previously, we could not get the advantage of asynchronous while starting the transactions. |
|||
|
|||
> **The new way has a significant performance and scalability gain. The old usage (directly using LINQ over the repositories) will be removed in the next major version (5.0).** You have a lot of time for the change, but we recommend to immediately take the action since the old usage has a big **scalability problem**. |
|||
|
|||
### Actions to Take |
|||
|
|||
* Use the repository's queryable feature as explained before. |
|||
* If you've overridden `CreateFilteredQuery` in a class derived from `CrudAppService`, you should override the `CreateFilteredQueryAsync` instead and remove the `CreateFilteredQuery` in your class. |
|||
* If you've overridden `WithDetails` in your custom repositories, remove it and override `WithDetailsAsync` instead. |
|||
* If you've used `DbContext` or `DbSet` properties in your custom repositories, use `GetDbContextAsync()` and `GetDbSetAsync()` methods instead of them. |
|||
|
|||
You can re-build your solution and check the `Obsolete` warnings to find some of the usages need to change. |
|||
|
|||
#### About IRepository Async Extension Methods |
|||
|
|||
Using IRepository Async Extension Methods has no such a problem. The examples below are pretty fine: |
|||
|
|||
````csharp |
|||
var countAll = await _personRepository |
|||
.CountAsync(); |
|||
|
|||
var count = await _personRepository |
|||
.CountAsync(x => x.Name.StartsWith("A")); |
|||
|
|||
var book1984 = await _bookRepository |
|||
.FirstOrDefaultAsync(x => x.Name == "John"); |
|||
```` |
|||
|
|||
See the [repository documentation](https://docs.abp.io/en/abp/4.2/Repositories#iqueryable-async-operations) to understand the relation between `IQueryable` and asynchronous operations. |
|||
|
|||
## .NET Package Upgrades |
|||
|
|||
ABP uses the latest 5.0.* .NET packages. If your application is using 5.0.0 packages, you may get an error on build. We recommend to depend on the .NET packages like `5.0.*` in the `.csproj` files to use the latest patch versions. |
|||
|
|||
Example: |
|||
|
|||
````xml |
|||
<PackageReference Include="Microsoft.EntityFrameworkCore.Tools" Version="5.0.*" /> |
|||
```` |
|||
|
|||
## Blazorise Library Upgrade |
|||
|
|||
If you are upgrading to 4.2, you also need also upgrade the following packages in your Blazor application; |
|||
|
|||
* `Blazorise.Bootstrap` to `0.9.3-preview6` |
|||
* `Blazorise.Icons.FontAwesome` to `0.9.3-preview6` |
|||
@ -1,5 +1,6 @@ |
|||
# ABP Framework Migration Guides |
|||
|
|||
* [3.3.x to 4.0 Migration Guide](Abp-4_0.md) |
|||
* [2.9.x to 3.0 Migration Guide](../UI/Angular/Migration-Guide-v3.md) |
|||
* [4.x to 4.2](Abp-4_2.md) |
|||
* [3.3.x to 4.0](Abp-4_0.md) |
|||
* [2.9.x to 3.0](../UI/Angular/Migration-Guide-v3.md) |
|||
|
|||
|
|||
@ -0,0 +1,64 @@ |
|||
# Page Alerts |
|||
|
|||
A page alert is useful for displaying an important message to the user. The ABP Framework provides an easy way to show the following alert to the user. |
|||
|
|||
 |
|||
|
|||
You can simply import `PageAlertService` from `@abp/ng.theme.shared` and utilize it as follows: |
|||
|
|||
```js |
|||
import { PageAlertService } from '@abp/ng.theme.shared'; |
|||
|
|||
@Component({ |
|||
// ... |
|||
}) |
|||
export class MyComponent { |
|||
constructor(private service: PageAlertService) {} |
|||
|
|||
showWarning() { |
|||
this.service.show({ |
|||
type: 'warning', |
|||
message: |
|||
'We will have a service interruption between 02:00 AM and 04:00 AM at October 23, 2023!', |
|||
title: 'Service Interruption', |
|||
}); |
|||
} |
|||
} |
|||
``` |
|||
|
|||
## `SHOW` |
|||
|
|||
The method `show` accepts a single object that is type of `PageAlert` |
|||
|
|||
```js |
|||
export interface PageAlert { |
|||
type: 'primary' | 'secondary' | 'success' | 'danger' | 'warning' | 'info' | 'light' | 'dark'; |
|||
message: string; |
|||
title?: string; |
|||
dismissible?: boolean; |
|||
messageLocalizationParams?: string[]; |
|||
titleLocalizationParams?: string[]; |
|||
} |
|||
``` |
|||
|
|||
* `type` (Required): Defines what type of alert will be shown |
|||
* `message` (Required): The message who will be shown, also works with localization as well. |
|||
* `title` (Optional): The title of the message. If it is not provided, the title will be hidden. |
|||
* `dismissible` (Optional): Default is `true`. If enabled, a button on the top right corner will be shown to the users so that they can dismiss the message. |
|||
* `messageLocalizationParams` and `titleLocalizationParams` (Optional): If the message and/or the title is a key for localization service and contains some parameters, these fields could be used to pass those parameters. |
|||
|
|||
### An example with Localization |
|||
|
|||
```typescript |
|||
this.service.show({ |
|||
type: 'danger', |
|||
message: 'AbpAccount::PagerInfo{0}{1}{2}', |
|||
messageLocalizationParams: ['10', '20', '30'], |
|||
title: 'AbpAccount::EntityNotFoundErrorMessage', |
|||
titleLocalizationParams: ['Test', 'id123'], |
|||
}); |
|||
``` |
|||
|
|||
 |
|||
|
|||
|
|||
@ -1,3 +1,380 @@ |
|||
# Angular UI: Testing |
|||
# Unit Testing Angular UI |
|||
|
|||
TODO |
|||
ABP Angular UI is tested like any other Angular application. So, [the guide here](https://angular.io/guide/testing) applies to ABP too. That said, we would like to point out some **unit testing topics specific to ABP Angular applications**. |
|||
|
|||
## Setup |
|||
|
|||
In Angular, unit tests use [Karma](https://karma-runner.github.io/) and [Jasmine](https://jasmine.github.io) by default. Although we like Jest more, we chose not to deviate from these defaults, so **the application template you download will have Karma and Jasmine preconfigured**. You can find the Karma configuration inside the _karma.conf.js_ file in the root folder. You don't have to do anything. Adding a spec file and running `npm test` will work. |
|||
|
|||
## Basics |
|||
|
|||
An over-simplified spec file looks like this: |
|||
|
|||
```js |
|||
import { CoreTestingModule } from "@abp/ng.core/testing"; |
|||
import { ThemeBasicTestingModule } from "@abp/ng.theme.basic/testing"; |
|||
import { ThemeSharedTestingModule } from "@abp/ng.theme.shared/testing"; |
|||
import { ComponentFixture, TestBed, waitForAsync } from "@angular/core/testing"; |
|||
import { NgxValidateCoreModule } from "@ngx-validate/core"; |
|||
import { MyComponent } from "./my.component"; |
|||
|
|||
describe("MyComponent", () => { |
|||
let fixture: ComponentFixture<MyComponent>; |
|||
|
|||
beforeEach( |
|||
waitForAsync(() => { |
|||
TestBed.configureTestingModule({ |
|||
declarations: [MyComponent], |
|||
imports: [ |
|||
CoreTestingModule.withConfig(), |
|||
ThemeSharedTestingModule.withConfig(), |
|||
ThemeBasicTestingModule.withConfig(), |
|||
NgxValidateCoreModule, |
|||
], |
|||
providers: [ |
|||
/* mock providers here */ |
|||
], |
|||
}).compileComponents(); |
|||
}) |
|||
); |
|||
|
|||
beforeEach(() => { |
|||
fixture = TestBed.createComponent(MyComponent); |
|||
fixture.detectChanges(); |
|||
}); |
|||
|
|||
it("should be initiated", () => { |
|||
expect(fixture.componentInstance).toBeTruthy(); |
|||
}); |
|||
}); |
|||
``` |
|||
|
|||
If you take a look at the imports, you will notice that we have prepared some testing modules to replace built-in ABP modules. This is necessary for providing mocks for some features which otherwise would break your tests. Please remember to **use testing modules** and **call their `withConfig` static method**. |
|||
|
|||
## Tips |
|||
|
|||
### Angular Testing Library |
|||
|
|||
Although you can test your code with Angular TestBed, you may find [Angular Testing Library](https://testing-library.com/docs/angular-testing-library/intro) a good alternative. |
|||
|
|||
The simple example above can be written with Angular Testing Library as follows: |
|||
|
|||
```js |
|||
import { CoreTestingModule } from "@abp/ng.core/testing"; |
|||
import { ThemeBasicTestingModule } from "@abp/ng.theme.basic/testing"; |
|||
import { ThemeSharedTestingModule } from "@abp/ng.theme.shared/testing"; |
|||
import { ComponentFixture } from "@angular/core/testing"; |
|||
import { NgxValidateCoreModule } from "@ngx-validate/core"; |
|||
import { render } from "@testing-library/angular"; |
|||
import { MyComponent } from "./my.component"; |
|||
|
|||
describe("MyComponent", () => { |
|||
let fixture: ComponentFixture<MyComponent>; |
|||
|
|||
beforeEach(async () => { |
|||
const result = await render(MyComponent, { |
|||
imports: [ |
|||
CoreTestingModule.withConfig(), |
|||
ThemeSharedTestingModule.withConfig(), |
|||
ThemeBasicTestingModule.withConfig(), |
|||
NgxValidateCoreModule, |
|||
], |
|||
providers: [ |
|||
/* mock providers here */ |
|||
], |
|||
}); |
|||
|
|||
fixture = result.fixture; |
|||
}); |
|||
|
|||
it("should be initiated", () => { |
|||
expect(fixture.componentInstance).toBeTruthy(); |
|||
}); |
|||
}); |
|||
``` |
|||
|
|||
Very similar, as you can see. The real difference kicks in when we use queries and fire events. |
|||
|
|||
```js |
|||
// other imports |
|||
import { getByLabelText, screen } from "@testing-library/angular"; |
|||
import userEvent from "@testing-library/user-event"; |
|||
|
|||
describe("MyComponent", () => { |
|||
beforeEach(/* removed for sake of brevity */); |
|||
|
|||
it("should display advanced filters", () => { |
|||
const filters = screen.getByTestId("author-filters"); |
|||
const nameInput = getByLabelText(filters, /name/i) as HTMLInputElement; |
|||
expect(nameInput.offsetWidth).toBe(0); |
|||
|
|||
const advancedFiltersBtn = screen.getByRole("link", { name: /advanced/i }); |
|||
userEvent.click(advancedFiltersBtn); |
|||
|
|||
expect(nameInput.offsetWidth).toBeGreaterThan(0); |
|||
|
|||
userEvent.type(nameInput, "fooo{backspace}"); |
|||
expect(nameInput.value).toBe("foo"); |
|||
}); |
|||
}); |
|||
``` |
|||
|
|||
The **queries in Angular Testing Library follow practices for maintainable tests**, the user event package provides a **human-like interaction** with the DOM, and the library in general has **a clear API** that simplifies component testing. Please find some useful links below: |
|||
|
|||
- [Queries](https://testing-library.com/docs/dom-testing-library/api-queries) |
|||
- [User Event](https://testing-library.com/docs/ecosystem-user-event) |
|||
- [Examples](https://github.com/testing-library/angular-testing-library/tree/master/apps/example-app/app/examples) |
|||
|
|||
### Clearing DOM After Each Spec |
|||
|
|||
One thing to remember is that Karma runs tests in real browser instances. That means, you will be able to see the result of your test code, but also have problems with components attached to the document body which may not get cleared after each test, even when you configure Karma to do so. |
|||
|
|||
We have prepared a simple function with which you can clear any leftover DOM elements after each test. |
|||
|
|||
```js |
|||
// other imports |
|||
import { clearPage } from "@abp/ng.core/testing"; |
|||
|
|||
describe("MyComponent", () => { |
|||
let fixture: ComponentFixture<MyComponent>; |
|||
|
|||
afterEach(() => clearPage(fixture)); |
|||
|
|||
beforeEach(async () => { |
|||
const result = await render(MyComponent, { |
|||
/* removed for sake of brevity */ |
|||
}); |
|||
fixture = result.fixture; |
|||
}); |
|||
|
|||
// specs here |
|||
}); |
|||
``` |
|||
|
|||
Please make sure you use it because Karma will fail to remove dialogs otherwise and you will have multiple copies of modals, confirmation boxes, and alike. |
|||
|
|||
### Waiting |
|||
|
|||
Some components, modals, in particular, work off-detection-cycle. In other words, you cannot reach DOM elements inserted by these components immediately after opening them. Similarly, inserted elements are not immediately destroyed upon closing them. |
|||
|
|||
For this purpose, we have prepared a `wait` function. |
|||
|
|||
```js |
|||
// other imports |
|||
import { wait } from "@abp/ng.core/testing"; |
|||
|
|||
describe("MyComponent", () => { |
|||
beforeEach(/* removed for sake of brevity */); |
|||
|
|||
it("should open a modal", async () => { |
|||
const openModalBtn = screen.getByRole("button", { name: "Open Modal" }); |
|||
userEvent.click(openModalBtn); |
|||
|
|||
await wait(fixture); |
|||
|
|||
const modal = screen.getByRole("dialog"); |
|||
|
|||
expect(modal).toBeTruthy(); |
|||
|
|||
/* wait again after closing the modal */ |
|||
}); |
|||
}); |
|||
``` |
|||
|
|||
The `wait` function takes a second parameter, i.e. timeout (default: `0`). Try not to use it though. Using a timeout bigger than `0` is usually a signal that something is not quite right. |
|||
|
|||
## Testing Example |
|||
|
|||
Here is an example test suite. It doesn't cover all, but gives quite a good idea about what the testing experience will be like. |
|||
|
|||
```js |
|||
import { clearPage, CoreTestingModule, wait } from "@abp/ng.core/testing"; |
|||
import { ThemeBasicTestingModule } from "@abp/ng.theme.basic/testing"; |
|||
import { ThemeSharedTestingModule } from "@abp/ng.theme.shared/testing"; |
|||
import { ComponentFixture } from "@angular/core/testing"; |
|||
import { |
|||
NgbCollapseModule, |
|||
NgbDatepickerModule, |
|||
NgbDropdownModule, |
|||
} from "@ng-bootstrap/ng-bootstrap"; |
|||
import { NgxValidateCoreModule } from "@ngx-validate/core"; |
|||
import { CountryService } from "@proxy/countries"; |
|||
import { |
|||
findByText, |
|||
getByLabelText, |
|||
getByRole, |
|||
getByText, |
|||
queryByRole, |
|||
render, |
|||
screen, |
|||
} from "@testing-library/angular"; |
|||
import userEvent from "@testing-library/user-event"; |
|||
import { BehaviorSubject, of } from "rxjs"; |
|||
import { CountryComponent } from "./country.component"; |
|||
|
|||
const list$ = new BehaviorSubject({ |
|||
items: [{ id: "ID_US", name: "United States of America" }], |
|||
totalCount: 1, |
|||
}); |
|||
|
|||
describe("Country", () => { |
|||
let fixture: ComponentFixture<CountryComponent>; |
|||
|
|||
afterEach(() => clearPage(fixture)); |
|||
|
|||
beforeEach(async () => { |
|||
const result = await render(CountryComponent, { |
|||
imports: [ |
|||
CoreTestingModule.withConfig(), |
|||
ThemeSharedTestingModule.withConfig(), |
|||
ThemeBasicTestingModule.withConfig(), |
|||
NgxValidateCoreModule, |
|||
NgbCollapseModule, |
|||
NgbDatepickerModule, |
|||
NgbDropdownModule, |
|||
], |
|||
providers: [ |
|||
{ |
|||
provide: CountryService, |
|||
useValue: { |
|||
getList: () => list$, |
|||
}, |
|||
}, |
|||
], |
|||
}); |
|||
|
|||
fixture = result.fixture; |
|||
}); |
|||
|
|||
it("should display advanced filters", () => { |
|||
const filters = screen.getByTestId("country-filters"); |
|||
const nameInput = getByLabelText(filters, /name/i) as HTMLInputElement; |
|||
expect(nameInput.offsetWidth).toBe(0); |
|||
|
|||
const advancedFiltersBtn = screen.getByRole("link", { name: /advanced/i }); |
|||
userEvent.click(advancedFiltersBtn); |
|||
|
|||
expect(nameInput.offsetWidth).toBeGreaterThan(0); |
|||
|
|||
userEvent.type(nameInput, "fooo{backspace}"); |
|||
expect(nameInput.value).toBe("foo"); |
|||
|
|||
userEvent.click(advancedFiltersBtn); |
|||
expect(nameInput.offsetWidth).toBe(0); |
|||
}); |
|||
|
|||
it("should have a heading", () => { |
|||
const heading = screen.getByRole("heading", { name: "Countries" }); |
|||
expect(heading).toBeTruthy(); |
|||
}); |
|||
|
|||
it("should render list in table", async () => { |
|||
const table = await screen.findByTestId("country-table"); |
|||
|
|||
const name = getByText(table, "United States of America"); |
|||
expect(name).toBeTruthy(); |
|||
}); |
|||
|
|||
it("should display edit modal", async () => { |
|||
const actionsBtn = screen.queryByRole("button", { name: /actions/i }); |
|||
userEvent.click(actionsBtn); |
|||
|
|||
const editBtn = screen.getByRole("button", { name: /edit/i }); |
|||
userEvent.click(editBtn); |
|||
|
|||
await wait(fixture); |
|||
|
|||
const modal = screen.getByRole("dialog"); |
|||
const modalHeading = queryByRole(modal, "heading", { name: /edit/i }); |
|||
expect(modalHeading).toBeTruthy(); |
|||
|
|||
const closeBtn = getByText(modal, "×"); |
|||
userEvent.click(closeBtn); |
|||
|
|||
await wait(fixture); |
|||
|
|||
expect(screen.queryByRole("dialog")).toBeFalsy(); |
|||
}); |
|||
|
|||
it("should display create modal", async () => { |
|||
const newBtn = screen.getByRole("button", { name: /new/i }); |
|||
userEvent.click(newBtn); |
|||
|
|||
await wait(fixture); |
|||
|
|||
const modal = screen.getByRole("dialog"); |
|||
const modalHeading = queryByRole(modal, "heading", { name: /new/i }); |
|||
|
|||
expect(modalHeading).toBeTruthy(); |
|||
}); |
|||
|
|||
it("should validate required name field", async () => { |
|||
const newBtn = screen.getByRole("button", { name: /new/i }); |
|||
userEvent.click(newBtn); |
|||
|
|||
await wait(fixture); |
|||
|
|||
const modal = screen.getByRole("dialog"); |
|||
const nameInput = getByRole(modal, "textbox", { |
|||
name: /^name/i, |
|||
}) as HTMLInputElement; |
|||
|
|||
userEvent.type(nameInput, "x"); |
|||
userEvent.type(nameInput, "{backspace}"); |
|||
|
|||
const nameError = await findByText(modal, /required/i); |
|||
expect(nameError).toBeTruthy(); |
|||
}); |
|||
|
|||
it("should delete a country", () => { |
|||
const getSpy = spyOn(fixture.componentInstance.list, "get"); |
|||
const deleteSpy = jasmine.createSpy().and.returnValue(of(null)); |
|||
fixture.componentInstance.service.delete = deleteSpy; |
|||
|
|||
const actionsBtn = screen.queryByRole("button", { name: /actions/i }); |
|||
userEvent.click(actionsBtn); |
|||
|
|||
const deleteBtn = screen.getByRole("button", { name: /delete/i }); |
|||
userEvent.click(deleteBtn); |
|||
|
|||
const confirmText = screen.getByText("AreYouSure"); |
|||
expect(confirmText).toBeTruthy(); |
|||
|
|||
const confirmBtn = screen.getByRole("button", { name: "Yes" }); |
|||
userEvent.click(confirmBtn); |
|||
|
|||
expect(deleteSpy).toHaveBeenCalledWith(list$.value.items[0].id); |
|||
expect(getSpy).toHaveBeenCalledTimes(1); |
|||
}); |
|||
}); |
|||
``` |
|||
|
|||
## CI Configuration |
|||
|
|||
You would need a different configuration for your CI environment. To set up a new configuration for your unit tests, find the test project in _angular.json_ file and add one as seen below: |
|||
|
|||
```json |
|||
// angular.json |
|||
|
|||
"test": { |
|||
"builder": "@angular-devkit/build-angular:karma", |
|||
"options": { /* several options here */ }, |
|||
"configurations": { |
|||
"production": { |
|||
"karmaConfig": "karma.conf.prod.js" |
|||
} |
|||
} |
|||
} |
|||
``` |
|||
|
|||
Now you can copy the _karma.conf.js_ as _karma.conf.prod.js_ and use any configuration you like in it. Please check [Karma configuration file document](http://karma-runner.github.io/5.2/config/configuration-file.html) for config options. |
|||
|
|||
Finally, don't forget to run your CI tests with the following command: |
|||
|
|||
```sh |
|||
npm test -- --prod |
|||
``` |
|||
|
|||
## See Also |
|||
|
|||
* [ABP Community Video - Unit Testing with the Angular UI](https://community.abp.io/articles/unit-testing-with-the-angular-ui-p4l550q3) |
|||
|
|||
|
After Width: | Height: | Size: 75 KiB |
|
After Width: | Height: | Size: 64 KiB |
@ -0,0 +1,57 @@ |
|||
# Blazor UI: Page Progress |
|||
|
|||
Page Progress is used to show a progress bar indicator on top of the page and to show to the user that currently a long running process is in the work. |
|||
|
|||
By default you don't need to do anything to show the progress indicator, as all the work is done automatically by the ABP Framework internals. This means that all calls to the ABP backend (through your HTTP API) will activate page progress and show the loading indicator. |
|||
|
|||
This doesn't mean that you don't have the control over it. On the contrary, if you want to show progress for your own processes, it is really easy to do. All you have to do is to use inject and use the `IUiPageProgressService`. |
|||
|
|||
## Example |
|||
|
|||
First, inject the `IUiPageProgressService` into your page/component. |
|||
|
|||
```cs |
|||
@inject IUiPageProgressService pageProgressService |
|||
``` |
|||
|
|||
Next, invoke the `Go` method in `IUiPageProgressService`. It's that simple: |
|||
|
|||
```cs |
|||
Task OnClick() |
|||
{ |
|||
return pageProgressService.Go(null); |
|||
} |
|||
``` |
|||
|
|||
The previous example will show the progress with a default settings. If, for example you want to change the progress color you can override it by setting the options through the `Go` method. |
|||
|
|||
```cs |
|||
Task OnClick() |
|||
{ |
|||
return pageProgressService.Go(null, options => |
|||
{ |
|||
options.Type = UiPageProgressType.Warning; |
|||
}); |
|||
} |
|||
``` |
|||
|
|||
## Breakdown |
|||
|
|||
The first parameter of the `Go` needs a little explanation. In the previous example we have set it to `null` which means, once called it will show an _indeterminate_ indicator and will cycle the loading animation indefinitely, until we hide the progress. You also have the option of defining the actual percentage of the progress and the code is the same, just instead of sending it the `null` you will send it a number between `0` and `100`. |
|||
|
|||
```cs |
|||
pageProgressService.Go(25) |
|||
``` |
|||
|
|||
### Valid values |
|||
|
|||
1. `null` - show _indeterminate_ indicator |
|||
2. `>= 0` and `<= 100` - show the regular _percentage_ progress |
|||
|
|||
### Hiding progress |
|||
|
|||
To hide the progress just set the actual values to something other then the _Valid value_. |
|||
|
|||
```cs |
|||
pageProgressService.Go(-1) |
|||
``` |
|||
@ -1,3 +1,9 @@ |
|||
@using Volo.Abp.Ui.Branding |
|||
@using Volo.Abp.Ui.Branding |
|||
@inject IBrandingProvider BrandingProvider |
|||
<a class="navbar-brand" href="">@BrandingProvider.AppName</a> |
|||
<a class="navbar-brand" href=""> |
|||
@if (!BrandingProvider.LogoUrl.IsNullOrWhiteSpace()) |
|||
{ |
|||
<img src="@BrandingProvider.LogoUrl" alt="@BrandingProvider.AppName" > |
|||
} |
|||
@BrandingProvider.AppName |
|||
</a> |
|||
|
|||
@ -0,0 +1,12 @@ |
|||
using System; |
|||
using Volo.Abp.DependencyInjection; |
|||
|
|||
namespace Volo.Abp.AspNetCore.Components.WebAssembly.DependencyInjection |
|||
{ |
|||
public class WebAssemblyClientScopeServiceProviderAccessor : |
|||
IClientScopeServiceProviderAccessor, |
|||
ISingletonDependency |
|||
{ |
|||
public IServiceProvider ServiceProvider { get; set; } |
|||
} |
|||
} |
|||
@ -0,0 +1,21 @@ |
|||
using System; |
|||
using System.Threading.Tasks; |
|||
|
|||
namespace Volo.Abp.AspNetCore.Components.Progression |
|||
{ |
|||
public interface IUiPageProgressService |
|||
{ |
|||
/// <summary>
|
|||
/// An event raised after the notification is received.
|
|||
/// </summary>
|
|||
public event EventHandler<UiPageProgressEventArgs> ProgressChanged; |
|||
|
|||
/// <summary>
|
|||
/// Sets the progress percentage.
|
|||
/// </summary>
|
|||
/// <param name="percentage">Value of the progress from 0 to 100, or null for indeterminate progress.</param>
|
|||
/// <param name="options">Additional options.</param>
|
|||
/// <returns>Awaitable task.</returns>
|
|||
Task Go(int? percentage, Action<UiPageProgressOptions> options = null); |
|||
} |
|||
} |
|||
@ -0,0 +1,16 @@ |
|||
using System; |
|||
using System.Threading.Tasks; |
|||
using Volo.Abp.DependencyInjection; |
|||
|
|||
namespace Volo.Abp.AspNetCore.Components.Progression |
|||
{ |
|||
public class NullUiPageProgressService : IUiPageProgressService, ISingletonDependency |
|||
{ |
|||
public event EventHandler<UiPageProgressEventArgs> ProgressChanged; |
|||
|
|||
public Task Go(int? percentage, Action<UiPageProgressOptions> options = null) |
|||
{ |
|||
return Task.CompletedTask; |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,17 @@ |
|||
using System; |
|||
|
|||
namespace Volo.Abp.AspNetCore.Components.Progression |
|||
{ |
|||
public class UiPageProgressEventArgs : EventArgs |
|||
{ |
|||
public UiPageProgressEventArgs(int? percentage, UiPageProgressOptions options) |
|||
{ |
|||
Percentage = percentage; |
|||
Options = options; |
|||
} |
|||
|
|||
public int? Percentage { get; } |
|||
|
|||
public UiPageProgressOptions Options { get; } |
|||
} |
|||
} |
|||
@ -0,0 +1,13 @@ |
|||
namespace Volo.Abp.AspNetCore.Components.Progression |
|||
{ |
|||
/// <summary>
|
|||
/// Options to override page progress appearance.
|
|||
/// </summary>
|
|||
public class UiPageProgressOptions |
|||
{ |
|||
/// <summary>
|
|||
/// Type or color, of the page progress.
|
|||
/// </summary>
|
|||
public UiPageProgressType Type { get; set; } |
|||
} |
|||
} |
|||
@ -0,0 +1,11 @@ |
|||
namespace Volo.Abp.AspNetCore.Components.Progression |
|||
{ |
|||
public enum UiPageProgressType |
|||
{ |
|||
Default, |
|||
Info, |
|||
Success, |
|||
Warning, |
|||
Error, |
|||
} |
|||
} |
|||
Some files were not shown because too many files changed in this diff
Loading…
Reference in new issue