La création d'applications Web de niveau entreprise peut être complexe et prendre du temps.
ABP Commercial offre l'infrastructure de base parfaite nécessaire pour tous les ASP.NET Core modernes de niveau entreprise solutions basées. De la conception au déploiement, l'ensemble du cycle de développement est optimisé par les fonctionnalités et modules intégrés d'ABP.
",
+ "StartupTemplatesShortDescription": "Les modèles de démarrage vous permettent de démarrer votre projet en quelques secondes.",
+ "UIFrameworksOptions": "Options de cadres d'interface utilisateur;",
+ "DatabaseProviderOptions": "Options du fournisseur de base de données;",
+ "PreBuiltApplicationModules": "Modules d'application prédéfinis",
+ "PreBuiltApplicationModulesShortDescription": "Les exigences d'application les plus courantes sont déjà développées pour vous en tant que modules réutilisables.",
+ "Account": "Compte",
+ "Blogging": "Bloguer",
+ "Identity": "Identité",
+ "IdentityServer": "Serveur d'identité",
+ "Saas": "Saas",
+ "LanguageManagement": "Gestion de la langue",
+ "TextTemplateManagement": "Gestion des modèles de texte",
+ "See All Modules": "VoirTousModules",
+ "ABPSuite": "Suite ABP",
+ "AbpSuiteShortDescription": "ABP Suite est un outil complémentaire à ABP Commercial.",
+ "AbpSuiteExplanation": "Il vous permet de créer des pages Web en quelques minutes. C'est un outil global .NET Core qui peut être installé à partir de la ligne de commande. Il peut créer une nouvelle solution ABP, générer des pages CRUD de la base de données vers le front-end.",
+ "Details": "Des détails",
+ "LeptonTheme": "Thème Lepton",
+ "ProfessionalModernUIThemes": "Thèmes d'interface utilisateur professionnels et modernes",
+ "LeptonThemeExplanation": "Lepton fournit une gamme de thèmes d'administration Bootstrap qui servent de base solide pour tout projet nécessitant un tableau de bord d'administration.",
+ "DefaultTheme": "Thème par défaut",
+ "MaterialTheme": "Thème matériel",
+ "Default2Theme": "Thème par défaut 2",
+ "DarkTheme": "Thème sombre",
+ "DarkBlueTheme": "Thème bleu foncé",
+ "LightTheme": "Thème léger",
+ "ProudToWorkWith": "Fier de travailler avec",
+ "OurConsumers": "Des centaines d'entreprises et de développeurs dans plus de 50 pays dans le monde font confiance à ABP Commercial.",
+ "JoinOurConsumers": "Rejoignez-les et créez rapidement des produits incroyables.",
+ "AdditionalServicesExplanation": "Avez-vous besoin de services supplémentaires ou personnalisés? Nous et nos partenaires pouvons fournir; ",
+ "CustomProjectDevelopment": "Développement de projets personnalisés",
+ "CustomProjectDevelopmentExplanation": "Développeurs dédiés pour vos projets personnalisés.",
+ "PortingExistingProjects": "Portage de projets existants",
+ "PortingExistingProjectsExplanation": "Migration de vos anciens projets vers la plateforme ABP.",
+ "LiveSupport": "Support en direct",
+ "LiveSupportExplanation": "Option d'assistance à distance en direct lorsque vous en avez besoin.",
+ "Training": "Formation",
+ "TrainingExplanation": "Une formation dédiée pour vos développeurs.",
+ "OnBoarding": "Intégration",
+ "OnBoardingExplanation": "Aide à la configuration de vos environnements de développement, CI et CD.",
+ "PrioritizedTechnicalSupport": "Assistance technique prioritaire",
+ "PremiumSupportExplanation": "Outre le grand support communautaire du framework ABP, notre équipe de support répond aux questions techniques et aux problèmes des utilisateurs commerciaux avec une priorité élevée.",
+ "SeeTheSupportOptions": "Voir les options de support",
+ "Contact": "Contacter",
+ "TellUsWhatYouNeed": "Dites-nous ce dont vous avez besoin.",
+ "YourMessage": "Votre message",
+ "YourFullName": "Ton nom complet",
+ "EmailField": "Adresse e-mail",
+ "YourEmailAddress": "Votre adresse email",
+ "HowMayWeHelpYou": "Comment pouvons nous vous aider?",
+ "SendMessage": "Envoyer le message",
+ "Success": "Succès",
+ "WeWillReplyYou": "Nous avons reçu votre message et vous contacterons sous peu.",
+ "GoHome": "Rentrer chez soi",
+ "CreateLiveDemo": "Créer une démo en direct",
+ "RegisterToTheNewsletter": "Inscrivez-vous à la newsletter pour recevoir des informations concernant ABP.IO, y compris les nouvelles versions, etc.",
+ "EnterYourEmailOrLogin": "Saisissez votre adresse e-mail pour créer votre démo ou Connectez-vous à l'aide de votre compte existant.",
+ "ApplicationTemplate": "Modèle d'application",
+ "ApplicationTemplateExplanation": "Le modèle de démarrage d'application est utilisé pour créer une nouvelle application Web.",
+ "EfCoreProvider": "Entity Framework (prend en charge SQL Server, MySQL, PostgreSQL, Oracle et autres )",
+ "AlreadyIncludedInTemplateModules": "Les modules suivants sont déjà inclus et configurés dans ce modèle:",
+ "ApplicationTemplateArchitecture": "Ce modèle d'application prend également en charge l'architecture à plusieurs niveaux dans laquelle la couche d'interface utilisateur, la couche API et le service d'authentification sont physiquement séparés.",
+ "SeeTheGuideOrGoToTheLiveDemo": "Consultez le guide du développeur pour obtenir des informations techniques sur ce modèle ou accédez à la démo en direct.",
+ "DeveloperGuide": "Guide du développeur",
+ "ModuleTemplate": "Modèle de module",
+ "ModuleTemplateExplanation1": "Vous souhaitez créer un module et le réutiliser dans différentes applications? Ce modèle de démarrage prépare tout pour commencer à créer un module d'application réutilisable ou un microservice .",
+ "ModuleTemplateExplanation2": "
Vous pouvez prendre en charge un ou plusieurs frameworks d'interface utilisateur, un ou plusieurs fournisseurs de bases de données pour un seul module. Le modèle de démarrage est configuré pour exécuter et tester votre module dans une application minimale en plus de l'infrastructure de test d'unité et d'intégration.
Consultez le guide du développeur pour obtenir des informations techniques sur ce modèle.
",
+ "WithAllStyleOptions": "avec toutes les options de style",
+ "Demo": "Démo",
+ "SeeAllModules": "Voir tous les modules",
+ "ABPCLIExplanation": "ABP CLI (Command Line Interface) est un outil de ligne de commande pour effectuer certaines opérations courantes pour les solutions basées sur ABP.",
+ "ABPSuiteEasilyCURD": "ABP Suite est un outil qui vous permet de créer facilement des pages CRUD",
+ "WeAreHereToHelp": "Nous sommes ici pour Aide ",
+ "BrowseOrAskQuestion": "Vous pouvez parcourir nos rubriques d'aide ou rechercher dans les questions fréquemment posées, ou vous pouvez nous poser une question en utilisant le formulaire de contact .",
+ "SearchQuestionPlaceholder": "Rechercher dans les questions fréquemment posées",
+ "WhatIsTheABPCommercial": "Qu'est-ce que la publicité ABP?",
+ "WhatAreDifferencesThanAbpFramework": "Quelles sont les différences entre le Framework ABP open source et le ABP Commercial?",
+ "ABPCommercialExplanation": "ABP Commercial est un ensemble de modules, d'outils, de thèmes et de services premium basés sur le framework ABP open source. ABP Commercial est développé et soutenu par la même équipe derrière le cadre ABP.",
+ "WhatAreDifferencesThanABPFrameworkExplanation": "
Framework ABP est un framework de développement d'applications modulaire, thématique et compatible avec les micro-services pour ASP.NET Core. Il fournit une architecture complète et une infrastructure solide pour vous permettre de vous concentrer sur votre propre code métier plutôt que de vous répéter à chaque nouveau projet. Il est basé sur les meilleures pratiques de développement logiciel et les outils populaires que vous connaissez déjà.
Le framework ABP est entièrement gratuit, open source et géré par la communauté. Il fournit également un thème gratuit et des modules prédéfinis (par exemple, la gestion des identités et la gestion des locataires).
",
+ "VisitTheFrameworkVSCommercialDocument": "Pour plus d'informations, consultez le lien suivant: {1} ",
+ "ABPCommercialFollowingBenefits": "ABP Commercial ajoute les avantages suivants en plus du cadre ABP;",
+ "Professional": "Professionnel",
+ "UIThemes": "Thèmes de l'interface utilisateur",
+ "EnterpriseModules": "Des modules d'application prédéfinis, prêts pour l'entreprise, riches en fonctionnalités (par exemple, gestion d'Identity Server, gestion SaaS, gestion des langues)",
+ "ToolingToSupport": "Outils pour soutenir votre productivité de développement (par exemple, ABP Suite )",
+ "PremiumSupportLink": "Assistance Premium ",
+ "WhatDoIDownloadABPCommercial": "Que dois-je télécharger lorsque j'achète l'ABP Commercial?",
+ "CreateUnlimitedSolutions": "Une fois que vous avez acheté une licence commerciale ABP, vous pourrez créer des solutions illimitées comme décrit dans le document Premiers pas .",
+ "ABPCommercialSolutionExplanation": "Lorsque vous créez une nouvelle application, vous obtenez une solution Visual Studio (un modèle de démarrage) en fonction de vos préférences. La solution téléchargée contient des modules et des thèmes commerciaux déjà installés et configurés pour vous. Vous pouvez supprimer un module préinstallé ou ajouter un autre module si vous le souhaitez. Tous les modules et thèmes sont utilisés par défaut dans des packages NuGet / NPM.",
+ "StartDevelopWithTutorials": "La solution téléchargée est bien architecturée et documentée. Vous pouvez commencer à développer votre propre code d'entreprise basé sur celui-ci en suivant les didacticiels ",
+ "TryTheCommercialDemo": "Vous pouvez essayer la démonstration pour voir un exemple d'application créé à l'aide du modèle de démarrage ABP Commercial.",
+ "HowManyProducts": "Combien de produits / solutions différents puis-je créer à l'aide d'ABP Commercial?",
+ "HowManyProductsExplanation": "Il n'y a pas de limite pour créer un projet ABP. Vous pouvez créer autant de projets que vous le souhaitez, les développer et les télécharger sur différents serveurs.",
+ "HowManyDevelopers": "Combien de développeurs peuvent travailler sur l'ABP Commercial?",
+ "HowManyDevelopersExplanation": "Les licences commerciales ABP sont par développeur. Différents types de licence ont des limites de développeur différentes. Cependant, vous pouvez ajouter plus de développeurs à n'importe quel type de licence chaque fois que vous en avez besoin. Consultez la page des prix pour connaître les types de licence, les limites pour les développeurs et les coûts supplémentaires pour les développeurs.",
+ "ChangingLicenseType": "Puis-je changer mon type de licence à l'avenir?",
+ "ChangingLicenseTypeExplanation": "Vous pouvez toujours ajouter de nouveaux développeurs dans votre même type de licence. Voir aussi \"Combien de développeurs peuvent travailler sur l'ABP Commercial?\". Vous pouvez également passer à une licence supérieure en payant la différence de prix calculée. Lorsque vous passez à un plan de licence supérieur, vous bénéficiez des avantages du nouveau plan, mais la mise à niveau de la licence ne modifie pas la date d'expiration de la licence.",
+ "LicenseExtendUpgradeDiff": "Quelle est la différence entre l'extension de licence et la mise à niveau?",
+ "LicenseExtendUpgradeDiffExplanation": " Extension: en prolongeant / renouvelant votre licence, vous continuerez à bénéficier d'une assistance premium et à des mises à jour majeures pour les modules et les thèmes. De plus, vous pourrez continuer à créer de nouveaux projets. Et vous pourrez toujours utiliser ABP Suite, ce qui accélère votre développement. Mise à niveau: En mettant à jour votre licence, vous passerez à un plan de licence supérieur qui vous permettra d’obtenir des avantages supplémentaires . Consultez le tableau de comparaison des licences pour vérifier les différences entre les plans de licence. En revanche, lors de la mise à niveau, la date d'expiration de votre licence ne changera pas! Strong > Pour prolonger la date de fin de votre licence, vous devez prolonger votre licence.",
+ "LicenseRenewalCost": "Quel est le coût du renouvellement de la licence après 1 an?",
+ "LicenseRenewalCostExplanation": "Le taux de renouvellement (extension) de toutes les licences perpétuelles ABP Commercial correspond à {0} du prix catalogue de la licence. Le prix de renouvellement de la licence d'équipe standard est de {1} $, la licence professionnelle standard de {2} $ et la licence d'entreprise standard de {3} $. Si vous êtes déjà client, connectez-vous à votre compte pour consulter les tarifs de renouvellement disponibles.",
+ "HowDoIRenewMyLicense": "Comment renouveler ma licence?",
+ "HowDoIRenewMyLicenseExplanation": "Vous pouvez renouveler votre licence en accédant à la page de gestion de l'organisation . Afin de profiter de nos tarifs réduits de renouvellement anticipé, assurez-vous de renouveler avant l'expiration de votre licence. Cependant, ne vous inquiétez pas de ne pas savoir quand votre opportunité de renouvellement anticipé se termine. Vous recevrez 2 e-mails de rappel avant l'expiration de votre abonnement. Nous les enverrons dans les 30 jours, 7 jours avant l'expiration.",
+ "IsSourceCodeIncluded": "Ma licence inclut-elle le code source des modules et thèmes commerciaux?",
+ "IsSourceCodeIncludedExplanation1": "Dépend du type de licence que vous avez acheté:",
+ "IsSourceCodeIncludedExplanation2": " Équipe : votre solution utilise les modules et les thèmes en tant que packages NuGet et NPM. Il n'inclut pas leur code source. De cette façon, vous pouvez facilement mettre à niveau ces modules et thèmes chaque fois qu'une nouvelle version est disponible. Cependant, vous ne pouvez pas obtenir le code source des modules et des thèmes.",
+ "IsSourceCodeIncludedExplanation3": " Entreprise / Entreprise : en plus de la licence Team, vous pouvez télécharger le code source de tout module ou thème dont vous avez besoin. Vous pouvez même supprimer les références de package NuGet / NPM pour un module particulier et ajouter son code source directement à votre solution pour le modifier complètement.",
+ "IsSourceCodeIncludedExplanation4": "
L'inclusion du code source d'un module dans votre solution vous donne le maximum de liberté pour personnaliser ce module. Cependant, il ne sera pas possible de mettre à jour automatiquement le module lorsqu'une nouvelle version est publiée.
Aucune des licences n'inclut le code source d'ABP Suite, qui est un outil externe qui génère du code pour vous et vous aide à votre développement.
Consultez la page de tarification pour connaître les autres différences entre les types de licence.
",
+ "ChangingDevelopers": "Puis-je changer les développeurs enregistrés de mon organisation à l'avenir?",
+ "ChangingDevelopersExplanation": "En plus d'ajouter de nouveaux développeurs à votre licence, vous pouvez également modifier les développeurs existants (vous pouvez supprimer un développeur et en ajouter un nouveau sur le même siège) sans aucun coût supplémentaire.",
+ "WhatHappensWhenLicenseEnds": "Que se passe-t-il à la fin de ma période de licence?",
+ "WhatHappensWhenLicenseEndsExplanation1": "Le type de licence commerciale ABP est licence perpétuelle . Une fois votre licence expirée, vous pouvez continuer à développer votre projet. Et vous n'êtes pas obligé de renouveler votre licence. Votre licence est livrée avec un plan de mise à jour et de support d'un an prêt à l'emploi. Pour continuer à recevoir de nouvelles fonctionnalités, des améliorations de performances, des corrections de bogues, une assistance et continuer à utiliser ABP Suite, assurez-vous de renouveler votre plan chaque année. Lorsque votre licence expirera, vous ne pourrez plus bénéficier des avantages suivants;",
+ "WhatHappensWhenLicenseEndsExplanation2": "Vous ne pouvez pas créer de nouvelles solutions en utilisant ABP Commercial, mais vous pouvez continuer à développer vos applications existantes pour toujours.",
+ "WhatHappensWhenLicenseEndsExplanation3": "Vous pourrez obtenir des mises à jour pour les modules et les thèmes de votre version MAJEURE. Par example; si vous utilisez la v3.2.0 d'un module, vous pouvez toujours obtenir des mises à jour pour la v3.x.x (v3.3.0, v3.5.2 ... etc.) de ce module. Mais vous ne pouvez pas obtenir de mises à jour pour la prochaine version majeure (comme v4.x, v5.x)",
+ "WhatHappensWhenLicenseEndsExplanation4": "Vous ne pouvez pas installer de nouveaux modules et thèmes ajoutés à la plate-forme ABP Commercial après la fin de votre licence.",
+ "WhatHappensWhenLicenseEndsExplanation5": "Vous ne pouvez pas utiliser ABP Suite.",
+ "WhatHappensWhenLicenseEndsExplanation6": "Vous ne pouvez plus bénéficier de l ' assistance premium .",
+ "WhatHappensWhenLicenseEndsExplanation7": "Vous pouvez renouveler votre abonnement si vous souhaitez continuer à bénéficier de ces avantages. Il y a une réduction de 20% lorsque vous renouvelez votre abonnement.",
+ "WhenShouldIRenewMyLicense": "Quand dois-je renouveler ma licence?",
+ "WhenShouldIRenewMyLicenseExplanation1": "Si vous renouvelez votre licence dans un délai de 1 mois après l'expiration de votre licence, une réduction de 20% sera appliquée au prix total de la licence.",
+ "WhenShouldIRenewMyLicenseExplanation2": "Si vous renouvelez votre licence 1 mois après la date d'expiration de votre licence, le prix de renouvellement sera le même que le prix d'achat de la licence et il n'y aura pas de réduction pour votre renouvellement.",
+ "TrialPlan": "Avez-vous un plan d'essai?",
+ "TrialPlanExplanation": "Pour l'instant, ABP Commercial n'a pas de plan d'essai. Pour les licences Team, nous offrons une garantie de remboursement de 30 jours. Vous pouvez simplement demander un remboursement dans les 30 premiers jours. Pour les licences Business et Enterprise, nous fournissons un remboursement de 60% en 30 jours. En effet, les licences Business et Enterprise incluent le code source complet de tous les modules et thèmes.",
+ "DoYouAcceptBankWireTransfer": "Acceptez-vous les virements bancaires?",
+ "DoYouAcceptBankWireTransferExplanation": "Oui, nous acceptons les virements bancaires. Après avoir envoyé le montant de la licence par virement bancaire, envoyez-nous votre reçu et le type de licence demandé par e-mail. Nos coordonnées bancaires internationales:",
+ "HowToUpgrade": "Comment mettre à niveau des applications existantes lorsqu'une nouvelle version est disponible?",
+ "HowToUpgradeExplanation1": "Lorsque vous créez une nouvelle application à l'aide d'ABP Commercial, tous les modules et le thème sont utilisés en tant que packages NuGet et NPM. Ainsi, vous pouvez facilement mettre à jour les packages lorsqu'une nouvelle version est disponible.",
+ "HowToUpgradeExplanation2": "Outre les mises à niveau standard de NuGet / NPM, ABP CLI fournit une commande de mise à jour qui recherche et met automatiquement à niveau tous les packages associés à ABP dans votre solution.",
+ "DatabaseSupport": "Quels systèmes de base de données sont pris en charge?",
+ "DatabaseSupportExplanation": "ABP Framework lui-même est indépendant de la base de données et peut fonctionner avec n'importe quel fournisseur de base de données de par sa nature. Consultez le document d'accès aux données pour obtenir la liste des fournisseurs actuellement mis en œuvre.",
+ "UISupport": "Quels frameworks d'interface utilisateur sont pris en charge?",
+ "Supported": "Prise en charge",
+ "UISupportExplanation": "ABP Framework lui-même est indépendant du framework d'interface utilisateur et peut fonctionner avec n'importe quel framework d'interface utilisateur. Cependant, les modèles de démarrage, les interfaces utilisateur de module et les thèmes n'étaient pas implémentés pour tous les frameworks d'interface utilisateur. Consultez le document de démarrage pour la liste à jour des options d'interface utilisateur.",
+ "MicroserviceSupport": "Prend-il en charge l'architecture des micro-services?",
+ "MicroserviceSupportExplanation1": "L'un des principaux objectifs du cadre ABP est de fournir une infrastructure pratique pour créer des solutions de micro-services. Consultez le document architecture de micro-service pour comprendre comment cela aide à créer des systèmes de micro-service.",
+ "MicroserviceSupportExplanation2": "Tous les modules ABP Commercial sont conçus pour prendre en charge les scénarios de déploiement de micro-services (avec sa propre API et sa propre base de données) en suivant le document bonnes pratiques de développement de modules .",
+ "MicroserviceSupportExplanation3": "Nous fournissons un exemple de solution de démonstration de micro-service qui présente une implémentation d'architecture de micro-service pour vous aider à créer votre propre solution.",
+ "MicroserviceSupportExplanation4": "Donc, la réponse courte est \" oui, il prend en charge l'architecture de micro-services \".",
+ "MicroserviceSupportExplanation5": "Cependant, un système de micro-service est une solution et chaque solution aura des exigences différentes, une topologie de réseau, des scénarios de communication, des possibilités d'authentification, des décisions de séparation / partage de base de données, des configurations d'exécution, des intégrations de systèmes tiers et bien d'autres.",
+ "MicroserviceSupportExplanation6": "Le Framework ABP et l'ABP Commercial fournissent une infrastructure pour des scénarios de micro-services, des modules compatibles avec les micro-services, des exemples et de la documentation pour vous aider à créer votre propre solution. Mais ne vous attendez pas à télécharger directement la solution de vos rêves préconfigurée pour vous. Vous devrez le comprendre et rassembler certaines parties en fonction de vos besoins.",
+ "WhereCanIDownloadSourceCode": "Où puis-je télécharger le code source?",
+ "WhereCanIDownloadSourceCodeExplanation": "Vous pouvez télécharger le code source de tous les modules ABP, packages angulaires et thèmes via ABP Suite ou ABP CLI. Voir Comment télécharger le code source? ",
+ "ComputerLimitation": "Combien d'ordinateurs un développeur peut-il se connecter lors du développement d'ABP?",
+ "ComputerLimitationExplanation": "Nous autorisons spécifiquement les {0} ordinateurs par développeur individuel / sous licence. Chaque fois qu'un développeur a besoin de développer des produits commerciaux ABP sur une troisième machine, un e-mail doit être envoyé à license@abp.io expliquant la situation et nous procéderons ensuite à l'allocation appropriée dans notre système.",
+ "RefundPolicy": "Avez-vous une politique de remboursement?",
+ "RefundPolicyExplanation": "Vous pouvez demander un remboursement dans les 30 jours suivant l'achat de votre licence. Les types de licence Business et Enterprise ont une option de téléchargement de code source, par conséquent, les remboursements ne sont pas disponibles pour les entreprises et les entreprises (et toutes les licences qui incluent un droit de recevoir le code source). De plus, aucun remboursement n'est effectué pour les renouvellements et les achats de deuxième licence.",
+ "HowCanIRefundVat": "Comment puis-je rembourser la TVA?",
+ "HowCanIRefundVatExplanation1": "Si vous avez effectué le paiement avec 2Checkout, vous pouvez rembourser la TVA via votre compte 2Checkout:",
+ "HowCanIRefundVatExplanation2": "Connectez-vous à votre compte 2Checkout ",
+ "HowCanIRefundVatExplanation3": "Trouvez la commande appropriée et appuyez sur \"Rembourser la TVA tardive\" (entrez votre numéro de TVA)",
+ "HowCanIGetMyInvoice": "Comment puis-je obtenir ma facture?",
+ "HowCanIGetMyInvoiceExplanation": "Il existe 2 passerelles de paiement pour l'achat d'une licence: PayU et 2Checkout. Si vous achetez votre licence via la passerelle 2Checkout, elle envoie la facture PDF à votre adresse e-mail, voir Facturation 2Checkout. Si vous achetez via la passerelle PayU ou par virement bancaire, nous préparerons et vous enverrons votre facture. Vous pouvez demander votre facture sur la page de gestion de l'organisation. ",
+ "Forum": "Forum",
+ "SupportExplanation": "ABP Commercial Licences fournit un support de forum premium par une équipe composée d'experts ABP Framework.",
+ "PrivateTicket": "Billet privé",
+ "PrivateTicketExplanation": "Enterprise License comprend également un support privé avec e-mail et système de ticket.",
+ "AbpSuiteExplanation1": "ABP Suite vous permet de créer des pages Web en quelques minutes. C'est un outil .NET Core Global qui peut être installé à partir de la ligne de commande.",
+ "AbpSuiteExplanation2": "Il peut créer une nouvelle solution ABP, générer des pages CRUD de la base de données vers le front-end. Pour une présentation technique, consultez le document ",
+ "FastEasy": "Rapide et facile",
+ "AbpSuiteExplanation3": "ABP Suite vous permet de créer facilement des pages CRUD. Il vous suffit de définir votre entité et ses propriétés, laissez le reste à ABP Suite pour vous! ABP Suite génère tout le code nécessaire pour votre page CRUD en quelques secondes. Il prend en charge les interfaces utilisateur Angular, MVC et Blazor.",
+ "RichOptions": "Options riches",
+ "AbpSuiteExplanation4": "ABP Suite prend en charge plusieurs options d'interface utilisateur telles que Razor Pages et Angular . Il prend également en charge plusieurs bases de données telles que MongoDB et toutes les bases de données prises en charge par EntityFramework Core < / strong> (MS SQL Server, Oracle, MySql, PostgreSQL et plus ).",
+ "AbpSuiteExplanation5": "La bonne chose est que vous n'avez pas à vous soucier de ces options. ABP Suite comprend votre type de projet et génère le code de votre projet et place le code généré au bon endroit dans votre projet.",
+ "SourceCode": "Code source",
+ "AbpSuiteExplanation6": "ABP Suite génère le code source pour vous! Il ne génère pas de fichiers magiques pour générer la page Web. ABP Suite génère le code source pour Entity, Repository, Application Service, Code First Migration, JavaScript / TypeScript et CSHTML / HTML ainsi que les interfaces nécessaires. ABP Suite génère également le code selon les meilleures pratiques de développement logiciel, vous n'avez donc pas à vous soucier de la qualité du code généré.",
+ "AbpSuiteExplanation7": "Puisque vous avez le code source des blocs de construction de la page CRUD générée dans les couches d'application correctes, vous pouvez facilement modifier le code source et injecter votre logique personnalisée / commerciale dans le code généré.",
+ "CrossPlatform": "Plateforme croisée",
+ "AbpSuiteExplanation8": "ABP Suite est construit avec .NET Core et est multiplateforme. Il fonctionne comme une application Web sur votre ordinateur local. Vous pouvez l'exécuter sur Windows , Mac et Linux ",
+ "OtherFeatures": "Autres caractéristiques",
+ "OtherFeatures1": "Met à jour facilement les packages NuGet et NPM de votre solution.",
+ "OtherFeatures2": "Régénère les pages déjà générées à partir de zéro.",
+ "OtherFeatures3": "Crée de nouvelles solutions",
+ "ThanksForCreatingProject": "Merci d'avoir créé votre projet!",
+ "HotToRunSolution": "Comment faire fonctionner votre solution?",
+ "HotToRunSolutionExplanation": "Consultez le document de mise en route pour savoir comment configurer et exécuter votre solution.",
+ "GettingStarted": "Commencer",
+ "WebAppDevTutorial": "Tutoriel de développement d'applications Web",
+ "WebAppDevTutorialExplanation": "Consultez le document du didacticiel de développement d'applications Web pour un exemple de développement étape par étape.",
+ "Document": "Document",
+ "UsingABPSuiteToCURD": "Utilisation d'ABP Suite pour la génération et l'outillage de pages CRUD",
+ "SeeABPSuiteDocument": "Consultez le document ABP Suite pour en savoir plus sur l'utilisation d'ABP Suite.",
+ "AskQuestionsOnSupport": "Vous pouvez poser des questions sur le support commercial ABP.",
+ "Documentation": "Documentation",
+ "SeeModulesDocument": "Voir le document des modules pour une liste de tous les modules commerciaux (pro) et leurs documents.",
+ "Pricing": "Tarification",
+ "PricingExplanation": "Choisissez les fonctionnalités et fonctionnalités dont votre entreprise a besoin aujourd'hui. Mettez à niveau facilement au fur et à mesure que votre entreprise se développe.",
+ "Team": "Équipe",
+ "Business": "Affaires",
+ "Enterprise": "Entreprise",
+ "Custom": "Personnalisé",
+ "IncludedDeveloperLicenses": "Licences de développeur incluses",
+ "CustomLicenceOrAdditionalServices": "Besoin d'une licence personnalisée ou de services supplémentaires?",
+ "CustomOrVolumeLicense": "Licence personnalisée ou en volume",
+ "LiveTrainingSupport": "Formation et assistance en direct",
+ "AndMore": "et plus",
+ "AdditionalDeveloperLicense": "Licence de développeur supplémentaire",
+ "ProjectCount": "Nombre de projets",
+ "AllProModules": "Tous les modules pro",
+ "AllProThemes": "Tous les thèmes pro",
+ "AllProStartupTemplates": "Tous les modèles de démarrage pro",
+ "SourceCodeOfAllModules": "Code source de tous les modules",
+ "SourceCodeOfAllThemes": "Code source de tous les thèmes",
+ "PerpetualLicense": "Licence perpétuelle",
+ "UnlimitedServerDeployment": "Déploiement de serveur illimité",
+ "YearUpgrade": "Mise à jour d'un an",
+ "YearPremiumForumSupport": "1 an de support premium pour le forum",
+ "ForumSupportIncidentCountYear": "Nombre d'incidents de support du forum / an",
+ "PrivateTicketEmailSupport": "Support privé pour les tickets et e-mail",
+ "BuyNow": "Acheter maintenant",
+ "PayViaAmexCard": "Comment puis-je payer via ma carte AMEX?",
+ "PayViaAmexCardDescription": "La passerelle de paiement par défaut «Iyzico» peut refuser certaines cartes de crédit AMEX en raison des mesures de sécurité. Dans ce cas, vous pouvez payer via la passerelle de paiement alternative «2Checkout».",
+ "ThankYou": "Merci",
+ "InvalidReCaptchaErrorMessage": "Une erreur s'est produite lors de la vérification de reCAPTCHA. Veuillez réessayer."
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/fi.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/fi.json
new file mode 100644
index 0000000000..b2da832b2a
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Commercial/Localization/Resources/fi.json
@@ -0,0 +1,296 @@
+{
+ "culture": "fi",
+ "texts": {
+ "OrganizationManagement": "Organisaation hallinta",
+ "OrganizationList": "Organisaatioluettelo",
+ "Volo.AbpIo.Commercial:010003": "Et ole tämän organisaation omistaja!",
+ "OrganizationNotFoundMessage": "Organisaatiota ei löytynyt!",
+ "DeveloperCount": "Kohdistetut / yhteensä kehittäjät",
+ "QuestionCount": "Jäljellä olevat / yhteensä kysymykset",
+ "Unlimited": "Rajoittamaton",
+ "Owners": "Omistajat",
+ "AddMember": "Lisää jäsen",
+ "AddOwner": "Lisää omistaja",
+ "AddDeveloper": "Lisää kehittäjä",
+ "UserName": "Käyttäjätunnus",
+ "Name": "Nimi",
+ "EmailAddress": "Sähköpostiosoite",
+ "Developers": "Kehittäjät",
+ "LicenseType": "Lisenssi-tyyppi",
+ "Manage": "Hallitse",
+ "StartDate": "Aloituspäivämäärä",
+ "EndDate": "Päättymispäivä",
+ "Modules": "Moduulit",
+ "LicenseExtendMessage": "Käyttöoikeutesi päättymispäivä on pidennetty {0}",
+ "LicenseUpgradeMessage": "Lisenssisi on päivitetty versioon {0}",
+ "LicenseAddDeveloperMessage": "{0} kehittäjää lisäsi lisenssiisi",
+ "Volo.AbpIo.Commercial:010004": "Määritettyä käyttäjää ei löydy! Käyttäjän on oltava jo rekisteröitynyt.",
+ "MyOrganizations": "Omat organisaatiot",
+ "ApiKey": "API-avain",
+ "UserNameNotFound": "Käyttäjää, jolla on käyttäjänimi {0}, ei ole",
+ "SuccessfullyAddedToNewsletter": "Kiitos, että tilasit uutiskirjeemme!",
+ "MyProfile": "Profiilini",
+ "EmailNotValid": "Ole hyvä ja syötä toimiva sähköpostiosoite.",
+ "JoinOurMarketingNewsletter": "Liity markkinointikirjeeseemme",
+ "WouldLikeToReceiveMarketingMaterials": "Haluaisin saada markkinointimateriaalia, kuten tuotetarjouksia ja erikoistarjouksia.",
+ "StartUsingYourLicenseNow": "Aloita lisenssin käyttö nyt!",
+ "WelcomePage": "Tervetuloa sivu",
+ "UnsubscriptionExpireEmail": "Peruuta lisenssin vanhentumispäivämäärän muistutussähköpostit",
+ "UnsubscribeLicenseExpireEmailReminderMessage": "Tämä sähköpostitilaus sisältää vain muistutuksen lisenssin vanhentumispäivästä.",
+ "UnsubscribeFromLicenseExpireEmails": "Jos et halua vastaanottaa lisenssin voimassaolon päättymispäivää koskevia sähköposteja, voit peruuttaa tilauksen milloin tahansa.",
+ "Unsubscribe": "Lopeta tilaus",
+ "NotOrganizationMember": "Et ole minkään organisaation jäsen.",
+ "UnsubscribeLicenseExpirationEmailSuccessTitle": "Tilauksen peruutus onnistui",
+ "UnsubscribeLicenseExpirationEmailSuccessMessage": "Et enää saa muistutuksia lisenssin vanhentumispäivästä.",
+ "IndexPageHeroSection": " Täydellinen verkkokehitysympäristö sisäänrakennettu kehys ",
+ "AbpCommercialShortDescription": "ABP Commercial tarjoaa valmiita sovellusmoduuleja, nopean sovelluskehitystyökalun, ammattimaiset käyttöliittymäteemat, ensiluokkaisen tuen ja paljon muuta.",
+ "LiveDemo": "Live-esittely",
+ "GetLicence": "Hanki lisenssi",
+ "Application": "Sovellus",
+ "StartupTemplates": "Käynnistysmallit",
+ "Startup": "Aloittaa",
+ "Templates": "Mallit",
+ "Developer": "Kehittäjä",
+ "Tools": "Työkalut",
+ "Premium": "Premium",
+ "PremiumSupport": "Ensiluokkainen tuki",
+ "PremiumForumSupport": "Premium-foorumin tuki",
+ "UI": "UI",
+ "Themes": "Teemat",
+ "JoinOurNewsletter": "Liity uutiskirjeemme",
+ "Send": "Lähettää",
+ "Learn": "Oppia",
+ "AdditionalServices": "Lisäpalvelut",
+ "WhatIsABPFramework": "MIKÄ ON ABP-KEHYS?",
+ "OpenSourceBaseFramework": "Avoimen lähdekoodin peruskehys",
+ "ABPFrameworkExplanation": "
ABP Commercial perustuu ABP Frameworkiin, avoimen lähdekoodin ja yhteisövetoiseen verkkosovelluskehykseen ASP.NET Core -sovellukselle.
ABP Framework tarjoaa erinomaisen infrastruktuurin ylläpitettävien, laajennettavien tiedostojen kirjoittamiseen. ja testattava koodi parhailla käytänteillä.
Rakennettu ja integroitu jo tunnettujen suosittujen työkalujen kanssa. Matala oppimiskäyrä, helppo sopeutuminen, mukava kehitys.
Yritysluokan verkkosovellusten rakentaminen voi olla monimutkaista ja aikaa vievää.
ABP Commercial tarjoaa täydellisen perusinfrastruktuurin, joka tarvitaan kaikille nykyaikaisille yritystason ASP.NET-ytimille perustuvat ratkaisut. ABP: n sisäänrakennetut ominaisuudet ja moduulit tukevat koko suunnittelusykliä suunnittelusta käyttöönottoon asti.
",
+ "StartupTemplatesShortDescription": "Käynnistysmallien avulla voit aloittaa projektisi muutamassa sekunnissa.",
+ "UIFrameworksOptions": "Käyttöliittymäkehysvaihtoehdot;",
+ "DatabaseProviderOptions": "Tietokannan tarjoajan vaihtoehdot;",
+ "PreBuiltApplicationModules": "Valmiit sovellusmoduulit",
+ "PreBuiltApplicationModulesShortDescription": "Yleisimmät sovellusvaatimukset on jo kehitetty sinulle uudelleenkäytettävinä moduuleina.",
+ "Account": "Tili",
+ "Blogging": "Bloggaaminen",
+ "Identity": "Identiteetti",
+ "IdentityServer": "Identity Server",
+ "Saas": "Saas",
+ "LanguageManagement": "Kielen hallinta",
+ "TextTemplateManagement": "Tekstimallien hallinta",
+ "See All Modules": "Katso kaikki moduulit",
+ "ABPSuite": "ABP Suite",
+ "AbpSuiteShortDescription": "ABP Suite on ABP-mainosmateriaalia täydentävä työkalu.",
+ "AbpSuiteExplanation": "Sen avulla voit rakentaa verkkosivuja muutamassa minuutissa. Se on .NET Core Global -työkalu, joka voidaan asentaa komentoriviltä. Se voi luoda uuden ABP-ratkaisun, luoda CRUD-sivuja tietokannasta käyttöliittymään.",
+ "Details": "Yksityiskohdat",
+ "LeptonTheme": "Lepton-teema",
+ "ProfessionalModernUIThemes": "Ammattimaiset, modernit käyttöliittymäteemat",
+ "LeptonThemeExplanation": "Lepton tarjoaa valikoiman Bootstrap-järjestelmänvalvojan teemoja, jotka ovat vankka perusta mille tahansa projektille, joka vaatii järjestelmänvalvojan hallintapaneelin.",
+ "DefaultTheme": "Oletusteema",
+ "MaterialTheme": "Materiaaliteema",
+ "Default2Theme": "Oletus 2 teema",
+ "DarkTheme": "Tumma teema",
+ "DarkBlueTheme": "Tummansininen teema",
+ "LightTheme": "Kevyt teema",
+ "ProudToWorkWith": "Ylpeä työskennellä",
+ "OurConsumers": "Sadat yritykset ja kehittäjät yli 50 maassa ympäri maailmaa luottavat ABP Commercialiin.",
+ "JoinOurConsumers": "Liity heihin ja rakenna upeita tuotteita nopeasti.",
+ "AdditionalServicesExplanation": "Tarvitsetko lisä- tai mukautettuja palveluita? Me ja kumppanimme voimme tarjota; ",
+ "CustomProjectDevelopment": "Mukautettu projektikehitys",
+ "CustomProjectDevelopmentExplanation": "Omistetut kehittäjät mukautetuille projekteillesi.",
+ "PortingExistingProjects": "Olemassa olevien projektien siirtäminen",
+ "PortingExistingProjectsExplanation": "Vanhojen projektien siirtäminen ABP-alustalle.",
+ "LiveSupport": "Live-tuki",
+ "LiveSupportExplanation": "Live-etätukivaihtoehto, kun tarvitset sitä.",
+ "Training": "Koulutus",
+ "TrainingExplanation": "Oma koulutus kehittäjille.",
+ "OnBoarding": "Perehdytys",
+ "OnBoardingExplanation": "Auta määrittämään kehitys-, CI- ja CD-ympäristösi.",
+ "PrioritizedTechnicalSupport": "Ensisijainen tekninen tuki",
+ "PremiumSupportExplanation": "ABP-kehyksen suuren yhteisötuen lisäksi tukitiimimme vastaa ensisijaisesti kaupallisten käyttäjien teknisiin kysymyksiin ja ongelmiin.",
+ "SeeTheSupportOptions": "Katso tukivaihtoehdot",
+ "Contact": "Ottaa yhteyttä",
+ "TellUsWhatYouNeed": "Kerro meille mitä tarvitset.",
+ "YourMessage": "Viestisi",
+ "YourFullName": "Koko nimesi",
+ "EmailField": "Sähköpostiosoite",
+ "YourEmailAddress": "Sähköpostiosoitteesi",
+ "HowMayWeHelpYou": "Kuinka voimme auttaa sinua?",
+ "SendMessage": "Lähetä viesti",
+ "Success": "Menestys",
+ "WeWillReplyYou": "Saimme viestisi ja otamme sinuun pian yhteyttä.",
+ "GoHome": "Mene kotiin",
+ "CreateLiveDemo": "Luo live-esittely",
+ "RegisterToTheNewsletter": "Rekisteröidy uutiskirjeeseen saadaksesi tietoa ABP.IO: sta, mukaan lukien uudet julkaisut jne.",
+ "EnterYourEmailOrLogin": "Kirjoita sähköpostiosoitteesi luodaksesi esittely tai kirjaudu sisään nykyisellä tililläsi.",
+ "ApplicationTemplate": "Sovelluksen malli",
+ "ApplicationTemplateExplanation": "Sovelluksen käynnistysmallia käytetään uuden verkkosovelluksen luomiseen.",
+ "EfCoreProvider": "Entity Framework (tukee SQL Serveriä, MySQL: ää, PostgreSQL: ää, Oracle ja muita )",
+ "AlreadyIncludedInTemplateModules": "Seuraavat moduulit ovat jo mukana ja määritetty tähän malliin:",
+ "ApplicationTemplateArchitecture": "Tämä sovellusmalli tukee myös porrastettua arkkitehtuuria, jossa käyttöliittymäkerros, API-kerros ja todennuspalvelu on fyysisesti erotettu.",
+ "SeeTheGuideOrGoToTheLiveDemo": "Katso teknistä tietoa tästä mallista kehittäjän oppaasta tai siirry live-esittelyyn.",
+ "DeveloperGuide": "Kehittäjän opas",
+ "ModuleTemplate": "Moduulimalli",
+ "ModuleTemplateExplanation1": "Haluatko luoda moduulin ja käyttää sitä uudelleen eri sovelluksissa? Tämä käynnistysmalli valmistaa kaiken aloittamaan uudelleenkäytettävän sovellusmoduulin tai mikropalvelun luomisen.",
+ "ModuleTemplateExplanation2": "
Voit tukea yhtä tai useampaa käyttöliittymäkehystä, yhtä tai useampaa tietokantapalvelua yhdelle moduulille. Käynnistysmalli on määritetty suorittamaan ja testaamaan moduulia pienimmässä sovelluksessa yksikön ja integraatiotestiinfrastruktuurin lisäksi.
Katso teknistä tietoa mallista kehittäjän oppaasta.
",
+ "WithAllStyleOptions": "kaikilla tyylivaihtoehdoilla",
+ "Demo": "Demo",
+ "SeeAllModules": "Katso kaikki moduulit",
+ "ABPCLIExplanation": "ABP CLI (Command Line Interface) on komentorivityökalu joidenkin yleisten toimintojen suorittamiseen ABP-pohjaisiin ratkaisuihin.",
+ "ABPSuiteEasilyCURD": "ABP Suite on työkalu, jonka avulla voit helposti luoda CRUD-sivuja",
+ "WeAreHereToHelp": " Apua olemme täällä",
+ "BrowseOrAskQuestion": "Voit selata ohjeaiheitamme tai etsiä usein kysyttyjä kysymyksiä tai voit esittää meille kysymyksiä yhteydenottolomakkeella .",
+ "SearchQuestionPlaceholder": "Hae usein kysyttyjä kysymyksiä",
+ "WhatIsTheABPCommercial": "Mikä on ABP-kauppa?",
+ "WhatAreDifferencesThanAbpFramework": "Mitä eroja on avoimen lähdekoodin ABP Frameworkilla ja ABP Commercialilla?",
+ "ABPCommercialExplanation": "ABP Commercial on joukko ensiluokkaisia moduuleja, työkaluja, teemoja ja palveluja, jotka on rakennettu avoimen lähdekoodin ABP-kehyksen päälle. ABP Commercial kehittää ja tukee samaa tiimiä ABP-kehyksen takana.",
+ "WhatAreDifferencesThanABPFrameworkExplanation": "
ABP-kehys on modulaarinen, teemoitettava, mikropalvelujen kanssa yhteensopiva sovelluskehys ASP.NET Core -sovellukselle. Se tarjoaa täydellisen arkkitehtuurin ja vahvan infrastruktuurin, joka saa sinut keskittymään omaan yrityskoodiin sen sijaan, että toistat itsesi jokaisessa uudessa projektissa. Se perustuu ohjelmistokehityksen parhaisiin käytäntöihin ja jo tiedettyihin suosittuihin työkaluihin.
ABP-kehykset ovat täysin ilmaisia, avoimen lähdekoodin ja yhteisölähtöisiä. Se tarjoaa myös ilmaisen teeman ja joitain valmiita moduuleja (esim. Henkilöllisyyden hallinta ja vuokralaisten hallinta).
",
+ "VisitTheFrameworkVSCommercialDocument": "Vieraile seuraavalla linkillä saadaksesi lisätietoja {1} ",
+ "ABPCommercialFollowingBenefits": "ABP Commercial lisää seuraavat edut ABP-kehyksen päälle;",
+ "Professional": "Ammattilainen",
+ "UIThemes": "Käyttöliittymän teemat",
+ "EnterpriseModules": "Yrityskäyttöiset, monipuoliset, valmiit sovellusmoduulit (esim. Identity Server -hallinta, SaaS-hallinta, kielen hallinta)",
+ "ToolingToSupport": "Työkalut, jotka tukevat kehitystesi tuottavuutta (esim. ABP Suite )",
+ "PremiumSupportLink": "Ensiluokkainen tuki ",
+ "WhatDoIDownloadABPCommercial": "Mitä lataan ostaessani ABP Commercial -palvelua?",
+ "CreateUnlimitedSolutions": "Kun olet ostanut ABP Commercial -lisenssin, voit luoda rajoittamattomia ratkaisuja, kuten Aloittaminen -asiakirjassa kuvataan.",
+ "ABPCommercialSolutionExplanation": "Kun luot uuden sovelluksen, saat Visual Studio -ratkaisun (käynnistysmalli) mieltymystesi perusteella. Ladatussa ratkaisussa on kaupallisia moduuleja ja teemoja, jotka on jo asennettu ja määritetty sinulle. Voit poistaa esiasennetun moduulin tai lisätä toisen moduulin, jos haluat. Kaikissa moduuleissa ja teemoissa käytetään oletuksena NuGet / NPM-paketteja.",
+ "StartDevelopWithTutorials": "Ladattu ratkaisu on hyvin suunniteltu ja dokumentoitu. Voit aloittaa oman yrityskoodisi kehittämisen sen perusteella opetusohjelmien mukaisesti",
+ "TryTheCommercialDemo": "Voit kokeilla esittelyä nähdäksesi esimerkkisovelluksen, joka on luotu ABP Commercial -käynnistysmallin avulla.",
+ "HowManyProducts": "Kuinka monta erilaista tuotetta / ratkaisua voin rakentaa ABP Commercial -sovelluksella?",
+ "HowManyProductsExplanation": "ABP-projektin luomiselle ei ole rajoituksia. Voit luoda niin monta projektia kuin haluat, kehittää ja ladata ne eri palvelimille.",
+ "HowManyDevelopers": "Kuinka monta kehittäjää voi työskennellä ABP-kaupassa?",
+ "HowManyDevelopersExplanation": "ABP Commercial -lisenssit ovat kehittäjää kohti. Eri lisenssityypeillä on erilaiset kehittäjien rajoitukset. Voit kuitenkin lisätä lisää kehittäjiä mihin tahansa lisenssityyppiin milloin tahansa. Katso hinnat -sivulta lisenssityypit, kehittäjien rajoitukset ja kehittäjien lisäkustannukset.",
+ "ChangingLicenseType": "Voinko muuttaa lisenssityyppiäni tulevaisuudessa?",
+ "ChangingLicenseTypeExplanation": "Voit aina lisätä uusia kehittäjiä samaan lisenssityyppiin. Katso myös \"Kuinka monta kehittäjää voi työskennellä ABP Commercialilla?\". Voit myös päivittää korkeampaan lisenssiin maksamalla lasketun hintaeron. Kun päivität korkeampaan lisenssisuunnitelmaan, saat uuden suunnitelman edut, mutta lisenssin päivitys ei muuta lisenssin voimassaolon päättymispäivää.",
+ "LicenseExtendUpgradeDiff": "Mitä eroa on lisenssin laajennuksella ja päivityksellä?",
+ "LicenseExtendUpgradeDiffExplanation": " Laajentaminen: Laajentamalla / uusimalla käyttöoikeutta saat jatkossakin ensiluokkaista tukea ja saat tärkeitä päivityksiä moduuleille ja teemoille. Lisäksi voit jatkaa uusien projektien luomista. Ja voit silti käyttää ABP Suitea, joka vauhdittaa kehitystäsi. Päivittäminen: Päivittämällä lisenssisi edistyt korkeampaan lisenssisuunnitelmaan, jonka avulla voit saada lisäetuja . Katso lisenssisuunnitelmien väliset erot lisenssien vertailutaulukosta . Toisaalta, kun päivität, lisenssin vanhentumispäivä ei muutu! Strong > Lisenssin päättymispäivän jatkamiseksi sinun on jatkettava käyttöoikeutta.",
+ "LicenseRenewalCost": "Mitkä ovat lisenssin uusimiskustannukset vuoden kuluttua?",
+ "LicenseRenewalCostExplanation": "Kaikkien ABP Commercial perpetual -lisenssien uusimis- (laajennus) hinta on {0} lisenssiluettelohinnasta. Tavallisen tiimilisenssin uusimishinta on $ {1}, tavallinen yrityslisenssi on $ {2} ja tavallinen yrityslisenssi on $ {3}. Jos olet jo asiakas, kirjaudu tilillesi ja tarkista käytettävissä oleva uusimishinta.",
+ "HowDoIRenewMyLicense": "Kuinka uusin lisenssin?",
+ "HowDoIRenewMyLicenseExplanation": "Voit uusia käyttöoikeutesi siirtymällä organisaation hallintasivulle . Jotta voisit hyödyntää alennettujen varhaisen uusimisen hintojamme, muista uusia ennen lisenssin voimassaolon päättymistä. Älä huoli siitä, ettet tiedä, milloin Varhaisen uudistamisen mahdollisuus sulkeutuu. Saat kaksi muistutussähköpostia ennen tilauksesi päättymistä. Lähetämme ne 30 päivää, 7 päivää ennen vanhenemista.",
+ "IsSourceCodeIncluded": "Sisältääkö lisenssini kaupallisten moduulien ja teemojen lähdekoodin?",
+ "IsSourceCodeIncludedExplanation1": "Riippuu ostamastasi lisenssityypistä:",
+ "IsSourceCodeIncludedExplanation2": " Tiimi : Ratkaisusi käyttää moduuleja ja teemoja NuGet- ja NPM-paketteina. Se ei sisällä heidän lähdekoodiaan. Tällä tavoin voit helposti päivittää nämä moduulit ja teemat aina, kun uusi versio on saatavana. Et kuitenkaan voi saada moduulien ja teemojen lähdekoodia.",
+ "IsSourceCodeIncludedExplanation3": " Yritys / yritys : Tiimilisenssin lisäksi voit ladata minkä tahansa tarvitsemasi moduulin tai teeman lähdekoodin. Voit jopa poistaa tietyn moduulin NuGet / NPM-pakettiviitteet ja lisätä sen lähdekoodin suoraan ratkaisuusi sen muuttamiseksi kokonaan.",
+ "IsSourceCodeIncludedExplanation4": "
Moduulin lähdekoodin sisällyttäminen ratkaisuun antaa sinulle maksimaalisen vapauden mukauttaa moduulia. Tällöin moduulia ei voida päivittää automaattisesti, kun uusi versio julkaistaan.
Mikään lisensseistä ei sisällä ABP Suiten lähdekoodia, joka on ulkoinen työkalu, joka tuottaa koodia sinulle ja auttaa kehitykseen.
",
+ "ChangingDevelopers": "Voinko muuttaa organisaationi rekisteröityneitä kehittäjiä tulevaisuudessa?",
+ "ChangingDevelopersExplanation": "Uusien kehittäjien lisäämisen lisenssiin lisäksi voit myös muuttaa olemassa olevia kehittäjiä (voit poistaa kehittäjän ja lisätä uuden samalle paikalle) ilman lisäkustannuksia.",
+ "WhatHappensWhenLicenseEnds": "Mitä tapahtuu, kun lisenssikauteni päättyy?",
+ "WhatHappensWhenLicenseEndsExplanation1": "ABP Commercial -lisenssityyppi on jatkuva lisenssi . Kun lisenssi vanhenee, voit jatkaa projektisi kehittämistä. Ja sinun ei tarvitse uudistaa lisenssiäsi. Lisenssisi mukana toimitetaan yhden vuoden päivitys- ja tukisuunnitelma. Jos haluat edelleen saada uusia ominaisuuksia, suorituskyvyn parannuksia, virhekorjauksia, tukea ja jatkaa ABP Suite -sovelluksen käyttöä, muista uusia suunnitelma joka vuosi. Kun lisenssi vanhenee, et voi saada enemmän seuraavista eduista;",
+ "WhatHappensWhenLicenseEndsExplanation2": "Et voi luoda uusia ratkaisuja ABP Commercial -sovelluksella, mutta voit jatkaa olemassa olevien sovellusten kehittämistä ikuisesti.",
+ "WhatHappensWhenLicenseEndsExplanation3": "Voit saada päivityksiä moduuleista ja teemoista MAJOR-versiostasi. Esimerkiksi; jos käytät moduulin v3.2.0-versiota, voit silti saada päivityksiä kyseisen moduulin versiolle v3.x.x (v3.3.0, v3.5.2 ... jne.). Mutta et voi saada päivityksiä seuraavaan pääversioon (kuten v4.x, v5.x)",
+ "WhatHappensWhenLicenseEndsExplanation4": "Et voi asentaa uusia moduuleja ja teemoja, jotka on lisätty ABP Commercial -alustalle käyttöoikeuden päättymisen jälkeen.",
+ "WhatHappensWhenLicenseEndsExplanation5": "Et voi käyttää ABP Suitea.",
+ "WhatHappensWhenLicenseEndsExplanation6": "Et voi enää saada premium-tukea .",
+ "WhatHappensWhenLicenseEndsExplanation7": "Voit uusia tilauksesi, jos haluat jatkaa näiden etujen saamista. Tilauksen uusimisessa on 20% alennus.",
+ "WhenShouldIRenewMyLicense": "Milloin minun pitäisi uusia lisenssini?",
+ "WhenShouldIRenewMyLicenseExplanation1": "Jos uusit lisenssin kuukauden kuluessa lisenssin voimassaolon päättymisestä, lisenssin kokonaishintaan sovelletaan 20% alennusta.",
+ "WhenShouldIRenewMyLicenseExplanation2": "Jos uusit lisenssin kuukauden kuluttua lisenssin voimassaolon päättymispäivästä, uusimishinta on sama kuin lisenssin ostohinta eikä uusimiselle ole alennusta.",
+ "TrialPlan": "Onko sinulla kokeilusuunnitelma?",
+ "TrialPlanExplanation": "Toistaiseksi ABP Commercialilla ei ole kokeilusuunnitelmaa. Joukkueen lisensseille tarjoamme 30 päivän rahanpalautustakuun. Voit vain pyytää hyvitystä 30 ensimmäisen päivän aikana. Yritys- ja yrityslisensseille hyvitämme 60% 30 päivässä. Tämä johtuu siitä, että yritys- ja yrityslisenssit sisältävät kaikkien moduulien ja teemojen täydellisen lähdekoodin.",
+ "DoYouAcceptBankWireTransfer": "Hyväksytkö pankkisiirron?",
+ "DoYouAcceptBankWireTransferExplanation": "Kyllä, hyväksymme pankkisiirron. Kun olet lähettänyt lisenssimäärän pankkisiirrolla, lähetä meille kuitti ja pyydetty lisenssityyppi sähköpostitse. Kansainväliset pankkitilitiedot:",
+ "HowToUpgrade": "Kuinka päivittää olemassa olevia sovelluksia, kun uusi versio on saatavilla?",
+ "HowToUpgradeExplanation1": "Kun luot uuden sovelluksen ABP Commercial -sovelluksella, kaikkia moduuleja ja teemaa käytetään NuGet- ja NPM-paketteina. Joten voit päivittää paketit helposti, kun uusi versio on saatavilla.",
+ "HowToUpgradeExplanation2": "Normaalien NuGet / NPM-päivitysten lisäksi ABP CLI tarjoaa päivityskomennon, joka etsii ja päivittää kaikki ratkaisusi ABP-paketit.",
+ "DatabaseSupport": "Mitä tietokantajärjestelmiä tuetaan?",
+ "DatabaseSupportExplanation": "ABP Framework itsessään on tietokanta-agnostinen ja voi luonteeltaan työskennellä minkä tahansa tietokantapalvelujen kanssa. Katso tietojen käyttöasiakirjasta luettelo tällä hetkellä toteutetuista palveluntarjoajista.",
+ "UISupport": "Mitä käyttöliittymäkehyksiä tuetaan?",
+ "Supported": "Tuettu",
+ "UISupportExplanation": "ABP Framework itsessään on käyttöliittymäkehyksen agnostikko ja voi toimia minkä tahansa käyttöliittymäkehyksen kanssa. Käynnistysmalleja, moduulin käyttöliittymiä ja teemoja ei kuitenkaan otettu käyttöön kaikissa käyttöliittymäkehyksissä. Katso käyttöliittymävaihtoehtojen ajantasainen luettelo aloitusasiakirjasta .",
+ "MicroserviceSupport": "Tukeeko se mikropalveluarkkitehtuuria?",
+ "MicroserviceSupportExplanation1": "Yksi sivutuotekehyksen tärkeimmistä tavoitteista on tarjota kätevä infrastruktuuri mikropalveluratkaisujen luomiseksi. Katso mikropalveluarkkitehtuuri -asiakirjasta, miten se auttaa luomaan mikropalvelujärjestelmiä.",
+ "MicroserviceSupportExplanation2": "Kaikki ABP Commercial -moduulit on suunniteltu tukemaan mikropalvelujen käyttöönottotilanteita (omalla sovellusliittymällä ja tietokannalla) noudattamalla moduulien kehittämisen parhaita käytäntöjä -asiakirjaa.",
+ "MicroserviceSupportExplanation3": "Tarjoamme esimerkin mikropalveluiden esittelyratkaisusta , joka osoittaa mikropalveluarkkitehtuurin toteutuksen auttaakseen sinua luomaan oman ratkaisun.",
+ "MicroserviceSupportExplanation4": "Lyhyt vastaus on \" kyllä, se tukee mikropalveluarkkitehtuuria \".",
+ "MicroserviceSupportExplanation5": "Mikropalvelujärjestelmä on kuitenkin ratkaisu, ja jokaisella ratkaisulla on erilaiset vaatimukset, verkon topologia, viestintätavat, todennusmahdollisuudet, tietokannan erottamis- / jakamispäätökset, ajonaikaiset kokoonpanot, kolmannen osapuolen järjestelmäintegraatiot ja paljon muuta.",
+ "MicroserviceSupportExplanation6": "ABP Framework ja ABP Commercial tarjoavat infrastruktuurin mikropalvelumalleja varten, mikropalvelujen kanssa yhteensopivia moduuleja, näytteitä ja dokumentaatiota, jotka auttavat sinua rakentamaan oman ratkaisun. Mutta älä odota lataavan unelmaratkaisua suoraan sinulle valmiiksi. Sinun täytyy ymmärtää se ja tuoda joitain osia toiveidesi mukaan.",
+ "WhereCanIDownloadSourceCode": "Mistä voin ladata lähdekoodin?",
+ "WhereCanIDownloadSourceCodeExplanation": "Voit ladata kaikkien ABP-moduulien, kulmapakettien ja teemojen lähdekoodin ABP Suite- tai ABP CLI -palvelun kautta. Katso Lähdekoodin lataaminen? ",
+ "ComputerLimitation": "Kuinka monta tietokonetta kehittäjä voi kirjautua kehittäessään ABP: tä?",
+ "ComputerLimitationExplanation": "Sallimme erikseen {0} tietokoneet yksittäistä / lisensoitua kehittäjää kohti. Aina kun kehittäjän on tarpeen kehittää ABP Commercial -tuotteita kolmannella koneella, on lähetettävä sähköpostiviesti osoitteeseen license@abp.io, jossa selitetään tilanne, ja sitten suoritamme sopivan varauksen järjestelmäämme.",
+ "RefundPolicy": "Onko sinulla hyvityskäytäntö?",
+ "RefundPolicyExplanation": "Voit pyytää hyvitystä 30 päivän kuluessa lisenssiostoksestasi. Yritys- ja yrityslisenssityypeillä on lähdekoodin latausvaihtoehto, joten hyvityksiä ei ole saatavana yrityksille ja yrityksille (eikä lisensseille, jotka sisältävät oikeuden saada lähdekoodia). Lisäksi uusimista ja toisen lisenssin ostoja ei hyvitetä.",
+ "HowCanIRefundVat": "Kuinka voin palauttaa arvonlisäveron?",
+ "HowCanIRefundVatExplanation1": "Jos suoritit maksun käyttämällä 2Checkoutia, voit palauttaa arvonlisäveron 2Checkout-tilisi kautta:",
+ "HowCanIRefundVatExplanation2": "Kirjaudu sisään 2Tarkista -tiliisi.",
+ "HowCanIRefundVatExplanation3": "Etsi sopiva tilaus ja paina \"Hyvitä myöhässä oleva alv\" (kirjoita ALV-tunnuksesi)",
+ "HowCanIGetMyInvoice": "Kuinka saan laskuni?",
+ "HowCanIGetMyInvoiceExplanation": "Lisenssin ostamiseen on 2 maksuyhdyskäytävää: PayU ja 2Checkout. Jos ostat lisenssin 2Checkout gatewayn kautta, se lähettää PDF-laskun sähköpostiosoitteeseesi, katso 2Tarkista laskutus. Jos ostat PayU-yhdyskäytävän kautta tai pankkisiirrolla, valmistelemme ja lähetämme laskusi. Voit pyytää laskua organisaation hallintasivulta ",
+ "Forum": "Foorumi",
+ "SupportExplanation": "ABP Commercial -lisenssit tarjoavat ensiluokkaisen foorumituen ABP-kehyksen asiantuntijoista koostuvalle tiimille.",
+ "PrivateTicket": "Yksityinen lippu",
+ "PrivateTicketExplanation": "Yrityslisenssi sisältää myös yksityisen tuen sähköposti- ja lippujärjestelmällä.",
+ "AbpSuiteExplanation1": "ABP Suiten avulla voit luoda verkkosivuja muutamassa minuutissa. Se on .NET Core Global -työkalu, joka voidaan asentaa komentoriviltä.",
+ "AbpSuiteExplanation2": "Se voi luoda uuden ABP-ratkaisun, luoda CRUD-sivuja tietokannasta käyttöliittymään. Katso tekninen yleiskatsaus asiakirjasta ",
+ "FastEasy": "Nopea ja helppo",
+ "AbpSuiteExplanation3": "ABP Suiten avulla voit luoda helposti CRUD-sivuja. Sinun tarvitsee vain määritellä entiteetti ja sen ominaisuudet, antaa loput ABP Suiten puolestasi! ABP Suite luo kaikki tarvittavat koodit CRUD-sivullesi muutamassa sekunnissa. Se tukee Angular-, MVC- ja Blazor-käyttöliittymiä.",
+ "RichOptions": "Monipuoliset vaihtoehdot",
+ "AbpSuiteExplanation4": "ABP Suite tukee useita käyttöliittymävaihtoehtoja, kuten Razor Pages ja Kulma .Se tukee myös useita tietokantoja, kuten MongoDB ja kaikkia EntityFramework Core <: n tukemia tietokantoja. / strong> (MS SQL Server, Oracle, MySql, PostgreSQL ja lisää ).",
+ "AbpSuiteExplanation5": "Hyvä asia on, että sinun ei tarvitse huolehtia näistä vaihtoehdoista. ABP Suite ymmärtää projektisi tyypin ja luo koodin projektillesi ja sijoittaa luodun koodin oikeaan paikkaan projektissasi.",
+ "SourceCode": "Lähdekoodi",
+ "AbpSuiteExplanation6": "ABP Suite luo lähdekoodin sinulle! Se ei luo taikuustiedostoja verkkosivun luomiseksi. ABP Suite luo lähdekoodin Entity, Repository, Application Service, Code First Migration, JavaScript / TypeScript ja CSHTML / HTML sekä tarvittavat liitännät. ABP Suite tuottaa koodin myös ohjelmistokehityksen parhaiden käytäntöjen mukaisesti, joten sinun ei tarvitse huolehtia luodun koodin laadusta.",
+ "AbpSuiteExplanation7": "Koska luotun CRUD-sivun rakennuspalikoiden lähdekoodi on oikeissa sovelluskerroksissa, voit helposti muokata lähdekoodia ja pistää mukautetun / bussiness-logiikan luotuun koodiin.",
+ "CrossPlatform": "Cross Platform",
+ "AbpSuiteExplanation8": "ABP Suite on rakennettu .NET Core -sovelluksella ja se on alustojen välinen. Se toimii verkkosovelluksena paikallisella tietokoneellasi. Voit käyttää sitä Windows , Mac ja Linux ",
+ "OtherFeatures": "Muut ominaisuudet",
+ "OtherFeatures1": "Päivittää ratkaisusi NuGet ja NPM -paketit helposti.",
+ "OtherFeatures2": "Palauttaa jo luodut sivut tyhjästä.",
+ "OtherFeatures3": "Luo uusia ratkaisuja",
+ "ThanksForCreatingProject": "Kiitos projektin luomisesta!",
+ "HotToRunSolution": "Kuinka ajaa ratkaisusi?",
+ "HotToRunSolutionExplanation": "Katso aloitusasiakirjasta lisätietoja ratkaisun määrittämisestä ja ajamisesta.",
+ "GettingStarted": "Päästä alkuun",
+ "WebAppDevTutorial": "Verkkosovellusten kehittäjän opetusohjelma",
+ "WebAppDevTutorialExplanation": "Katso vaiheittainen kehitysnäyte verkkosovellusten kehittämisoppaasta.",
+ "Document": "Asiakirja",
+ "UsingABPSuiteToCURD": "ABP Suiten käyttäminen CRUD-sivunluontia ja työkaluja varten",
+ "SeeABPSuiteDocument": "Katso ABP Suite -asiakirjasta ABP Suiten käyttö.",
+ "AskQuestionsOnSupport": "Voit esittää kysymyksiä ABP: n kaupallisesta tuesta.",
+ "Documentation": "Dokumentointi",
+ "SeeModulesDocument": "Katso moduulidokumentista luettelo kaikista kaupallisista (pro) moduuleista ja niiden asiakirjoista.",
+ "Pricing": "Hinnoittelu",
+ "PricingExplanation": "Valitse ominaisuudet ja toiminnot, joita yrityksesi tarvitsee tänään. Päivitä helposti, kun yrityksesi kasvaa.",
+ "Team": "Tiimi",
+ "Business": "Liiketoiminta",
+ "Enterprise": "Yritys",
+ "Custom": "Mukautettu",
+ "IncludedDeveloperLicenses": "Mukana kehittäjälisenssit",
+ "CustomLicenceOrAdditionalServices": "Tarvitsetko mukautettua lisenssiä tai lisäpalveluja?",
+ "CustomOrVolumeLicense": "Mukautettu tai volyymilisenssi",
+ "LiveTrainingSupport": "Live-koulutus ja tuki",
+ "AndMore": "ja enemmän",
+ "AdditionalDeveloperLicense": "Kehittäjän lisenssi",
+ "ProjectCount": "Projektilaskenta",
+ "AllProModules": "Kaikki pro-moduulit",
+ "AllProThemes": "Kaikki pro-teemat",
+ "AllProStartupTemplates": "Kaikki pro-käynnistysmallit",
+ "SourceCodeOfAllModules": "Kaikkien moduulien lähdekoodi",
+ "SourceCodeOfAllThemes": "Kaikkien teemojen lähdekoodi",
+ "PerpetualLicense": "Jatkuva lisenssi",
+ "UnlimitedServerDeployment": "Rajoittamaton palvelimen käyttöönotto",
+ "YearUpgrade": "1 vuoden päivitys",
+ "YearPremiumForumSupport": "Yhden vuoden premium-foorumin tuki",
+ "ForumSupportIncidentCountYear": "Foorumin tuen tapahtumien määrä / vuosi",
+ "PrivateTicketEmailSupport": "Yksityinen lippu- ja sähköpostituki",
+ "BuyNow": "Osta nyt",
+ "PayViaAmexCard": "Kuinka voin maksaa AMEX-korttini kautta?",
+ "PayViaAmexCardDescription": "Oletusmaksuyhdyskäytävä 'Iyzico' voi hylätä tietyt AMEX-luottokortit turvatoimenpiteiden takia. Tässä tapauksessa voit maksaa vaihtoehtoisen maksuyhdyskäytävän '2Checkout' kautta.",
+ "ThankYou": "Kiitos",
+ "InvalidReCaptchaErrorMessage": "ReCAPTCHA: n vahvistamisessa tapahtui virhe. Yritä uudelleen."
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/FR.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/FR.json
new file mode 100644
index 0000000000..8c382f760e
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/FR.json
@@ -0,0 +1,144 @@
+{
+ "culture": "fr",
+ "texts": {
+ "Permission:CommunityArticle": "Article communautaire",
+ "Permission:Edit": "Éditer",
+ "Waiting": "Attendre",
+ "Approved": "Approuvé",
+ "Rejected": "Rejeté",
+ "Wait": "Attendez",
+ "Approve": "Approuver",
+ "Reject": "Rejeter",
+ "ReadArticle": "Lire l'article",
+ "Status": "Statut",
+ "ContentSource": "Source du contenu",
+ "Details": "Des détails",
+ "Url": "URL",
+ "Title": "Titre",
+ "CreationTime": "Temps de creation",
+ "Save": "Sauvegarder",
+ "SameUrlAlreadyExist": "La même URL existe déjà si vous souhaitez ajouter cet article, vous devez changer l'url!",
+ "UrlIsNotValid": "L'URL n'est pas valide.",
+ "UrlNotFound": "URL introuvable.",
+ "UrlContentNotFound": "Contenu de l'URL introuvable.",
+ "Summary": "Résumé",
+ "MostRead": "Les plus lus",
+ "Latest": "Dernier",
+ "ContributeAbpCommunity": "Contribuez à la communauté ABP",
+ "SubmitYourArticle": "Soumettez votre message",
+ "ContributionGuide": "Guide de contribution",
+ "BugReport": "Rapport d'erreur",
+ "SeeAllArticles": "Voir tous les articles",
+ "WelcomeToABPCommunity!": "Bienvenue dans la communauté ABP!",
+ "MyProfile": "Mon profil",
+ "MyOrganizations": "Mes organisations",
+ "EmailNotValid": "S'il vous plaît, mettez une adresse email valide.",
+ "FeatureRequest": "Demande de fonctionnalité",
+ "CreateArticleTitleInfo": "Titre du message à afficher dans la liste des messages.",
+ "CreateArticleSummaryInfo": "Un bref résumé du message à afficher dans la liste des messages.",
+ "CreateArticleCoverInfo": "Pour créer un article efficace, ajoutez une photo de couverture. Téléchargez des images au format 16: 9 pour une meilleure vue. Taille maximale du fichier: 1 Mo.",
+ "ThisExtensionIsNotAllowed": "Cette extension n'est pas autorisée.",
+ "TheFileIsTooLarge": "Le fichier est trop volumineux.",
+ "GoToTheArticle": "Aller à l'article",
+ "Contribute": "Contribuer",
+ "OverallProgress": "Les progrès d'ensemble",
+ "Done": "Fait",
+ "Open": "Ouvert",
+ "Closed": "Fermé",
+ "LatestQuestionOnThe": "Dernière question sur le",
+ "Stackoverflow": "Stackoverflow",
+ "Votes": "les votes",
+ "Answer": "Répondre",
+ "Views": "vues",
+ "Answered": "Répondu",
+ "WaitingForYourAnswer": "J'attends ta réponse",
+ "Asked": "demandé",
+ "AllQuestions": "Toutes les questions",
+ "NextVersion": "Version suivante",
+ "MilestoneErrorMessage": "Impossible d'obtenir les détails de l'étape actuelle à partir de Github.",
+ "QuestionItemErrorMessage": "Impossible d'obtenir les derniers détails de la question de Stackoverflow.",
+ "Oops": "Oops!",
+ "CreateArticleSuccessMessage": "L'article a été soumis avec succès. Il sera publié après un examen de l'administrateur du site.",
+ "ChooseCoverImage": "Choisissez une image de couverture ...",
+ "CoverImage": "Image de couverture",
+ "ShareYourExperiencesWithTheABPFramework": "Partagez vos expériences avec le Framework ABP!",
+ "Optional": "Optionnel",
+ "UpdateUserWebSiteInfo": "Exemple: https://johndoe.com",
+ "UpdateUserTwitterInfo": "Exemple: johndoe",
+ "UpdateUserGithubInfo": "Exemple: johndoe",
+ "UpdateUserLinkedinInfo": "Exemple: https: //www.linkedin.com / ...",
+ "UpdateUserCompanyInfo": "Exemple: Volosoft",
+ "UpdateUserJobTitleInfo": "Exemple: développeur de logiciels",
+ "UserName": "Nom d'utilisateur",
+ "Company": "Compagnie",
+ "PersonalWebsite": "Site Web personnel",
+ "RegistrationDate": "Date d'inscription",
+ "Social": "Social",
+ "Biography": "Biographie",
+ "HasNoPublishedArticlesYet": "n'a pas encore d'articles publiés",
+ "Author": "Auteur",
+ "LatestGithubAnnouncements": "Dernières annonces Github",
+ "SeeAllAnnouncements": "Voir toutes les annonces",
+ "LatestBlogPost": "Dernier article de blog",
+ "Edit": "Éditer",
+ "ProfileImageChange": "Changer l'image de profil",
+ "BlogItemErrorMessage": "Impossible d'obtenir les derniers détails du billet de blog d'ABP.",
+ "PlannedReleaseDate": "Date de sortie prévue",
+ "CommunityArticleRequestErrorMessage": "Impossible d'obtenir la dernière demande d'article de Github.",
+ "ArticleRequestFromGithubIssue": "Il n'y a actuellement aucune demande d'article.",
+ "LatestArticles": "Derniers messages",
+ "ArticleRequests": "Demandes d'articles",
+ "AllArticleRequests": "Voir toutes les demandes d'articles",
+ "SubscribeToTheNewsletter": "Abonnez-vous à la newsletter",
+ "NewsletterEmailDefinition": "Obtenez des informations sur les événements d'ABP, comme les nouvelles versions, les sources gratuites, les articles, etc.",
+ "NoThanks": "Non merci",
+ "MaybeLater": "Peut-être plus tard",
+ "JoinOurArticleNewsletter": "Rejoignez notre newsletter d'article",
+ "Community": "Communauté",
+ "Marketing": "Commercialisation",
+ "CommunityPrivacyPolicyConfirmation": "J'accepte les conditions générales et la politique de confidentialité .",
+ "ArticleRequestMessageTitle": " Ouvrez un problème sur le GitHub pour demander un article / didacticiel que vous souhaitez voir sur ce site Web.",
+ "ArticleRequestMessageBody": "Ici, la liste des articles demandés par la communauté. Voulez-vous écrire un article demandé? Veuillez cliquer sur la demande et vous joindre à la discussion.",
+ "Language": "Langue",
+ "CreateArticleLanguageInfo": "La langue du contenu de l'article.",
+ "VideoPost": "Message vidéo",
+ "Article": "Article",
+ "Read": "Lis",
+ "CreateGithubArticleUrlInfo": "URL GitHub d'origine de l'article.",
+ "CreateVideoContentUrlInfo": "URL Youtube d'origine du message.",
+ "CreateExternalArticleUrlInfo": "URL externe d'origine de l'article.",
+ "VideoContentForm": "Soumettre une vidéo sur YouTube",
+ "GithubPostForm": "Soumettre un article sur GitHub",
+ "ExternalPostForm": "Soumettre un contenu externe",
+ "HowToPost": "Comment publier?",
+ "Posts": "Des postes",
+ "VideoUrl": "URL de la vidéo",
+ "GithubArticleUrl": "URL de l'article Github",
+ "ExternalArticleUrl": "URL de l'article externe",
+ "CreatePostCoverInfo": "Pour créer un article efficace, ajoutez une photo de couverture. Téléchargez des images au format 16: 9 pour une meilleure vue. Taille maximale du fichier: 1 Mo.",
+ "ThankYouForContribution": "Merci de contribuer à la communauté ABP.",
+ "GithubArticle": "Article Github",
+ "GithubArticleSubmitStepOne": " 1. Rédigez un article sur n'importe quel référentiel GitHub public au format Markdown. Exemple de ",
+ "GithubArticleSubmitStepTwo": " 2. Envoyez l'URL de votre article à l'aide du formulaire.",
+ "GithubArticleSubmitStepThree": " 3. Votre article sera rendu sur ce site Web.",
+ "YoutubeVideo": "Vidéo Youtube",
+ "YoutubeVideoSubmitStepOne": " 1. Publiez votre vidéo sur YouTube.",
+ "YoutubeVideoSubmitStepTwo": " 2. Envoyez l'URL de la vidéo à l'aide du formulaire.",
+ "YoutubeVideoSubmitStepThree": " 3. Les visiteurs pourront visionner votre contenu vidéo directement sur ce site Web.",
+ "ExternalContent": "Contenu externe",
+ "ExternalContentSubmitStepOne": " 1. Créez un contenu sur n'importe quelle plate-forme publique (support, votre propre blog ou partout où vous le souhaitez).",
+ "ExternalContentSubmitStepTwo": " 2. Envoyez votre URL de contenu à l'aide du formulaire.",
+ "ExternalContentSubmitStepThree": " 3. Les visiteurs sont redirigés vers le contenu du site Web d'origine.",
+ "ChooseYourContentType": "Veuillez choisir la manière dont vous souhaitez ajouter votre contenu.",
+ "PostContentViaGithub": "Je souhaite ajouter mon article avec GitHub conformément aux règles de démarque.",
+ "PostContentViaYoutube": "Je souhaite partager mes vidéos disponibles sur Youtube ici.",
+ "PostContentViaExternalSource": "Je souhaite ajouter le contenu que j'ai publié sur une autre plate-forme ici.",
+ "GitHubUserNameValidationMessage": "Votre nom d'utilisateur Github ne peut pas inclure d'espaces, veuillez vous assurer que votre nom d'utilisateur Github est correct.",
+ "PersonalSiteUrlValidationMessage": "L'URL de votre site personnel ne peut pas inclure d'espaces, veuillez vous assurer que l'URL de votre site personnel est correcte.",
+ "TwitterUserNameValidationMessage": "Votre nom d'utilisateur Twitter ne peut pas inclure d'espaces, veuillez vous assurer que votre nom d'utilisateur Twitter est correct.",
+ "LinkedinUrlValidationMessage": "Votre URL Linkedin ne peut pas inclure d'espace blanc, veuillez vous assurer que votre URL Linkedin est correcte.",
+ "NoPostsFound": "Aucun article trouvé!",
+ "SearchInPosts": "Rechercher dans les messages ...",
+ "MinimumSearchContent": "Vous devez saisir au moins 3 caractères!"
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json
index 82033e5849..7be579f1fb 100644
--- a/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/en.json
@@ -139,6 +139,9 @@
"LinkedinUrlValidationMessage": "Your Linkedin URL can not include whitespace, please be sure your Linkedin URL is correct.",
"NoPostsFound": "No posts found!",
"SearchInPosts": "Search in posts...",
- "MinimumSearchContent": "You must enter at least 3 characters!"
+ "MinimumSearchContent": "You must enter at least 3 characters!",
+ "Volo.AbpIo.Domain:060001": "Source URL(\"{ArticleUrl}\") is not Github URL",
+ "Volo.AbpIo.Domain:060002": "Article Content is not available from Github(\"{ArticleUrl}\") resource.",
+ "Volo.AbpIo.Domain:060003": "No article content found!"
}
}
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/fi.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/fi.json
new file mode 100644
index 0000000000..cba33bfc30
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Community/Localization/Resources/fi.json
@@ -0,0 +1,144 @@
+{
+ "culture": "fi",
+ "texts": {
+ "Permission:CommunityArticle": "Yhteisön artikkeli",
+ "Permission:Edit": "Muokata",
+ "Waiting": "Odottaa",
+ "Approved": "Hyväksytty",
+ "Rejected": "Hylätty",
+ "Wait": "Odota",
+ "Approve": "Hyväksyä",
+ "Reject": "Hylätä",
+ "ReadArticle": "Lue artikkeli",
+ "Status": "Tila",
+ "ContentSource": "Sisältölähde",
+ "Details": "Yksityiskohdat",
+ "Url": "URL-osoite",
+ "Title": "Otsikko",
+ "CreationTime": "Luomisaika",
+ "Save": "Tallentaa",
+ "SameUrlAlreadyExist": "Sama URL-osoite on jo olemassa, jos haluat lisätä tämän artikkelin, vaihda URL-osoite!",
+ "UrlIsNotValid": "URL-osoite ei kelpaa.",
+ "UrlNotFound": "URL-osoitetta ei löydy.",
+ "UrlContentNotFound": "URL-osoitteen sisältöä ei löydy.",
+ "Summary": "Yhteenveto",
+ "MostRead": "Luetuimmat",
+ "Latest": "Viimeisin",
+ "ContributeAbpCommunity": "Osallistu sivutuotteiden yhteisöön",
+ "SubmitYourArticle": "Lähetä viesti",
+ "ContributionGuide": "Contribution Guide",
+ "BugReport": "Virhe raportti",
+ "SeeAllArticles": "Katso kaikki viestit",
+ "WelcomeToABPCommunity!": "Tervetuloa ABP-yhteisöön!",
+ "MyProfile": "Profiilini",
+ "MyOrganizations": "Omat organisaatiot",
+ "EmailNotValid": "Ole hyvä ja syötä toimiva sähköpostiosoite.",
+ "FeatureRequest": "Ominaisuuspyyntö",
+ "CreateArticleTitleInfo": "Viestiluettelossa näytettävän viestin nimi.",
+ "CreateArticleSummaryInfo": "Lyhyt yhteenveto viestistä, joka näytetään postituslistalla.",
+ "CreateArticleCoverInfo": "Lisää tehokkaan artikkelin luomiseksi kansikuva. Lataa 16: 9-kuvasuhteen kuvat parhaan näkymän saamiseksi. Tiedoston enimmäiskoko: 1 Mt.",
+ "ThisExtensionIsNotAllowed": "Tätä laajennusta ei sallita.",
+ "TheFileIsTooLarge": "Tiedosto on liian suuri.",
+ "GoToTheArticle": "Siirry artikkeliin",
+ "Contribute": "Osallistu",
+ "OverallProgress": "Kokonaisedistyminen",
+ "Done": "Tehty",
+ "Open": "Avata",
+ "Closed": "Suljettu",
+ "LatestQuestionOnThe": "Viimeisin kysymys",
+ "Stackoverflow": "Pinoaminen",
+ "Votes": "ääntä",
+ "Answer": "Vastaus",
+ "Views": "näkymät",
+ "Answered": "Vastasi",
+ "WaitingForYourAnswer": "Odotan vastaustasi",
+ "Asked": "kysyi",
+ "AllQuestions": "Kaikki kysymykset",
+ "NextVersion": "Seuraava versio",
+ "MilestoneErrorMessage": "Nykyisiä virstanpylväitä koskevia tietoja ei saatu Githubilta.",
+ "QuestionItemErrorMessage": "Viimeisimmät kysymystiedot Stackoverflow'sta epäonnistui.",
+ "Oops": "Oho!",
+ "CreateArticleSuccessMessage": "Artikkeli on lähetetty onnistuneesti. Se julkaistaan sivuston järjestelmänvalvojan tarkistuksen jälkeen.",
+ "ChooseCoverImage": "Valitse kansikuva ...",
+ "CoverImage": "Kansikuva",
+ "ShareYourExperiencesWithTheABPFramework": "Jaa kokemuksesi ABP-puitteista!",
+ "Optional": "Valinnainen",
+ "UpdateUserWebSiteInfo": "Esimerkki: https://johndoe.com",
+ "UpdateUserTwitterInfo": "Esimerkki: johndoe",
+ "UpdateUserGithubInfo": "Esimerkki: johndoe",
+ "UpdateUserLinkedinInfo": "Esimerkki: https: //www.linkedin.com / ...",
+ "UpdateUserCompanyInfo": "Esimerkki: Volosoft",
+ "UpdateUserJobTitleInfo": "Esimerkki: Ohjelmistokehittäjä",
+ "UserName": "Käyttäjänimi",
+ "Company": "Yhtiö",
+ "PersonalWebsite": "Henkilökohtainen verkkosivusto",
+ "RegistrationDate": "rekisteröinti päivämäärä",
+ "Social": "Sosiaalinen",
+ "Biography": "Elämäkerta",
+ "HasNoPublishedArticlesYet": "ei ole vielä julkaissut artikkeleita",
+ "Author": "Kirjoittaja",
+ "LatestGithubAnnouncements": "Viimeisimmät Github-ilmoitukset",
+ "SeeAllAnnouncements": "Katso kaikki ilmoitukset",
+ "LatestBlogPost": "Viimeisin blogiviesti",
+ "Edit": "Muokata",
+ "ProfileImageChange": "Vaihda profiilikuva",
+ "BlogItemErrorMessage": "Viimeisimpiä blogiviestitietoja ei saatu ABP: ltä.",
+ "PlannedReleaseDate": "Suunniteltu julkaisupäivä",
+ "CommunityArticleRequestErrorMessage": "Uusinta artikkelipyyntöä ei saatu Githubilta.",
+ "ArticleRequestFromGithubIssue": "Artikkelipyyntöjä ei ole nyt.",
+ "LatestArticles": "Uusimmat viestit",
+ "ArticleRequests": "Artikkelipyynnöt",
+ "AllArticleRequests": "Katso kaikki artikkelipyynnöt",
+ "SubscribeToTheNewsletter": "Tilaa uutiskirje",
+ "NewsletterEmailDefinition": "Hanki tietoa ABP: n tapahtumista, kuten uusista julkaisuista, ilmaisista lähteistä, artikkeleista ja muusta.",
+ "NoThanks": "Ei kiitos",
+ "MaybeLater": "Ehkä myöhemmin",
+ "JoinOurArticleNewsletter": "Liity artikkeliuutiskirjeeseemme",
+ "Community": "Yhteisö",
+ "Marketing": "Markkinointi",
+ "CommunityPrivacyPolicyConfirmation": "Hyväksyn käyttöehdot ja tietosuojakäytännön .",
+ "ArticleRequestMessageTitle": " Avaa ongelma GitHubissa pyytääksesi artikkelia / opetusohjelmaa, jonka haluat nähdä tällä verkkosivustolla.",
+ "ArticleRequestMessageBody": "Tässä luettelo yhteisön pyytämistä artikkeleista. Haluatko kirjoittaa pyydetyn artikkelin? Napsauta pyyntöä ja liity keskusteluun.",
+ "Language": "Kieli",
+ "CreateArticleLanguageInfo": "Viestin sisällön kieli.",
+ "VideoPost": "Videoposti",
+ "Article": "Artikla",
+ "Read": "Lukea",
+ "CreateGithubArticleUrlInfo": "Artikkelin alkuperäinen GitHub-URL-osoite.",
+ "CreateVideoContentUrlInfo": "Viestin alkuperäinen Youtube-URL-osoite.",
+ "CreateExternalArticleUrlInfo": "Artikkelin alkuperäinen ulkoinen URL-osoite.",
+ "VideoContentForm": "Lähetä video YouTubessa",
+ "GithubPostForm": "Lähetä artikkeli GitHubista",
+ "ExternalPostForm": "Lähetä ulkoinen sisältö",
+ "HowToPost": "Kuinka lähettää?",
+ "Posts": "Viestit",
+ "VideoUrl": "Videon URL-osoite",
+ "GithubArticleUrl": "Github-artikkelien URL-osoite",
+ "ExternalArticleUrl": "Ulkoisen artikkelin URL-osoite",
+ "CreatePostCoverInfo": "Lisää kansikuva, jotta voit luoda tehokkaan viestin. Lataa 16: 9-kuvasuhteen kuvat parhaan näkymän saamiseksi. Tiedoston enimmäiskoko: 1 Mt.",
+ "ThankYouForContribution": "Kiitos osallistumisesta ABP-yhteisöön.",
+ "GithubArticle": "Github-artikkeli",
+ "GithubArticleSubmitStepOne": " 1. Kirjoita artikkeli mistä tahansa julkisesta GitHub-arkistosta Markdown-muodossa. esimerkki ",
+ "GithubArticleSubmitStepTwo": " 2. Lähetä artikkelin URL-osoite lomaketta käyttämällä.",
+ "GithubArticleSubmitStepThree": " 3. Artikkelisi renderöidään tällä verkkosivustolla.",
+ "YoutubeVideo": "Youtube-video",
+ "YoutubeVideoSubmitStepOne": " 1. Julkaise videosi YouTubessa.",
+ "YoutubeVideoSubmitStepTwo": " 2. Lähetä videon URL-osoite lomaketta käyttäen.",
+ "YoutubeVideoSubmitStepThree": " 3. Vierailijat voivat katsella videosisältöäsi suoraan tällä verkkosivustolla.",
+ "ExternalContent": "Ulkoinen sisältö",
+ "ExternalContentSubmitStepOne": " 1. Luo sisältöä mille tahansa julkiselle alustalle (media, oma blogi tai mihin tahansa haluat).",
+ "ExternalContentSubmitStepTwo": " 2. Lähetä sisällön URL-osoite lomaketta käyttämällä.",
+ "ExternalContentSubmitStepThree": " 3. Vierailijat ohjataan alkuperäisen verkkosivuston sisältöön.",
+ "ChooseYourContentType": "Valitse tapa, jolla haluat lisätä sisältöä.",
+ "PostContentViaGithub": "Haluan lisätä artikkelini GitHub -merkintäsääntöjen mukaisesti.",
+ "PostContentViaYoutube": "Haluan jakaa videoni, jotka ovat käytettävissä Youtube täällä.",
+ "PostContentViaExternalSource": "Haluan lisätä toisella alustalla julkaisemani sisällön tähän.",
+ "GitHubUserNameValidationMessage": "Github-käyttäjänimesi ei voi sisältää välilyöntiä. Varmista, että Github-käyttäjänimesi on oikea.",
+ "PersonalSiteUrlValidationMessage": "Henkilökohtaisen sivuston URL-osoite ei voi sisältää välilyöntiä. Varmista, että henkilökohtaisen sivuston URL-osoite on oikea.",
+ "TwitterUserNameValidationMessage": "Twitter-käyttäjänimesi ei voi sisältää välilyöntiä. Varmista, että Twitter-käyttäjänimesi on oikea.",
+ "LinkedinUrlValidationMessage": "Linkedin-URL-osoitteesi ei voi sisältää välilyöntiä. Varmista, että Linkedin-URL-osoitteesi on oikea.",
+ "NoPostsFound": "Viestejä ei löytynyt!",
+ "SearchInPosts": "Hae viesteistä ...",
+ "MinimumSearchContent": "Sinun on annettava vähintään 3 merkkiä!"
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/FR.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/FR.json
new file mode 100644
index 0000000000..b3b8d3e241
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/FR.json
@@ -0,0 +1,202 @@
+{
+ "culture": "fr",
+ "texts": {
+ "GetStarted": "Commencer - Modèles de démarrage",
+ "Create": "Créer",
+ "NewProject": "Nouveau projet",
+ "DirectDownload": "Téléchargement direct",
+ "ProjectName": "Nom du projet",
+ "ProjectType": "Type de projet",
+ "DatabaseProvider": "Fournisseur de base de données",
+ "DatabaseManagementSystem": "Système de gestion de base de données",
+ "NTier": "N-Tier",
+ "IncludeUserInterface": "Inclure l'interface utilisateur",
+ "CreateNow": "Créer maintenant",
+ "TheStartupProject": "Le projet de démarrage",
+ "Tutorial": "Didacticiel",
+ "UsingCLI": "Utilisation de la CLI",
+ "SeeDetails": "Voir les détails",
+ "AbpShortDescription": "ABP Framework est une infrastructure complète pour créer des applications Web modernes en suivant les meilleures pratiques et conventions de développement logiciel.",
+ "SourceCodeUpper": "CODE SOURCE",
+ "LatestReleaseLogs": "Derniers journaux de version",
+ "Infrastructure": "Infrastructure",
+ "Architecture": "Architecture",
+ "Modular": "Modulaire",
+ "DontRepeatYourself": "Ne vous répétez pas",
+ "DeveloperFocused": "Axé sur les développeurs",
+ "FullStackApplicationInfrastructure": "Infrastructure d'application complète.",
+ "DomainDrivenDesign": "Conception pilotée par le domaine",
+ "DomainDrivenDesignExplanation": "Conçu et développé sur la base des modèles et principes DDD. Fournit un modèle en couches pour votre application.",
+ "Authorization": "Autorisation",
+ "AuthorizationExplanation": "Autorisation avancée avec utilisateur, rôle et système d'autorisation précis. Construit sur la bibliothèque d'identité Microsoft.",
+ "MultiTenancy": "Locations multiples",
+ "MultiTenancyExplanationShort": "Les applications SaaS simplifiées! Multi-location intégrée de la base de données à l'interface utilisateur.",
+ "CrossCuttingConcerns": "Préoccupations transversales",
+ "CrossCuttingConcernsExplanationShort": "Infrastructure complète pour l'autorisation, la validation, la gestion des exceptions, la mise en cache, la journalisation d'audit, la gestion des transactions et plus encore.",
+ "BuiltInBundlingMinification": "Regroupement et minification intégrés",
+ "BuiltInBundlingMinificationExplanation": "Pas besoin d'utiliser des outils externes pour le regroupement et la minimisation. ABP offre un moyen plus simple, dynamique, puissant, modulaire et intégré!",
+ "VirtualFileSystem": "Système de fichiers virtuel",
+ "VirtualFileSystemExplanation": "Incorporez des vues, des scripts, des styles, des images ... dans des packages / bibliothèques et réutilisez-les dans différentes applications.",
+ "Theming": "Thème",
+ "ThemingExplanationShort": "Utilisez et personnalisez le thème d'interface utilisateur standard basé sur le bootstrap ou créez le vôtre.",
+ "BootstrapTagHelpersDynamicForms": "Aide aux balises Bootstrap et formulaires dynamiques",
+ "BootstrapTagHelpersDynamicFormsExplanation": "Au lieu d'écrire manuellement les détails répétitifs des composants d'amorçage, utilisez les assistants de balises d'ABP pour le simplifier et tirer parti d'intellisense. Créez rapidement des formulaires d'interface utilisateur basés sur un modèle C # à l'aide de l'assistant de balise de formulaire dynamique.",
+ "HTTPAPIsDynamicProxies": "API HTTP et proxys dynamiques",
+ "HTTPAPIsDynamicProxiesExplanation": "Exposez automatiquement les services d'application en tant qu'API HTTP de style REST et utilisez-les avec des proxys JavaScript et C # dynamiques.",
+ "CompleteArchitectureInfo": "Architecture moderne pour créer des solutions logicielles maintenables.",
+ "DomainDrivenDesignBasedLayeringModelExplanation": "Vous aide à implémenter une architecture en couches basée sur DDD et à créer une base de code maintenable.",
+ "DomainDrivenDesignBasedLayeringModelExplanationCont": "Fournit des modèles de démarrage, des abstractions, des classes de base, des services, de la documentation et des guides pour vous aider à développer votre application basée sur les modèles et principes DDD.",
+ "MicroserviceCompatibleModelExplanation": "Le cadre de base et les modules de pré-construction sont conçus avec l'architecture de microservice à l'esprit.",
+ "MicroserviceCompatibleModelExplanationCont": "Fournit une infrastructure, des intégrations, des exemples et de la documentation pour mettre en œuvre des solutions de microservices plus facilement, sans apporter de complexité supplémentaire si vous souhaitez une application monolithique.",
+ "ModularInfo": "ABP fournit un système de modules qui vous permet de développer des modules d'application réutilisables, de les relier aux événements du cycle de vie des applications et d'exprimer les dépendances entre les parties centrales de votre système.",
+ "PreBuiltModulesThemes": "Modules et thèmes prédéfinis",
+ "PreBuiltModulesThemesExplanation": "Les modules et thèmes open source et commerciaux sont prêts à être utilisés dans votre application professionnelle.",
+ "NuGetNPMPackages": "Packages NuGet et NPM",
+ "NuGetNPMPackagesExplanation": "Distribué sous forme de packages NuGet et NPM. Facile à installer et à mettre à jour.",
+ "ExtensibleReplaceable": "Extensible / remplaçable",
+ "ExtensibleReplaceableExplanation": "Tous les services et modules sont conçus pour l'extensibilité. Vous pouvez remplacer des services, des pages, des styles et des composants.",
+ "CrossCuttingConcernsExplanation2": "Réduisez la taille de votre base de code afin de rester concentré sur le code spécifique à votre entreprise.",
+ "CrossCuttingConcernsExplanation3": "N'envoyez pas de temps pour mettre en œuvre les exigences d'application communes sur plusieurs projets.",
+ "AuthenticationAuthorization": "Autorisation d'authentification",
+ "ExceptionHandling": "Gestion des exceptions",
+ "Validation": "Validation",
+ "DatabaseConnection": "Connexion à la base de données",
+ "TransactionManagement": "Gestion des transactions",
+ "AuditLogging": "Journalisation d'audit",
+ "Caching": "Mise en cache",
+ "Multitenancy": "Locations multiples",
+ "DataFiltering": "Filtrage des données",
+ "ConventionOverConfiguration": "Convention sur la configuration",
+ "ConventionOverConfigurationExplanation": "ABP implémente par défaut des conventions d'application communes avec une configuration minimale ou nulle.",
+ "ConventionOverConfigurationExplanationList1": "Enregistre automatiquement les services connus pour l'injection de dépendances.",
+ "ConventionOverConfigurationExplanationList2": "Expose les services d'application en tant qu'API HTTP par des conventions de dénomination.",
+ "ConventionOverConfigurationExplanationList3": "Crée des proxys client HTTP dynamiques pour C # et JavaScript.",
+ "ConventionOverConfigurationExplanationList4": "Fournit des référentiels par défaut pour vos entités.",
+ "ConventionOverConfigurationExplanationList5": "Gère l'unité de travail par demande Web ou méthode de service d'application.",
+ "ConventionOverConfigurationExplanationList6": "Publie des événements de création, de mise à jour et de suppression pour vos entités.",
+ "BaseClasses": "Classes de base",
+ "BaseClassesExplanation": "Classes de base prédéfinies pour les modèles d'application courants.",
+ "DeveloperFocusedExplanation": "ABP est destiné aux développeurs.",
+ "DeveloperFocusedExplanationCont": "Il vise à simplifier votre développement logiciel quotidien sans vous empêcher d'écrire du code de bas niveau.",
+ "SeeAllFeatures": "Voir toutes les fonctionnalités",
+ "CLI_CommandLineInterface": "CLI (interface de ligne de commande)",
+ "CLI_CommandLineInterfaceExplanation": "Inclut une CLI pour vous aider à automatiser la création de nouveaux projets et l'ajout de nouveaux modules.",
+ "StartupTemplates": "Modèles de démarrage",
+ "StartupTemplatesExplanation": "Divers modèles de démarrage fournissent une solution entièrement configurée pour démarrer rapidement votre développement.",
+ "BasedOnFamiliarTools": "Basé sur des outils familiers",
+ "BasedOnFamiliarToolsExplanation": "Construit sur et intégré avec des outils populaires que vous connaissez déjà. Faible courbe d'apprentissage, adaptation facile, développement confortable.",
+ "ORMIndependent": "ORM indépendant",
+ "ORMIndependentExplanation": "Le framework de base est indépendant de l'ORM / de la base de données et peut fonctionner avec n'importe quelle source de données. Les fournisseurs Entity Framework Core et MongoDB sont déjà disponibles.",
+ "Features": "Explorez les fonctionnalités du framework ABP",
+ "ABPCLI": "CLI ABP",
+ "Modularity": "Modularité",
+ "BootstrapTagHelpers": "Assistants de balises Bootstrap",
+ "DynamicForms": "Formulaires dynamiques",
+ "BundlingMinification": "Regroupement et minification",
+ "BackgroundJobs": "Emplois d'arrière-plan",
+ "BackgroundJobsExplanation": "Définissez des classes simples pour exécuter les travaux en arrière-plan comme mis en file d'attente. Utilisez le gestionnaire de tâches intégré ou intégrez le vôtre. Les intégrations Hangfire et RabbitMQ sont déjà disponibles.",
+ "DDDInfrastructure": "Infrastructure DDD",
+ "DomainDrivenDesignInfrastructure": "Infrastructure de conception pilotée par domaine",
+ "AutoRESTAPIs": "API REST automatiques",
+ "DynamicClientProxies": "Proxys clients dynamiques",
+ "DistributedEventBus": "Bus d'événements distribués",
+ "DistributedEventBusWithRabbitMQIntegration": "Bus d'événements distribués avec intégration RabbitMQ",
+ "TestInfrastructure": "Infrastructure de test",
+ "AuditLoggingEntityHistories": "Journalisation d'audit et historiques d'entités",
+ "ObjectToObjectMapping": "Mappage d'objet à objet",
+ "ObjectToObjectMappingExplanation": " Mappage objet à objet abstraction avec intégration AutoMapper.",
+ "EmailSMSAbstractions": "Abstractions par e-mail et SMS",
+ "EmailSMSAbstractionsWithTemplatingSupport": "Abstractions par e-mail et SMS avec prise en charge de la création de modèles",
+ "Localization": "Localisation",
+ "SettingManagement": "Gestion des paramètres",
+ "ExtensionMethods": "Méthodes d'extension",
+ "ExtensionMethodsHelpers": "Méthodes d'extension et aides",
+ "AspectOrientedProgramming": "Programmation orientée aspect",
+ "DependencyInjection": "Injection de dépendance",
+ "DependencyInjectionByConventions": "Injection de dépendances par conventions",
+ "ABPCLIExplanation": "ABP CLI (Command Line Interface) est un outil de ligne de commande pour effectuer certaines opérations courantes pour les solutions basées sur ABP.",
+ "ModularityExplanation": "ABP fournit une infrastructure complète pour créer vos propres modules d'application qui peuvent avoir des entités, des services, une intégration de base de données, des API, des composants d'interface utilisateur, etc.",
+ "MultiTenancyExplanation": "Le framework ABP prend non seulement en charge le développement d'applications multi-locataires, mais rend également votre code pratiquement inconscient de la multi-location.",
+ "MultiTenancyExplanation2": "Peut déterminer automatiquement le locataire actuel, isoler les données de différents locataires les uns des autres.",
+ "MultiTenancyExplanation3": "Prend en charge une base de données unique, une base de données par locataire et des approches hybrides.",
+ "MultiTenancyExplanation4": "Vous vous concentrez sur votre code métier et laissez le framework gérer la multi-location en votre nom.",
+ "BootstrapTagHelpersExplanation": "Au lieu d'écrire manuellement les détails répétitifs des composants d'amorçage, utilisez les aides de balises d'ABP pour le simplifier et tirer parti d'intellisense. Vous pouvez certainement utiliser Bootstrap chaque fois que vous en avez besoin.",
+ "DynamicFormsExplanation": "Les assistants dynamiques de formulaire et de balise d'entrée peuvent créer le formulaire complet à partir d'une classe C # comme modèle.",
+ "AuthenticationAuthorizationExplanation": "Options d'authentification et d'autorisation riches intégrées à ASP.NET Core Identity & IdentityServer4. Fournit un système d'autorisation extensible et détaillé.",
+ "CrossCuttingConcernsExplanation": "Ne vous répétez pas pour mettre en œuvre toutes ces choses courantes encore et encore. Concentrez-vous sur votre code métier et laissez ABP les automatiser par des conventions.",
+ "DatabaseConnectionTransactionManagement": "Connexion à la base de données et gestion des transactions",
+ "CorrelationIdTracking": "Suivi des identifiants de corrélation",
+ "BundlingMinificationExplanation": "ABP offre un système de regroupement et de minification simple, dynamique, puissant, modulaire et intégré.",
+ "VirtualFileSystemnExplanation": "Le système de fichiers virtuel permet de gérer des fichiers qui n'existent pas physiquement sur le système de fichiers (disque). Il est principalement utilisé pour incorporer des fichiers (js, css, image, cshtml ...) dans des assemblys et les utiliser comme des fichiers physiques lors de l'exécution.",
+ "ThemingExplanation": "Le système de thématisation permet de développer votre application et vos modules indépendamment du thème en définissant un ensemble de bibliothèques et de mises en page de base communes, basées sur le dernier framework Bootstrap.",
+ "DomainDrivenDesignInfrastructureExplanation": "Une infrastructure complète pour créer des applications en couches basées sur les modèles et principes de conception pilotée par domaine;",
+ "Specification": "spécification",
+ "Repository": "Dépôt",
+ "DomainService": "Service de domaine",
+ "ValueObject": "Objet de valeur",
+ "ApplicationService": "Service d'application",
+ "DataTransferObject": "Objet de transfert de données",
+ "AggregateRootEntity": "Racine agrégée, entité",
+ "AutoRESTAPIsExplanation": "ABP peut configurer automatiquement vos services d'application en tant que contrôleurs d'API par convention.",
+ "DynamicClientProxiesExplanation": "Utilisez facilement vos API à partir de clients JavaScript et C #.",
+ "DistributedEventBusWithRabbitMQIntegrationExplanation": "Publiez et consommez facilement des événements distribués à l'aide du bus d'événements distribués intégré avec l'intégration RabbitMQ disponible.",
+ "TestInfrastructureExplanation": "Le cadre a été développé en pensant aux tests unitaires et d'intégration. Vous fournit des classes de base pour vous faciliter la tâche. Les modèles de démarrage sont pré-configurés pour les tests.",
+ "AuditLoggingEntityHistoriesExplanation": "Journalisation d'audit intégrée pour les applications stratégiques. Requête, service, journalisation d'audit au niveau de la méthode et historiques d'entités avec des détails au niveau de la propriété.",
+ "EmailSMSAbstractionsWithTemplatingSupportExplanation": "Les abstractions IEmailSender et ISmsSender découpent la logique de votre application de l'infrastructure. Le système de modèle de courrier électronique avancé permet de créer et de localiser des modèles de courrier électronique et de les utiliser facilement en cas de besoin.",
+ "LocalizationExplanation": "Le système de localisation permet de créer des ressources dans des fichiers JSON simples et de les utiliser pour localiser votre interface utilisateur. Il prend en charge des scénarios avancés tels que l'héritage, les extensions et l'intégration JavaScript tout en étant entièrement compatible avec le système de localisation d'AspNet Core.",
+ "SettingManagementExplanation": "Définissez les paramètres de votre application et obtenez des valeurs à l'exécution en fonction de la configuration, du locataire et de l'utilisateur actuels.",
+ "ExtensionMethodsHelpersExplanation": "Ne vous répétez pas même pour des parties de code triviales. Les extensions et aides pour les types standard rendent votre code beaucoup plus propre et facile à écrire.",
+ "AspectOrientedProgrammingExplanation": "Fournit une infrastructure confortable pour créer des proxys dynamiques et mettre en œuvre la programmation orientée aspect. Interceptez n'importe quelle classe et exécutez votre code avant et après chaque exécution de méthode.",
+ "DependencyInjectionByConventionsExplanation": "Inutile d'inscrire manuellement vos classes à l'injection de dépendances. Enregistre automatiquement les types de services courants par convention. Pour les autres types de services, vous pouvez utiliser des interfaces et des attributs pour le rendre plus simple et sur place.",
+ "DataFilteringExplanation": "Définissez et utilisez des filtres de données qui sont automatiquement appliqués lorsque vous interrogez des entités de la base de données. Les filtres Soft Delete et MultiTenant sont fournis par défaut lorsque vous implémentez des interfaces simples.",
+ "PublishEvents": "Publier des événements",
+ "HandleEvents": "Gérer les événements",
+ "AndMore": "et plus...",
+ "Code": "Code",
+ "Result": "Résultat",
+ "SeeTheDocumentForMoreInformation": "Consultez le {0} document pour plus d'informations",
+ "IndexPageHeroSection": " open source Application Web Framework pour asp.net core ",
+ "UiFramework": "Cadre de l'interface utilisateur",
+ "EmailAddress": "Adresse e-mail",
+ "Mobile": "Mobile",
+ "ReactNative": "Réagir natif",
+ "Strong": "Fort",
+ "Complete": "Compléter",
+ "BasedLayeringModel": "Modèle de couches basé",
+ "Microservice": "Microservice",
+ "Compatible": "Compatible",
+ "MeeTTheABPCommunityInfo": "Notre mission est de créer un environnement où les développeurs peuvent s'entraider avec des articles, des tutoriels, des études de cas, etc. et rencontrer des personnes partageant les mêmes idées.",
+ "JoinTheABPCommunityInfo": "Impliquez-vous dans une communauté dynamique et devenez un contributeur au cadre ABP!",
+ "AllArticles": "Tous les articles",
+ "SubmitYourArticle": "Soumettez votre article",
+ "DynamicClientProxyDocument": "Consultez les documentations sur le proxy client dynamique pour JavaScript et C # .",
+ "EmailSMSAbstractionsDocument": "Consultez les documents e-mail et Envoi de SMS pour plus d'informations.",
+ "CreateProjectWizard": "Cet assistant crée un nouveau projet à partir du modèle de démarrage qui est correctement configuré pour démarrer rapidement votre projet.",
+ "TieredOption": "Crée une solution à plusieurs niveaux dans laquelle les couches API Web et Http sont physiquement séparées. Si elle n'est pas cochée, crée une solution en couches qui est moins complexe et adaptée à la plupart des scénarios.",
+ "SeparateIdentityServerOption": "Sépare le côté serveur en deux applications: la première est pour le serveur d'identité et la seconde pour votre API HTTP côté serveur.",
+ "UseslatestPreVersion": "Utilise la dernière version préliminaire",
+ "ReadTheDocumentation": " Lire La documentation ",
+ "Documentation": "Documentation",
+ "GettingStartedTutorial": "Tutoriel de mise en route",
+ "ApplicationDevelopmentTutorial": "Tutoriel de développement d'applications",
+ "TheStartupTemplate": "Le modèle de démarrage",
+ "InstallABPCLIInfo": "ABP CLI est le moyen le plus rapide de démarrer une nouvelle solution avec le framework ABP. Installez l'interface de ligne de commande ABP à l'aide d'une fenêtre de ligne de commande:",
+ "DifferentLevelOfNamespaces": "Vous pouvez utiliser différents niveaux d'espaces de noms; par exemple. BookStore, Acme.BookStore ou Acme.Retail.BookStore.",
+ "ABPCLIExamplesInfo": "La commande nouvelle crée une application MVC en couches avec Entity Framework Core comme fournisseur de base de données. Cependant, il a des options supplémentaires. Exemples:",
+ "SeeCliDocumentForMoreInformation": "Consultez le document ABP CLI pour plus d'options ou sélectionnez l'onglet \"Téléchargement direct\" ci-dessus.",
+ "Optional": "Optionnel",
+ "LocalFrameworkRef": "Conservez la référence de projet locale pour les packages de structure.",
+ "BlobStoring": "Stockage BLOB",
+ "BlobStoringExplanation": "Le système de stockage BLOB fournit une abstraction pour travailler avec les BLOB. ABP fournit des intégrations de fournisseur de stockage prédéfinies (Azure, AWS, système de fichiers, base de données, etc.) que vous pouvez facilement utiliser dans vos applications.",
+ "TextTemplating": "Création de modèles de texte",
+ "TextTemplatingExplanation": "La création de modèles de texte est utilisée pour rendre le contenu de manière dynamique en fonction d'un modèle et d'un modèle (un objet de données). Par exemple, vous pouvez l'utiliser pour créer des contenus d'e-mail dynamiques avec un modèle prédéfini.",
+ "MultipleUIOptions": "Options d'interface utilisateur multiples",
+ "MultipleDBOptions": "Fournisseurs de bases de données multiples",
+ "MultipleUIOptionsExplanation": "Le cadre de base est conçu comme indépendant de l'interface utilisateur et peut fonctionner avec tout type de système d'interface utilisateur, tandis que plusieurs options prédéfinies et intégrées sont fournies prêtes à l'emploi.",
+ "MultipleDBOptionsExplanation": "Le cadre peut fonctionner avec n'importe quelle source de données, tandis que les fournisseurs suivants sont officiellement développés et pris en charge;",
+ "SelectLanguage": "Choisir la langue",
+ "LatestArticleOnCommunity": "Dernier article sur la communauté ABP",
+ "Register": "S'inscrire",
+ "IsDownloadable": "Est téléchargeable"
+ }
+}
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/fi.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/fi.json
new file mode 100644
index 0000000000..9b15582b38
--- /dev/null
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/fi.json
@@ -0,0 +1,202 @@
+{
+ "culture": "fi",
+ "texts": {
+ "GetStarted": "Aloitus - Käynnistysmallit",
+ "Create": "Luoda",
+ "NewProject": "Uusi projekti",
+ "DirectDownload": "Suora lataus",
+ "ProjectName": "Projektin nimi",
+ "ProjectType": "Projektityyppi",
+ "DatabaseProvider": "Tietokannan tarjoaja",
+ "DatabaseManagementSystem": "Tietokannan ohjausjärjestelmä",
+ "NTier": "N-taso",
+ "IncludeUserInterface": "Sisällytä käyttöliittymä",
+ "CreateNow": "Luo nyt",
+ "TheStartupProject": "Käynnistysprojekti",
+ "Tutorial": "Opetusohjelma",
+ "UsingCLI": "CLI: n käyttö",
+ "SeeDetails": "Katso yksityiskohdat",
+ "AbpShortDescription": "ABP Framework on täydellinen infrastruktuuri nykyaikaisten verkkosovellusten luomiseen noudattamalla ohjelmistokehityksen parhaita käytäntöjä ja käytäntöjä.",
+ "SourceCodeUpper": "LÄHDEKOODI",
+ "LatestReleaseLogs": "Uusimmat julkaisulokit",
+ "Infrastructure": "Infrastruktuuri",
+ "Architecture": "Arkkitehtuuri",
+ "Modular": "Modulaarinen",
+ "DontRepeatYourself": "Älä toista itseäsi",
+ "DeveloperFocused": "Kehittäjä kohdennettu",
+ "FullStackApplicationInfrastructure": "Täyden pinon sovellusinfrastruktuuri.",
+ "DomainDrivenDesign": "Toimialueohjattu suunnittelu",
+ "DomainDrivenDesignExplanation": "Suunniteltu ja kehitetty DDD-mallien ja -periaatteiden perusteella. Tarjoaa kerrostetun mallin sovelluksellesi.",
+ "Authorization": "Valtuutus",
+ "AuthorizationExplanation": "Edistynyt käyttöoikeudet käyttäjän, roolin ja tarkan käyttöjärjestelmän avulla. Rakennettu Microsoft Identity -kirjastoon.",
+ "MultiTenancy": "Monivuokraus",
+ "MultiTenancyExplanationShort": "SaaS-sovellukset on tehty helpoksi! Integroitu monivuokraus tietokannasta käyttöliittymään.",
+ "CrossCuttingConcerns": "Laaja-alaiset huolenaiheet",
+ "CrossCuttingConcernsExplanationShort": "Täydellinen infrastruktuuri valtuutusta, validointia, poikkeusten käsittelyä, välimuistia, auditointilokia, tapahtumien hallintaa ja muuta varten.",
+ "BuiltInBundlingMinification": "Sisäänrakennettu niputtaminen ja pienentäminen",
+ "BuiltInBundlingMinificationExplanation": "Niputtamiseen ja pienentämiseen ei tarvitse käyttää ulkoisia työkaluja. ABP tarjoaa yksinkertaisemman, dynaamisemman, tehokkaamman, modulaarisemman ja sisäänrakennetun tavan!",
+ "VirtualFileSystem": "Virtuaalinen tiedostojärjestelmä",
+ "VirtualFileSystemExplanation": "Upota näkymät, komentosarjat, tyylit, kuvat ... paketteihin / kirjastoihin ja käytä niitä uudelleen eri sovelluksissa.",
+ "Theming": "Heidät",
+ "ThemingExplanationShort": "Käytä ja muokkaa bootstrap-pohjaista vakiokäyttöliittymän teemaa tai luo oma.",
+ "BootstrapTagHelpersDynamicForms": "Bootstrap Tag Helpers ja dynaamiset lomakkeet",
+ "BootstrapTagHelpersDynamicFormsExplanation": "Sen sijaan, että kirjoittaisit manuaalisesti käynnistysstrap-komponenttien toistuvia yksityiskohtia, käytä ABP: n tag-avustajia yksinkertaistaaksesi sitä ja hyödyntääksesi älykkäitä ominaisuuksia. Rakenna käyttöliittymälomakkeet nopeasti C # -mallin perusteella käyttämällä dynaamista lomaketunnisteen auttajaa.",
+ "HTTPAPIsDynamicProxies": "HTTP-sovellusliittymät ja dynaamiset välityspalvelimet",
+ "HTTPAPIsDynamicProxiesExplanation": "Altista sovelluspalvelut automaattisesti REST-tyylisiksi HTTP-sovellusliittymiksi ja kuluta niitä dynaamisilla JavaScript- ja C # -välityspalvelimilla.",
+ "CompleteArchitectureInfo": "Moderni arkkitehtuuri ylläpidettävien ohjelmistoratkaisujen luomiseksi.",
+ "DomainDrivenDesignBasedLayeringModelExplanation": "Auttaa sinua toteuttamaan DDD-pohjaisen kerrostetun arkkitehtuurin ja rakentamaan ylläpidettävän koodipohjan.",
+ "DomainDrivenDesignBasedLayeringModelExplanationCont": "Tarjoaa käynnistysmalleja, abstrakteja, perusluokkia, palveluja, dokumentaatiota ja oppaita, joiden avulla voit kehittää sovellustasi DDD-mallien ja -periaatteiden perusteella.",
+ "MicroserviceCompatibleModelExplanation": "Ydinkehys ja esirakennemoduulit on suunniteltu mikropalveluarkkitehtuuria ajatellen.",
+ "MicroserviceCompatibleModelExplanationCont": "Tarjoaa infrastruktuurin, integraatiot, näytteet ja dokumentaation mikropalveluratkaisujen toteuttamiseksi helpommin, mutta se ei tuo lisää monimutkaisuutta, jos haluat monoliittisen sovelluksen.",
+ "ModularInfo": "ABP tarjoaa moduulijärjestelmän, jonka avulla voit kehittää uudelleenkäytettäviä sovellusmoduuleja, sitoutua sovelluksen elinkaaren tapahtumiin ja ilmaista riippuvuuksia järjestelmän ydinosien välillä.",
+ "PreBuiltModulesThemes": "Valmiit moduulit ja teemat",
+ "PreBuiltModulesThemesExplanation": "Avoimen lähdekoodin ja kaupalliset moduulit ja teemat ovat käyttövalmiita yrityssovelluksessasi.",
+ "NuGetNPMPackages": "NuGet- ja NPM-paketit",
+ "NuGetNPMPackagesExplanation": "Jaettu NuGet- ja NPM-paketteina. Helppo asentaa ja päivittää.",
+ "ExtensibleReplaceable": "Laajennettavissa / vaihdettavissa",
+ "ExtensibleReplaceableExplanation": "Kaikki palvelut ja moduulit on suunniteltu laajennettavuutta ajatellen. Voit korvata palvelut, sivut, tyylit ja komponentit.",
+ "CrossCuttingConcernsExplanation2": "Pidä koodipohjasi pienempi, jotta voit keskittyä yritykseesi liittyvään koodiin.",
+ "CrossCuttingConcernsExplanation3": "Älä lähetä aikaa useiden projektien yhteisten hakemusvaatimusten toteuttamiseen.",
+ "AuthenticationAuthorization": "Todennus ja valtuutus",
+ "ExceptionHandling": "Poikkeusten käsittely",
+ "Validation": "Vahvistus",
+ "DatabaseConnection": "Tietokantayhteys",
+ "TransactionManagement": "Tapahtumien hallinta",
+ "AuditLogging": "Tarkastusten kirjaaminen",
+ "Caching": "Välimuisti",
+ "Multitenancy": "Monivärinen",
+ "DataFiltering": "Tietojen suodatus",
+ "ConventionOverConfiguration": "Kokoonpanon määritys",
+ "ConventionOverConfigurationExplanation": "ABP toteuttaa oletusarvoisesti yleiset sovelluskäytännöt minimaalisella tai nolla-kokoonpanolla.",
+ "ConventionOverConfigurationExplanationList1": "Auto rekisteröi tunnetut palvelut riippuvuusinjektioon.",
+ "ConventionOverConfigurationExplanationList2": "Paljastaa sovelluspalvelut HTTP-sovellusliittyminä nimeämällä käytäntöjä.",
+ "ConventionOverConfigurationExplanationList3": "Luo dynaamiset HTTP-asiakasvälityspalvelimet C #: lle ja JavaScriptille.",
+ "ConventionOverConfigurationExplanationList4": "Tarjoaa oletusvarastoja yhteisöillesi.",
+ "ConventionOverConfigurationExplanationList5": "Hallitsee työyksikköä verkkopyynnön tai sovelluspalvelumenetelmän mukaan.",
+ "ConventionOverConfigurationExplanationList6": "Julkaisee luoda, päivittää ja poistaa tapahtumia yhteisöillesi.",
+ "BaseClasses": "Perusluokat",
+ "BaseClassesExplanation": "Valmiiksi rakennettu perusluokka yleisiä sovelluskuvioita varten.",
+ "DeveloperFocusedExplanation": "ABP on kehittäjille.",
+ "DeveloperFocusedExplanationCont": "Sen tarkoituksena on yksinkertaistaa päivittäistä ohjelmistokehitystäsi samalla, kun se ei estä sinua kirjoittamasta matalan tason koodia.",
+ "SeeAllFeatures": "Katso kaikki ominaisuudet",
+ "CLI_CommandLineInterface": "CLI (komentoriviliitäntä)",
+ "CLI_CommandLineInterfaceExplanation": "Sisältää CLI: n, jonka avulla voit automatisoida uusien projektien luomisen ja uusien moduulien lisäämisen.",
+ "StartupTemplates": "Käynnistysmallit",
+ "StartupTemplatesExplanation": "Erilaiset käynnistysmallit tarjoavat täysin määritetyn ratkaisun kehityksen aloittamiseksi.",
+ "BasedOnFamiliarTools": "Perustuu tuttuihin työkaluihin",
+ "BasedOnFamiliarToolsExplanation": "Rakennettu ja integroitu jo tunnettujen suosittujen työkalujen kanssa. Matala oppimiskäyrä, helppo sopeutuminen, mukava kehitys.",
+ "ORMIndependent": "ORM riippumaton",
+ "ORMIndependentExplanation": "Ydinkehys on ORM / tietokannasta riippumaton ja voi toimia minkä tahansa tietolähteen kanssa. Entity Framework Core- ja MongoDB-palveluntarjoajat ovat jo saatavilla.",
+ "Features": "Tutustu ABP-kehyksen ominaisuuksiin",
+ "ABPCLI": "ABP CLI",
+ "Modularity": "Modulaarisuus",
+ "BootstrapTagHelpers": "Bootstrap Tag Helpers",
+ "DynamicForms": "Dynaamiset lomakkeet",
+ "BundlingMinification": "Niputtaminen ja minimointi",
+ "BackgroundJobs": "Taustatyöt",
+ "BackgroundJobsExplanation": "Määritä yksinkertaiset luokat taustalla olevien töiden suorittamiseksi jonossa. Käytä sisäänrakennettua työnhallintaa tai integroi oma. Hangfire ja RabbitMQ -integraatiot ovat jo käytettävissä.",
+ "DDDInfrastructure": "DDD-infrastruktuuri",
+ "DomainDrivenDesignInfrastructure": "Toimialueohjattu suunnittelun infrastruktuuri",
+ "AutoRESTAPIs": "Auto REST -sovellusliittymät",
+ "DynamicClientProxies": "Dynaamiset asiakaskohtaiset välityspalvelimet",
+ "DistributedEventBus": "Hajautettu tapahtumabussi",
+ "DistributedEventBusWithRabbitMQIntegration": "Hajautettu tapahtumaväylä RabbitMQ-integraatiolla",
+ "TestInfrastructure": "Testaa infrastruktuuri",
+ "AuditLoggingEntityHistories": "Tarkastusloki ja entiteettihistoria",
+ "ObjectToObjectMapping": "Object to Object Mapping",
+ "ObjectToObjectMappingExplanation": " Objektin kartoitus abstraktio AutoMapper-integraatiolla.",
+ "EmailSMSAbstractions": "Sähköposti ja tekstiviestit",
+ "EmailSMSAbstractionsWithTemplatingSupport": "Sähköposti- ja SMS-abstraktit mallintamistuen avulla",
+ "Localization": "Lokalisointi",
+ "SettingManagement": "Asetusten hallinta",
+ "ExtensionMethods": "Laajennusmenetelmät",
+ "ExtensionMethodsHelpers": "Laajennusmenetelmät ja auttajat",
+ "AspectOrientedProgramming": "Aspektiorientoitu ohjelmointi",
+ "DependencyInjection": "Riippuvuuden injektio",
+ "DependencyInjectionByConventions": "Riippuvuuden injektio yleissopimusten mukaan",
+ "ABPCLIExplanation": "ABP CLI (Command Line Interface) on komentorivityökalu joidenkin yleisten toimintojen suorittamiseen ABP-pohjaisiin ratkaisuihin.",
+ "ModularityExplanation": "ABP tarjoaa täydellisen infrastruktuurin omien sovellusmoduulien rakentamiseen. Niillä voi olla entiteettejä, palveluja, tietokantaintegraatioita, sovellusliittymiä, käyttöliittymäkomponentteja ja niin edelleen.",
+ "MultiTenancyExplanation": "ABP-kehys ei vain tue useiden vuokralaisten sovellusten kehittämistä, vaan tekee koodistasi myös enimmäkseen tietämättömän monivuokralaisesta.",
+ "MultiTenancyExplanation2": "Voi määrittää automaattisesti nykyisen vuokralaisen, eristää eri vuokralaisten tiedot toisistaan.",
+ "MultiTenancyExplanation3": "Tukee yhtä tietokantaa, tietokantaa vuokralaista kohti ja hybridi-lähestymistapoja.",
+ "MultiTenancyExplanation4": "Keskity yrityksesi koodiin ja annat kehyksen hoitamaan monivuokrausta puolestasi.",
+ "BootstrapTagHelpersExplanation": "Sen sijaan, että kirjoittaisit manuaalisesti uudelleenkäynnistyskomponenttien yksityiskohtia, käytä ABP: n tunnisteita yksinkertaistaaksesi sitä ja hyödyntäksesi älykkäitä ominaisuuksia. Voit ehdottomasti käyttää Bootstrapia milloin tahansa.",
+ "DynamicFormsExplanation": "Dynaamiset lomake- ja syöttötunnisteiden apurit voivat luoda täydellisen lomakkeen mallina C # -luokasta.",
+ "AuthenticationAuthorizationExplanation": "ASP.NET Core Identity & IdentityServer4 -palveluun integroidut monipuoliset todennus- ja todennusvaihtoehdot. Tarjoaa laajennettavan ja yksityiskohtaisen lupajärjestelmän.",
+ "CrossCuttingConcernsExplanation": "Älä toista itseäsi kaikkien näiden yleisten asioiden toteuttamiseksi uudelleen ja uudelleen. Keskity yrityskoodiin ja anna ABP: n automatisoida ne käytäntöjen mukaan.",
+ "DatabaseConnectionTransactionManagement": "Tietokantayhteys ja tapahtumien hallinta",
+ "CorrelationIdTracking": "Korrelaatio-Id-seuranta",
+ "BundlingMinificationExplanation": "ABP tarjoaa yksinkertaisen, dynaamisen, tehokkaan, modulaarisen ja sisäänrakennetun niputus- ja pienentämisjärjestelmän.",
+ "VirtualFileSystemnExplanation": "Virtuaalinen tiedostojärjestelmä mahdollistaa sellaisten tiedostojen hallinnan, joita ei ole fyysisesti tiedostojärjestelmässä (levyllä). Sitä käytetään pääasiassa upottamaan (js, css, kuva, cshtml ...) tiedostot kokoonpanoiksi ja käyttämään niitä kuin fyysisiä tiedostoja ajon aikana.",
+ "ThemingExplanation": "Theming-järjestelmän avulla voit kehittää sovellus- ja moduuliteemasi itsenäisesti määrittelemällä joukon yhteisiä peruskirjastoja ja asetteluja uusimman Bootstrap-kehyksen perusteella.",
+ "DomainDrivenDesignInfrastructureExplanation": "Täydellinen infrastruktuuri kerrostettujen sovellusten rakentamiseen, joka perustuu toimialueohjattuihin suunnittelumalleihin ja periaatteisiin;",
+ "Specification": "Erittely",
+ "Repository": "Arkisto",
+ "DomainService": "Verkkotunnuspalvelu",
+ "ValueObject": "Arvo-objekti",
+ "ApplicationService": "Sovelluspalvelu",
+ "DataTransferObject": "Tiedonsiirtokohde",
+ "AggregateRootEntity": "Kokonaisjuuri, entiteetti",
+ "AutoRESTAPIsExplanation": "ABP voi määrittää sovelluspalvelut automaattisesti API-ohjaimiksi sopimuksen mukaan.",
+ "DynamicClientProxiesExplanation": "Kuluta helposti API: si JavaScript- ja C # -asiakkailta.",
+ "DistributedEventBusWithRabbitMQIntegrationExplanation": "Julkaise ja kuluta jaettuja tapahtumia helposti käyttämällä sisäänrakennettua hajautettua tapahtumaväylää, jossa on käytettävissä RabbitMQ-integraatio.",
+ "TestInfrastructureExplanation": "Kehys on kehitetty yksikkö- ja integraatiotestaus ajatellen. Tarjoaa sinulle perusluokkia helpottamaan. Käynnistysmalleissa on valmiiksi konfiguroitu testaus.",
+ "AuditLoggingEntityHistoriesExplanation": "Sisäänrakennettu tarkastuskirjaus yrityskriittisille sovelluksille. Pyyntö-, palvelu-, menetelmätason tarkastusloki ja entiteettihistoria omaisuuden tason yksityiskohdilla.",
+ "EmailSMSAbstractionsWithTemplatingSupportExplanation": "IEmailSender- ja ISmsSender-abstraktit erottavat sovelluslogiikkasi infrastruktuurista. Edistyneen sähköpostimallijärjestelmän avulla voit luoda ja lokalisoida sähköpostimalleja ja käyttää niitä helposti tarvittaessa.",
+ "LocalizationExplanation": "Lokalisointijärjestelmä antaa mahdollisuuden luoda resursseja yksinkertaisiin JSON-tiedostoihin ja käyttää niitä käyttöliittymän lokalisointiin. Se tukee edistyneitä skenaarioita, kuten perintö, laajennukset ja JavaScript-integrointi, samalla kun se on täysin yhteensopiva AspNet Core -järjestelmän lokalisointijärjestelmän kanssa.",
+ "SettingManagementExplanation": "Määritä sovelluksesi asetukset ja hanki ajonaikaiset arvot nykyisen kokoonpanon, vuokralaisen ja käyttäjän perusteella.",
+ "ExtensionMethodsHelpersExplanation": "Älä toista itseäsi edes triviaalien koodiosien suhteen. Vakiotyyppien laajennukset ja apuvälineet tekevät koodistasi paljon puhtaamman ja helpommin kirjoitettavan.",
+ "AspectOrientedProgrammingExplanation": "Tarjoaa mukavan infrastruktuurin dynaamisten valtakirjojen luomiseen ja Aspect Oriented Programming -toiminnon toteuttamiseen. Kuuntele mikä tahansa luokka ja suorita koodisi ennen ja jälkeen jokaisen menetelmän suorituksen.",
+ "DependencyInjectionByConventionsExplanation": "Luokat ei tarvitse rekisteröidä riippuvuusinjektioon manuaalisesti. Rekisteröi yleiset palvelutyypit sopimuksen mukaan automaattisesti. Muun tyyppisissä palveluissa voit käyttää käyttöliittymiä ja määritteitä sen helpottamiseksi ja paikoillaan.",
+ "DataFilteringExplanation": "Määritä ja käytä tietosuodattimia, joita käytetään automaattisesti, kun kysyt entiteettejä tietokannasta. Pehmeä poisto- ja MultiTenant-suodattimet toimitetaan heti, kun otat käyttöön yksinkertaiset käyttöliittymät.",
+ "PublishEvents": "Julkaise tapahtumia",
+ "HandleEvents": "Käsittele tapahtumia",
+ "AndMore": "ja enemmän...",
+ "Code": "Koodi",
+ "Result": "Tulos",
+ "SeeTheDocumentForMoreInformation": "Katso lisätietoja {0} asiakirjasta ",
+ "IndexPageHeroSection": " avoin lähdekoodi verkkosovellus Framework asp.net-ytimelle ",
+ "UiFramework": "Käyttöliittymäkehys",
+ "EmailAddress": "Sähköpostiosoite",
+ "Mobile": "Matkapuhelin",
+ "ReactNative": "Reagoi Native",
+ "Strong": "Vahva",
+ "Complete": "Saattaa loppuun",
+ "BasedLayeringModel": "Perustuva kerrosmalli",
+ "Microservice": "Mikropalvelu",
+ "Compatible": "Yhteensopiva",
+ "MeeTTheABPCommunityInfo": "Tavoitteenamme on luoda ympäristö, jossa kehittäjät voivat auttaa toisiaan artikkeleilla, oppailla, tapaustutkimuksilla jne. Ja tavata samanmielisiä ihmisiä.",
+ "JoinTheABPCommunityInfo": "Ole mukana elävässä yhteisössä ja tule osallistujana sivutuotteiden kehykseen!",
+ "AllArticles": "Kaikki artikkelit",
+ "SubmitYourArticle": "Lähetä artikkelisi",
+ "DynamicClientProxyDocument": "Katso dynaamisen asiakkaan välityspalvelimen dokumentaatiot JavaScriptille ja C # .",
+ "EmailSMSAbstractionsDocument": "Katso lisätietoja sähköpostitse ja tekstiviestien lähettäminen -asiakirjoista.",
+ "CreateProjectWizard": "Tämä ohjattu toiminto luo uuden projektin käynnistysmallista, joka on määritetty oikein aloittamaan projekti.",
+ "TieredOption": "Luo porrastetun ratkaisun, jossa Web- ja Http-API-kerrokset erotetaan fyysisesti. Jos sitä ei ole valittu, luodaan kerrostettu ratkaisu, joka on vähemmän monimutkainen ja sopii useimpiin tilanteisiin.",
+ "SeparateIdentityServerOption": "Erottaa palvelinpuolen kahteen sovellukseen: Ensimmäinen on identiteettipalvelimelle ja toinen palvelinpuolen HTTP-sovellusliittymälle.",
+ "UseslatestPreVersion": "Käyttää uusinta julkaisua edeltävää versiota",
+ "ReadTheDocumentation": " Lue Dokumentaatio ",
+ "Documentation": "Dokumentointi",
+ "GettingStartedTutorial": "Aloitusopas",
+ "ApplicationDevelopmentTutorial": "Sovelluskehitysopastus",
+ "TheStartupTemplate": "Käynnistysmalli",
+ "InstallABPCLIInfo": "ABP CLI on nopein tapa aloittaa uusi ratkaisu ABP-kehyksellä. Asenna ABP CLI komentorivillä:",
+ "DifferentLevelOfNamespaces": "Voit käyttää eri nimiavaruustasoja; esimerkiksi. BookStore, Acme.BookStore tai Acme.Retail.BookStore.",
+ "ABPCLIExamplesInfo": " Uusi -komento luo kerrostetun MVC-sovelluksen , jonka tietokannan tarjoaja on Entity Framework Core . Sillä on kuitenkin muita vaihtoehtoja. Esimerkkejä:",
+ "SeeCliDocumentForMoreInformation": "Katso lisää vaihtoehtoja ABP CLI -asiakirjasta tai valitse yllä Suora lataus -välilehti.",
+ "Optional": "Valinnainen",
+ "LocalFrameworkRef": "Säilytä kehyspakettien paikalliset projektiviitteet.",
+ "BlobStoring": "BLOB-tallennus",
+ "BlobStoringExplanation": "BLOB-tallennusjärjestelmä tarjoaa abstraktin BLOBien kanssa työskentelemiseen. ABP tarjoaa joitain valmiita tallennuspalveluntarjoajan integraatioita (Azure, AWS, tiedostojärjestelmä, tietokanta jne.), Joita voit helposti käyttää sovelluksissasi.",
+ "TextTemplating": "Tekstin mallintaminen",
+ "TextTemplatingExplanation": "Tekstimallia käytetään sisällön renderointiin dynaamisesti mallin ja mallin (dataobjektin) perusteella. Voit käyttää sitä esimerkiksi dynaamisen sähköpostisisällön luomiseen valmiilla mallilla.",
+ "MultipleUIOptions": "Useita käyttöliittymävaihtoehtoja",
+ "MultipleDBOptions": "Useita tietokantapalveluja",
+ "MultipleUIOptionsExplanation": "Ydinkehys on suunniteltu käyttöliittymästä riippumattomaksi ja se voi toimia minkä tahansa tyyppisen käyttöliittymäjärjestelmän kanssa, kun taas useita valmiiksi rakennettuja ja integroituja vaihtoehtoja toimitetaan heti.",
+ "MultipleDBOptionsExplanation": "Kehys voi toimia minkä tahansa tietolähteen kanssa, kun taas seuraavat palveluntarjoajat on virallisesti kehitetty ja tuettu;",
+ "SelectLanguage": "Valitse kieli",
+ "LatestArticleOnCommunity": "Viimeisin artikkeli ABP-yhteisöstä",
+ "Register": "Rekisteröidy",
+ "IsDownloadable": "On ladattavissa"
+ }
+}
\ No newline at end of file
diff --git a/common.props b/common.props
index 593fe92213..a62f5b67b2 100644
--- a/common.props
+++ b/common.props
@@ -1,7 +1,7 @@
- latest
- 4.3.0
+ latest
+ 4.4.0$(NoWarn);CS1591;CS0436https://abp.io/assets/abp_nupkg.pnghttps://abp.io/
diff --git a/docs/en/Blog-Posts/2021-03-31 v4_3 Commercial Preview/POST.md b/docs/en/Blog-Posts/2021-03-31 v4_3 Commercial Preview/POST.md
index 50aab3559b..31946691ab 100644
--- a/docs/en/Blog-Posts/2021-03-31 v4_3 Commercial Preview/POST.md
+++ b/docs/en/Blog-Posts/2021-03-31 v4_3 Commercial Preview/POST.md
@@ -1,6 +1,6 @@
# ABP Commercial 4.3 RC Has Been Published
-ABP Commercial version 4.3 RC (Release Candidate) has been published alongside ABP Framework 4.3. RC (TODO: link). I will introduce the new features in this blog post. Here, a list of highlights for this release;
+ABP Commercial version 4.3 RC (Release Candidate) has been published alongside [ABP Framework 4.3. RC](https://blog.abp.io/abp/ABP-Framework-4.3-RC-Has-Been-Published). I will introduce the new features in this blog post. Here, a list of highlights for this release;
* The **microservice starter template** is getting more mature. We've also added a **service template** to add new microservices to the solution.
* New option for the application starter template to have a **separate database schema for tenant databases**.
@@ -15,7 +15,7 @@ Here, some other features already covered in the ABP Framework announcement, but
* **Email setting** management UI
* **Module extensibility** system is now available for the **Blazor UI** too.
-> This post doesn't cover the features and changes done on the ABP Framework side. Please also see the **ABP Framework 4.3. RC blog post** (TODO: link).
+> This post doesn't cover the features and changes done on the ABP Framework side. Please also see the **[ABP Framework 4.3. RC blog post](https://blog.abp.io/abp/ABP-Framework-4.3-RC-Has-Been-Published)**.
## The Migration Guide
@@ -69,7 +69,7 @@ Automatic migration only tries one time. If it fails, it writes the exception lo
### New Module: CMS Kit
-CMS Kit module initial version has been released with this version. As stated in the ABP Framework 4.3 announcement post (TODO: link), it should be considered premature for now.
+CMS Kit module initial version has been released with this version. As stated in the [ABP Framework 4.3 announcement post](https://blog.abp.io/abp/ABP-Framework-4.3-RC-Has-Been-Published), it should be considered premature for now.
For ABP Commercial application startup template, we are providing an option to include the CMS Kit into the solution while creating new solutions:
diff --git a/docs/en/Blog-Posts/2021-03-31 v4_3 Preview/POST.md b/docs/en/Blog-Posts/2021-03-31 v4_3 Preview/POST.md
index cfb9bf0eed..e1e04e7a90 100644
--- a/docs/en/Blog-Posts/2021-03-31 v4_3 Preview/POST.md
+++ b/docs/en/Blog-Posts/2021-03-31 v4_3 Preview/POST.md
@@ -10,11 +10,11 @@ We are super excited to announce the ABP Framework 4.3 RC (Release Candidate). H
* CLI support to easily add the **Basic Theme** into the solution.
* New **IInitLogger** service to write logs before dependency injection phase completed.
-Besides the new features above, we've done many performance improvements, enhancements and bug fixes on the current features. See the [4.3 milestone](https://github.com/abpframework/abp/milestone/49) on GitHub for all changes made on this version.
+Besides the new features above, we've done many performance improvements, enhancements and bug fixes on the current features. See the [4.3 milestone](https://github.com/abpframework/abp/milestone/49?closed=1) on GitHub for all changes made on this version.
-This version was a big development journey for us; [~160 issues](https://github.com/abpframework/abp/issues?q=is%3Aopen+is%3Aissue+milestone%3A4.3-preview) resolved, [~300 PRs](https://github.com/abpframework/abp/pulls?q=is%3Aopen+is%3Apr+milestone%3A4.3-preview) merged and **~1,700 commits** done only in the [main framework repository](https://github.com/abpframework/abp). **Thanks to the ABP Framework team and all the contributors.**
+This version was a big development journey for us; [~160 issues](https://github.com/abpframework/abp/issues?q=is%3Aissue+milestone%3A4.3-preview+is%3Aclosed) resolved, [~300 PRs](https://github.com/abpframework/abp/issues?q=is%3Apr+milestone%3A4.3-preview+is%3Aclosed) merged and **~1,700 commits** done only in the [main framework repository](https://github.com/abpframework/abp). **Thanks to the ABP Framework team and all the contributors.**
-> ABP Commercial 4.3 RC has also been published. We will write a separate blog post for it.
+> ABP Commercial 4.3 RC has also been published. Check out [the commercial blog post](https://blog.abp.io/abp/ABP-Commercial-4.3-RC-Has-Been-Published).
## The Migration Guide
@@ -63,7 +63,7 @@ CMS (Content Management System) Kit was a module we worked on for the last coupl
* **Reactions**: Allows users to react to content via emojis, like a smile, upvote, downvote, etc.
* **Rating**: This component is used to rate content by users.
-All features are separately usable. For example, you can create an image gallery and reuse the Comments and Tags features for the images. You can enable/disable features individually using the [Global Features System](https://docs.abp.io/en/abp/4.3/global-features).
+All features are separately usable. For example, you can create an image gallery and reuse the Comments and Tags features for the images. You can enable/disable features individually using the [Global Features System](https://docs.abp.io/en/abp/4.3/Global-Features).
> We will create a separate blog post for the CMS Kit module, so I keep it short.
diff --git a/docs/en/CLI-New-Command-Samples.md b/docs/en/CLI-New-Command-Samples.md
index eedf03353c..52ca533c81 100644
--- a/docs/en/CLI-New-Command-Samples.md
+++ b/docs/en/CLI-New-Command-Samples.md
@@ -126,6 +126,12 @@ Module are reusable sub applications used by your main project. Using ABP Module
```bash
abp new Acme.IssueManagement -t module --no-ui
```
+
+* Creates the module and adds it to your solution
+
+ ```bash
+ abp new Acme.IssueManagement -t module --add-to-solution-file
+ ```
## Create a solution from a specific version
diff --git a/docs/en/Community-Articles/2020-12-04-Event-Organizer/Post.md b/docs/en/Community-Articles/2020-12-04-Event-Organizer/Post.md
index b36d83d0ba..076ceb3aa0 100644
--- a/docs/en/Community-Articles/2020-12-04-Event-Organizer/Post.md
+++ b/docs/en/Community-Articles/2020-12-04-Event-Organizer/Post.md
@@ -447,11 +447,12 @@ namespace EventOrganizer.Events
public async Task> GetUpcomingAsync()
{
- var events = await AsyncExecuter.ToListAsync(
- _eventRepository
- .Where(x => x.StartTime > Clock.Now)
- .OrderBy(x => x.StartTime)
- );
+ var queryable = await _eventRepository.GetQueryableAsync();
+ var query = queryable
+ .Where(x => x.StartTime > Clock.Now)
+ .OrderBy(x => x.StartTime);
+
+ var events = await AsyncExecuter.ToListAsync(query);
return ObjectMapper.Map, List>(events);
}
@@ -653,11 +654,12 @@ namespace EventOrganizer.Events
public async Task> GetUpcomingAsync()
{
- var events = await AsyncExecuter.ToListAsync(
- _eventRepository
- .Where(x => x.StartTime > Clock.Now)
- .OrderBy(x => x.StartTime)
- );
+ var queryable = await _eventRepository.GetQueryableAsync();
+ var query = queryable
+ .Where(x => x.StartTime > Clock.Now)
+ .OrderBy(x => x.StartTime);
+
+ var events = await AsyncExecuter.ToListAsync(query);
return ObjectMapper.Map, List>(events);
}
@@ -666,7 +668,12 @@ namespace EventOrganizer.Events
{
var @event = await _eventRepository.GetAsync(id);
var attendeeIds = @event.Attendees.Select(a => a.UserId).ToList();
- var attendees = (await AsyncExecuter.ToListAsync(_userRepository.Where(u => attendeeIds.Contains(u.Id))))
+
+ var queryable = await _userRepository.GetQueryableAsync();
+ var query = queryable
+ .Where(u => attendeeIds.Contains(u.Id));
+
+ var attendees = (await AsyncExecuter.ToListAsync(query))
.ToDictionary(x => x.Id);
var result = ObjectMapper.Map(@event);
diff --git a/docs/en/Text-Templating-Razor.md b/docs/en/Text-Templating-Razor.md
new file mode 100644
index 0000000000..700d262b9b
--- /dev/null
+++ b/docs/en/Text-Templating-Razor.md
@@ -0,0 +1,569 @@
+# Razor Integration
+
+
+The Razor template is a standard C# class, so you can freely use the functions of C#, such as `dependency injection`, using `LINQ`, custom methods, and even using `Repository`.
+
+
+## Installation
+
+It is suggested to use the [ABP CLI](CLI.md) to install this package.
+
+### Using the ABP CLI
+
+Open a command line window in the folder of the project (.csproj file) and type the following command:
+
+````bash
+abp add-package Volo.Abp.TextTemplating.Razor
+````
+
+### Manual Installation
+
+If you want to manually install;
+
+1. Add the [Volo.Abp.TextTemplating.Razor](https://www.nuget.org/packages/Volo.Abp.TextTemplating.Razor) NuGet package to your project:
+
+````
+Install-Package Volo.Abp.TextTemplating.Razor
+````
+
+2. Add the `AbpTextTemplatingRazorModule` to the dependency list of your module:
+
+````csharp
+[DependsOn(
+ //...other dependencies
+ typeof(AbpTextTemplatingRazorModule) //Add the new module dependency
+ )]
+public class YourModule : AbpModule
+{
+}
+````
+
+## Add MetadataReference to CSharpCompilerOptions
+
+You need to add the `MetadataReference` of the type used in the template to `CSharpCompilerOptions's References`.
+
+````csharp
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ Configure(options =>
+ {
+ options.References.Add(MetadataReference.CreateFromFile(typeof(YourModule).Assembly.Location));
+ });
+}
+````
+
+## Add MetadataReference for a template.
+
+You can add some `MetadataReference` to the template
+
+````csharp
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ services.Configure(options =>
+ {
+ //Hello is template name.
+ options.TemplateReferences.Add("Hello", new List()
+ {
+ Assembly.Load("Microsoft.Extensions.Logging.Abstractions"),
+ Assembly.Load("Microsoft.Extensions.Logging")
+ }
+ .Select(x => MetadataReference.CreateFromFile(x.Location))
+ .ToList());
+ });
+}
+````
+
+## Defining Templates
+
+Before rendering a template, you should define it. Create a class inheriting from the `TemplateDefinitionProvider` base class:
+
+````csharp
+public class DemoTemplateDefinitionProvider : TemplateDefinitionProvider
+{
+ public override void Define(ITemplateDefinitionContext context)
+ {
+ context.Add(
+ new TemplateDefinition("Hello") //template name: "Hello"
+ .WithVirtualFilePath(
+ "/Demos/Hello/Hello.cshtml", //template content path
+ isInlineLocalized: true
+ )
+ );
+ }
+}
+````
+
+* `context` object is used to add new templates or get the templates defined by depended modules. Used `context.Add(...)` to define a new template.
+* `TemplateDefinition` is the class represents a template. Each template must have a unique name (that will be used while you are rendering the template).
+* `/Demos/Hello/Hello.cshtml` is the path of the template file.
+* `isInlineLocalized` is used to declare if you are using a single template for all languages (`true`) or different templates for each language (`false`). See the Localization section below for more.
+
+### The Template Base
+
+Every `cshtml` template page needs to inherit `RazorTemplatePageBase` or `RazorTemplatePageBase`.
+There are some useful properties in the base class that can be used in templates. eg: `Localizer`, `ServiceProvider`.
+
+### The Template Content
+
+`WithVirtualFilePath` indicates that we are using the [Virtual File System](Virtual-File-System.md) to store the template content. Create a `Hello.cshtml` file inside your project and mark it as "**embedded resource**" on the properties window:
+
+
+
+Example `Hello.cshtml` content is shown below:
+
+````
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+Hello @Model.Name
+````
+
+The `HelloModel` class is:
+````csharp
+namespace HelloModelNamespace
+{
+ public class HelloModel
+ {
+ public string Name { get; set; }
+ }
+}
+````
+
+The [Virtual File System](Virtual-File-System.md) requires to add your files in the `ConfigureServices` method of your [module](Module-Development-Basics.md) class:
+
+````csharp
+Configure(options =>
+{
+ options.FileSets.AddEmbedded("TextTemplateDemo");
+});
+````
+
+* `TextTemplateDemoModule` is the module class that you define your template in.
+* `TextTemplateDemo` is the root namespace of your project.
+
+## Rendering the Template
+
+`ITemplateRenderer` service is used to render a template content.
+
+### Example: Rendering a Simple Template
+
+````csharp
+public class HelloDemo : ITransientDependency
+{
+ private readonly ITemplateRenderer _templateRenderer;
+
+ public HelloDemo(ITemplateRenderer templateRenderer)
+ {
+ _templateRenderer = templateRenderer;
+ }
+
+ public async Task RunAsync()
+ {
+ var result = await _templateRenderer.RenderAsync(
+ "Hello", //the template name
+ new HelloModel
+ {
+ Name = "John"
+ }
+ );
+
+ Console.WriteLine(result);
+ }
+}
+````
+
+* `HelloDemo` is a simple class that injects the `ITemplateRenderer` in its constructor and uses it inside the `RunAsync` method.
+* `RenderAsync` gets two fundamental parameters:
+ * `templateName`: The name of the template to be rendered (`Hello` in this example).
+ * `model`: An object that is used as the `model` inside the template (a `HelloModel` object in this example).
+
+The result shown below for this example:
+
+````csharp
+Hello John :)
+````
+## Localization
+
+It is possible to localize a template content based on the current culture. There are two types of localization options described in the following sections.
+
+### Inline localization
+
+Inline localization uses the [localization system](Localization.md) to localize texts inside templates.
+
+#### Example: Reset Password Link
+
+Assuming you need to send an email to a user to reset her/his password. Here, the model/template content:
+
+````csharp
+namespace ResetMyPasswordModelNamespace
+{
+ public class ResetMyPasswordModel
+ {
+ public string Link { get; set; }
+
+ public string Name { get; set; }
+ }
+}
+````
+
+````html
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+@Localizer["ResetMyPassword", Model.Name]
+````
+
+`Localizer` service is used to localize the given key based on the current user culture. You need to define the `ResetMyPassword` key inside your localization file:
+
+````json
+"ResetMyPasswordTitle": "Reset my password",
+"ResetMyPassword": "Hi {0}, Click here to reset your password"
+````
+
+You also need to declare the localization resource to be used with this template, inside your template definition provider class:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ "PasswordReset", //Template name
+ typeof(DemoResource) //LOCALIZATION RESOURCE
+ ).WithVirtualFilePath(
+ "/Demos/PasswordReset/PasswordReset.cshtml", //template content path
+ isInlineLocalized: true
+ )
+);
+````
+
+That's all. When you render this template like that:
+
+````csharp
+var result = await _templateRenderer.RenderAsync(
+ "PasswordReset", //the template name
+ new PasswordResetModel
+ {
+ Name = "john",
+ Link = "https://abp.io/example-link?userId=123&token=ABC"
+ }
+);
+````
+
+You will see the localized result:
+
+````html
+Hi john, Click here to reset your password
+````
+
+> If you define the [default localization resource](Localization.md) for your application, then no need to declare the resource type for the template definition.
+
+### Multiple Contents Localization
+
+Instead of a single template that uses the localization system to localize the template, you may want to create different template files for each language. It can be needed if the template should be completely different for a specific culture rather than simple text localizations.
+
+#### Example: Welcome Email Template
+
+Assuming that you want to send a welcome email to your users, but want to define a completely different template based on the user culture.
+
+First, create a folder and put your templates inside it, like `en.cshtml`, `tr.cshtml`... one for each culture you support:
+
+
+
+Then add your template definition in the template definition provider class:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ name: "WelcomeEmail",
+ defaultCultureName: "en"
+ )
+ .WithVirtualFilePath(
+ "/Demos/WelcomeEmail/Templates", //template content folder
+ isInlineLocalized: false
+ )
+);
+````
+
+* Set **default culture name**, so it fallbacks to the default culture if there is no template for the desired culture.
+* Specify **the template folder** rather than a single template file.
+* Set `isInlineLocalized` to `false` for this case.
+
+That's all, you can render the template for the current culture:
+
+````csharp
+var result = await _templateRenderer.RenderAsync("WelcomeEmail");
+````
+
+> Skipped the modal for this example to keep it simple, but you can use models as just explained before.
+
+### Specify the Culture
+
+`ITemplateRenderer` service uses the current culture (`CultureInfo.CurrentUICulture`) if not specified. If you need, you can specify the culture as the `cultureName` parameter:
+
+````csharp
+var result = await _templateRenderer.RenderAsync(
+ "WelcomeEmail",
+ cultureName: "en"
+);
+````
+
+## Layout Templates
+
+Layout templates are used to create shared layouts among other templates. It is similar to the layout system in the ASP.NET Core MVC / Razor Pages.
+
+### Example: Email HTML Layout Template
+
+For example, you may want to create a single layout for all of your email templates.
+
+First, create a template file just like before:
+
+````html
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+
+
+
+
+
+
+ @Body
+
+
+````
+
+* A layout template must have a `Body` part as a place holder for the rendered child content.
+
+The register your template in the template definition provider:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ "EmailLayout",
+ isLayout: true //SET isLayout!
+ ).WithVirtualFilePath(
+ "/Demos/EmailLayout/EmailLayout.cshtml",
+ isInlineLocalized: true
+ )
+);
+````
+
+Now, you can use this template as the layout of any other template:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ name: "WelcomeEmail",
+ defaultCultureName: "en",
+ layout: "EmailLayout" //Set the LAYOUT
+ ).WithVirtualFilePath(
+ "/Demos/WelcomeEmail/Templates",
+ isInlineLocalized: false
+ )
+);
+````
+
+## Global Context
+
+ABP passes the `model` that can be used to access to the model inside the template. You can pass more global variables if you need.
+
+An example template content:
+
+````html
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+A global object value: @GlobalContext["myGlobalObject"]
+````
+
+This template assumes that that is a `myGlobalObject` object in the template rendering context. You can provide it like shown below:
+
+````csharp
+var result = await _templateRenderer.RenderAsync(
+ "GlobalContextUsage",
+ globalContext: new Dictionary
+ {
+ {"myGlobalObject", "TEST VALUE"}
+ }
+);
+````
+
+The rendering result will be:
+
+````
+A global object value: TEST VALUE
+````
+
+## Replacing the Existing Templates
+
+It is possible to replace a template defined by a module that used in your application. In this way, you can customize the templates based on your requirements without changing the module code.
+
+### Option-1: Using the Virtual File System
+
+The [Virtual File System](Virtual-File-System.md) allows you to override any file by placing the same file into the same path in your project.
+
+#### Example: Replace the Standard Email Layout Template
+
+ABP Framework provides an [email sending system](Emailing.md) that internally uses the text templating to render the email content. It defines a standard email layout template in the `/Volo/Abp/Emailing/Templates/Layout.cshtml` path. The unique name of the template is `Abp.StandardEmailTemplates.Layout` and this string is defined as a constant on the `Volo.Abp.Emailing.Templates.StandardEmailTemplates` static class.
+
+Do the following steps to replace the template file with your own;
+
+**1)** Add a new file into the same location (`/Volo/Abp/Emailing/Templates/Layout.cshtml`) in your project:
+
+
+
+**2)** Prepare your email layout template:
+
+````html
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+
+
+
+
+
+
+
This my header
+
+ @Body
+
+
+
+
+````
+
+This example simply adds a header and footer to the template and renders the content between them (see the *Layout Templates* section above to understand it).
+
+**3)** Configure the embedded resources in the `.csproj` file
+
+* Add [Microsoft.Extensions.FileProviders.Embedded](https://www.nuget.org/packages/Microsoft.Extensions.FileProviders.Embedded) NuGet package to the project.
+* Add `true` into the `...` section of your `.csproj` file.
+* Add the following code into your `.csproj` file:
+
+````xml
+
+
+
+
+````
+
+This makes the template files "embedded resource".
+
+**4)** Configure the virtual file system
+
+Configure the `AbpVirtualFileSystemOptions` in the `ConfigureServices` method of your [module](Module-Development-Basics.md) to add the embedded files into the virtual file system:
+
+```csharp
+Configure(options =>
+{
+ options.FileSets.AddEmbedded();
+});
+```
+
+`BookStoreDomainModule` should be your module name, in this example code.
+
+> Be sure that your module (directly or indirectly) [depends on](Module-Development-Basics.md) the `AbpEmailingModule`. Because the VFS can override files based on the dependency order.
+
+Now, your template will be used when you want to render the email layout template.
+
+### Option-2: Using the Template Definition Provider
+
+You can create a template definition provider class that gets the email layout template and changes the virtual file path for the template.
+
+**Example: Use the `/MyTemplates/EmailLayout.cshtml` file instead of the standard template**
+
+```csharp
+using Volo.Abp.DependencyInjection;
+using Volo.Abp.Emailing.Templates;
+using Volo.Abp.TextTemplating;
+
+namespace MyProject
+{
+ public class MyTemplateDefinitionProvider
+ : TemplateDefinitionProvider, ITransientDependency
+ {
+ public override void Define(ITemplateDefinitionContext context)
+ {
+ var emailLayoutTemplate = context.GetOrNull(StandardEmailTemplates.Layout);
+
+ emailLayoutTemplate
+ .WithVirtualFilePath(
+ "/MyTemplates/EmailLayout.cshtml",
+ isInlineLocalized: true
+ );
+ }
+ }
+}
+```
+
+You should still add the file `/MyTemplates/EmailLayout.cshtml` to the virtual file system as explained before. This approach allows you to locate templates in any folder instead of the folder defined by the depended module.
+
+Beside the template content, you can manipulate the template definition properties, like `DisplayName`, `Layout` or `LocalizationSource`.
+
+## Advanced Features
+
+This section covers some internals and more advanced usages of the text templating system.
+
+### Template Content Provider
+
+`ITemplateRenderer` is used to render the template, which is what you want for most of the cases. However, you can use the `ITemplateContentProvider` to get the raw (not rendered) template contents.
+
+> `ITemplateContentProvider` is internally used by the `ITemplateRenderer` to get the raw template contents.
+
+Example:
+
+````csharp
+public class TemplateContentDemo : ITransientDependency
+{
+ private readonly ITemplateContentProvider _templateContentProvider;
+
+ public TemplateContentDemo(ITemplateContentProvider templateContentProvider)
+ {
+ _templateContentProvider = templateContentProvider;
+ }
+
+ public async Task RunAsync()
+ {
+ var result = await _templateContentProvider
+ .GetContentOrNullAsync("Hello");
+
+ Console.WriteLine(result);
+ }
+}
+````
+
+The result will be the raw template content:
+
+````
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+Hello @Model.Name
+````
+
+* `GetContentOrNullAsync` returns `null` if no content defined for the requested template.
+* It can get a `cultureName` parameter that is used if template has different files for different cultures (see Multiple Contents Localization section above).
+
+### Template Content Contributor
+
+`ITemplateContentProvider` service uses `ITemplateContentContributor` implementations to find template contents. There is a single pre-implemented content contributor, `VirtualFileTemplateContentContributor`, which gets template contents from the virtual file system as described above.
+
+You can implement the `ITemplateContentContributor` to read raw template contents from another source.
+
+Example:
+
+````csharp
+public class MyTemplateContentProvider
+ : ITemplateContentContributor, ITransientDependency
+{
+ public async Task GetOrNullAsync(TemplateContentContributorContext context)
+ {
+ var templateName = context.TemplateDefinition.Name;
+
+ //TODO: Try to find content from another source
+ return null;
+ }
+}
+
+````
+
+Return `null` if your source can not find the content, so `ITemplateContentProvider` fallbacks to the next contributor.
+
+### Template Definition Manager
+
+`ITemplateDefinitionManager` service can be used to get the template definitions (created by the template definition providers).
+
+## See Also
+
+* [The source code of the sample application](https://github.com/abpframework/abp-samples/tree/master/TextTemplateDemo) developed and referred through this document.
+* [Localization system](Localization.md).
+* [Virtual File System](Virtual-File-System.md).
diff --git a/docs/en/Text-Templating-Scriban.md b/docs/en/Text-Templating-Scriban.md
new file mode 100644
index 0000000000..4d9be31dc8
--- /dev/null
+++ b/docs/en/Text-Templating-Scriban.md
@@ -0,0 +1,517 @@
+# Razor Integration
+
+## Installation
+
+It is suggested to use the [ABP CLI](CLI.md) to install this package.
+
+### Using the ABP CLI
+
+Open a command line window in the folder of the project (.csproj file) and type the following command:
+
+````bash
+abp add-package Volo.Abp.TextTemplating.Scriban
+````
+
+### Manual Installation
+
+If you want to manually install;
+
+1. Add the [Volo.Abp.TextTemplating.Scriban](https://www.nuget.org/packages/Volo.Abp.TextTemplating.Scriban) NuGet package to your project:
+
+````
+Install-Package Volo.Abp.TextTemplating.Scriban
+````
+
+2. Add the `AbpTextTemplatingScribanModule` to the dependency list of your module:
+
+````csharp
+[DependsOn(
+ //...other dependencies
+ typeof(AbpTextTemplatingScribanModule) //Add the new module dependency
+ )]
+public class YourModule : AbpModule
+{
+}
+````
+
+## Defining Templates
+
+Before rendering a template, you should define it. Create a class inheriting from the `TemplateDefinitionProvider` base class:
+
+````csharp
+public class DemoTemplateDefinitionProvider : TemplateDefinitionProvider
+{
+ public override void Define(ITemplateDefinitionContext context)
+ {
+ context.Add(
+ new TemplateDefinition("Hello") //template name: "Hello"
+ .WithVirtualFilePath(
+ "/Demos/Hello/Hello.tpl", //template content path
+ isInlineLocalized: true
+ )
+ );
+ }
+}
+````
+
+* `context` object is used to add new templates or get the templates defined by depended modules. Used `context.Add(...)` to define a new template.
+* `TemplateDefinition` is the class represents a template. Each template must have a unique name (that will be used while you are rendering the template).
+* `/Demos/Hello/Hello.tpl` is the path of the template file.
+* `isInlineLocalized` is used to declare if you are using a single template for all languages (`true`) or different templates for each language (`false`). See the Localization section below for more.
+
+### The Template Content
+
+`WithVirtualFilePath` indicates that we are using the [Virtual File System](Virtual-File-System.md) to store the template content. Create a `Hello.tpl` file inside your project and mark it as "**embedded resource**" on the properties window:
+
+
+
+Example `Hello.tpl` content is shown below:
+
+````
+Hello {%{{{model.name}}}%} :)
+````
+
+The [Virtual File System](Virtual-File-System.md) requires to add your files in the `ConfigureServices` method of your [module](Module-Development-Basics.md) class:
+
+````csharp
+Configure(options =>
+{
+ options.FileSets.AddEmbedded("TextTemplateDemo");
+});
+````
+
+* `TextTemplateDemoModule` is the module class that you define your template in.
+* `TextTemplateDemo` is the root namespace of your project.
+
+## Rendering the Template
+
+`ITemplateRenderer` service is used to render a template content.
+
+### Example: Rendering a Simple Template
+
+````csharp
+public class HelloDemo : ITransientDependency
+{
+ private readonly ITemplateRenderer _templateRenderer;
+
+ public HelloDemo(ITemplateRenderer templateRenderer)
+ {
+ _templateRenderer = templateRenderer;
+ }
+
+ public async Task RunAsync()
+ {
+ var result = await _templateRenderer.RenderAsync(
+ "Hello", //the template name
+ new HelloModel
+ {
+ Name = "John"
+ }
+ );
+
+ Console.WriteLine(result);
+ }
+}
+````
+
+* `HelloDemo` is a simple class that injects the `ITemplateRenderer` in its constructor and uses it inside the `RunAsync` method.
+* `RenderAsync` gets two fundamental parameters:
+ * `templateName`: The name of the template to be rendered (`Hello` in this example).
+ * `model`: An object that is used as the `model` inside the template (a `HelloModel` object in this example).
+
+The result shown below for this example:
+
+````csharp
+Hello John :)
+````
+
+### Anonymous Model
+
+While it is suggested to create model classes for the templates, it would be practical (and possible) to use anonymous objects for simple cases:
+
+````csharp
+var result = await _templateRenderer.RenderAsync(
+ "Hello",
+ new
+ {
+ Name = "John"
+ }
+);
+````
+
+In this case, we haven't created a model class, but created an anonymous object as the model.
+
+### PascalCase vs snake_case
+
+PascalCase property names (like `UserName`) is used as snake_case (like `user_name`) in the templates.
+
+## Localization
+
+It is possible to localize a template content based on the current culture. There are two types of localization options described in the following sections.
+
+### Inline localization
+
+Inline localization uses the [localization system](Localization.md) to localize texts inside templates.
+
+#### Example: Reset Password Link
+
+Assuming you need to send an email to a user to reset her/his password. Here, the template content:
+
+````
+{%{{{L "ResetMyPassword" model.name}}}%}
+````
+
+`L` function is used to localize the given key based on the current user culture. You need to define the `ResetMyPassword` key inside your localization file:
+
+````json
+"ResetMyPasswordTitle": "Reset my password",
+"ResetMyPassword": "Hi {0}, Click here to reset your password"
+````
+
+You also need to declare the localization resource to be used with this template, inside your template definition provider class:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ "PasswordReset", //Template name
+ typeof(DemoResource) //LOCALIZATION RESOURCE
+ ).WithVirtualFilePath(
+ "/Demos/PasswordReset/PasswordReset.tpl", //template content path
+ isInlineLocalized: true
+ )
+);
+````
+
+That's all. When you render this template like that:
+
+````csharp
+var result = await _templateRenderer.RenderAsync(
+ "PasswordReset", //the template name
+ new PasswordResetModel
+ {
+ Name = "john",
+ Link = "https://abp.io/example-link?userId=123&token=ABC"
+ }
+);
+````
+
+You will see the localized result:
+
+````csharp
+Hi john, Click here to reset your password
+````
+
+> If you define the [default localization resource](Localization.md) for your application, then no need to declare the resource type for the template definition.
+
+### Multiple Contents Localization
+
+Instead of a single template that uses the localization system to localize the template, you may want to create different template files for each language. It can be needed if the template should be completely different for a specific culture rather than simple text localizations.
+
+#### Example: Welcome Email Template
+
+Assuming that you want to send a welcome email to your users, but want to define a completely different template based on the user culture.
+
+First, create a folder and put your templates inside it, like `en.tpl`, `tr.tpl`... one for each culture you support:
+
+
+
+Then add your template definition in the template definition provider class:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ name: "WelcomeEmail",
+ defaultCultureName: "en"
+ )
+ .WithVirtualFilePath(
+ "/Demos/WelcomeEmail/Templates", //template content folder
+ isInlineLocalized: false
+ )
+);
+````
+
+* Set **default culture name**, so it fallbacks to the default culture if there is no template for the desired culture.
+* Specify **the template folder** rather than a single template file.
+* Set `isInlineLocalized` to `false` for this case.
+
+That's all, you can render the template for the current culture:
+
+````csharp
+var result = await _templateRenderer.RenderAsync("WelcomeEmail");
+````
+
+> Skipped the modal for this example to keep it simple, but you can use models as just explained before.
+
+### Specify the Culture
+
+`ITemplateRenderer` service uses the current culture (`CultureInfo.CurrentUICulture`) if not specified. If you need, you can specify the culture as the `cultureName` parameter:
+
+````csharp
+var result = await _templateRenderer.RenderAsync(
+ "WelcomeEmail",
+ cultureName: "en"
+);
+````
+
+## Layout Templates
+
+Layout templates are used to create shared layouts among other templates. It is similar to the layout system in the ASP.NET Core MVC / Razor Pages.
+
+### Example: Email HTML Layout Template
+
+For example, you may want to create a single layout for all of your email templates.
+
+First, create a template file just like before:
+
+````xml
+
+
+
+
+
+
+ {%{{{content}}}%}
+
+
+````
+
+* A layout template must have a **{%{{{content}}}%}** part as a place holder for the rendered child content.
+
+The register your template in the template definition provider:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ "EmailLayout",
+ isLayout: true //SET isLayout!
+ ).WithVirtualFilePath(
+ "/Demos/EmailLayout/EmailLayout.tpl",
+ isInlineLocalized: true
+ )
+);
+````
+
+Now, you can use this template as the layout of any other template:
+
+````csharp
+context.Add(
+ new TemplateDefinition(
+ name: "WelcomeEmail",
+ defaultCultureName: "en",
+ layout: "EmailLayout" //Set the LAYOUT
+ ).WithVirtualFilePath(
+ "/Demos/WelcomeEmail/Templates",
+ isInlineLocalized: false
+ )
+);
+````
+
+## Global Context
+
+ABP passes the `model` that can be used to access to the model inside the template. You can pass more global variables if you need.
+
+An example template content:
+
+````
+A global object value: {%{{{myGlobalObject}}}%}
+````
+
+This template assumes that that is a `myGlobalObject` object in the template rendering context. You can provide it like shown below:
+
+````csharp
+var result = await _templateRenderer.RenderAsync(
+ "GlobalContextUsage",
+ globalContext: new Dictionary
+ {
+ {"myGlobalObject", "TEST VALUE"}
+ }
+);
+````
+
+The rendering result will be:
+
+````
+A global object value: TEST VALUE
+````
+
+## Replacing the Existing Templates
+
+It is possible to replace a template defined by a module that used in your application. In this way, you can customize the templates based on your requirements without changing the module code.
+
+### Option-1: Using the Virtual File System
+
+The [Virtual File System](Virtual-File-System.md) allows you to override any file by placing the same file into the same path in your project.
+
+#### Example: Replace the Standard Email Layout Template
+
+ABP Framework provides an [email sending system](Emailing.md) that internally uses the text templating to render the email content. It defines a standard email layout template in the `/Volo/Abp/Emailing/Templates/Layout.tpl` path. The unique name of the template is `Abp.StandardEmailTemplates.Layout` and this string is defined as a constant on the `Volo.Abp.Emailing.Templates.StandardEmailTemplates` static class.
+
+Do the following steps to replace the template file with your own;
+
+**1)** Add a new file into the same location (`/Volo/Abp/Emailing/Templates/Layout.tpl`) in your project:
+
+
+
+**2)** Prepare your email layout template:
+
+````html
+
+
+
+
+
+
+
This my header
+
+ {%{{{content}}}%}
+
+
+
+
+````
+
+This example simply adds a header and footer to the template and renders the content between them (see the *Layout Templates* section above to understand it).
+
+**3)** Configure the embedded resources in the `.csproj` file
+
+* Add [Microsoft.Extensions.FileProviders.Embedded](https://www.nuget.org/packages/Microsoft.Extensions.FileProviders.Embedded) NuGet package to the project.
+* Add `true` into the `...` section of your `.csproj` file.
+* Add the following code into your `.csproj` file:
+
+````xml
+
+
+
+
+````
+
+This makes the template files "embedded resource".
+
+**4)** Configure the virtual file system
+
+Configure the `AbpVirtualFileSystemOptions` in the `ConfigureServices` method of your [module](Module-Development-Basics.md) to add the embedded files into the virtual file system:
+
+```csharp
+Configure(options =>
+{
+ options.FileSets.AddEmbedded();
+});
+```
+
+`BookStoreDomainModule` should be your module name, in this example code.
+
+> Be sure that your module (directly or indirectly) [depends on](Module-Development-Basics.md) the `AbpEmailingModule`. Because the VFS can override files based on the dependency order.
+
+Now, your template will be used when you want to render the email layout template.
+
+### Option-2: Using the Template Definition Provider
+
+You can create a template definition provider class that gets the email layout template and changes the virtual file path for the template.
+
+**Example: Use the `/MyTemplates/EmailLayout.tpl` file instead of the standard template**
+
+```csharp
+using Volo.Abp.DependencyInjection;
+using Volo.Abp.Emailing.Templates;
+using Volo.Abp.TextTemplating;
+
+namespace MyProject
+{
+ public class MyTemplateDefinitionProvider
+ : TemplateDefinitionProvider, ITransientDependency
+ {
+ public override void Define(ITemplateDefinitionContext context)
+ {
+ var emailLayoutTemplate = context.GetOrNull(StandardEmailTemplates.Layout);
+
+ emailLayoutTemplate
+ .WithVirtualFilePath(
+ "/MyTemplates/EmailLayout.tpl",
+ isInlineLocalized: true
+ );
+ }
+ }
+}
+```
+
+You should still add the file `/MyTemplates/EmailLayout.tpl` to the virtual file system as explained before. This approach allows you to locate templates in any folder instead of the folder defined by the depended module.
+
+Beside the template content, you can manipulate the template definition properties, like `DisplayName`, `Layout` or `LocalizationSource`.
+
+## Advanced Features
+
+This section covers some internals and more advanced usages of the text templating system.
+
+### Template Content Provider
+
+`ITemplateRenderer` is used to render the template, which is what you want for most of the cases. However, you can use the `ITemplateContentProvider` to get the raw (not rendered) template contents.
+
+> `ITemplateContentProvider` is internally used by the `ITemplateRenderer` to get the raw template contents.
+
+Example:
+
+````csharp
+public class TemplateContentDemo : ITransientDependency
+{
+ private readonly ITemplateContentProvider _templateContentProvider;
+
+ public TemplateContentDemo(ITemplateContentProvider templateContentProvider)
+ {
+ _templateContentProvider = templateContentProvider;
+ }
+
+ public async Task RunAsync()
+ {
+ var result = await _templateContentProvider
+ .GetContentOrNullAsync("Hello");
+
+ Console.WriteLine(result);
+ }
+}
+````
+
+The result will be the raw template content:
+
+````
+Hello {%{{{model.name}}}%} :)
+````
+
+* `GetContentOrNullAsync` returns `null` if no content defined for the requested template.
+* It can get a `cultureName` parameter that is used if template has different files for different cultures (see Multiple Contents Localization section above).
+
+### Template Content Contributor
+
+`ITemplateContentProvider` service uses `ITemplateContentContributor` implementations to find template contents. There is a single pre-implemented content contributor, `VirtualFileTemplateContentContributor`, which gets template contents from the virtual file system as described above.
+
+You can implement the `ITemplateContentContributor` to read raw template contents from another source.
+
+Example:
+
+````csharp
+public class MyTemplateContentProvider
+ : ITemplateContentContributor, ITransientDependency
+{
+ public async Task GetOrNullAsync(TemplateContentContributorContext context)
+ {
+ var templateName = context.TemplateDefinition.Name;
+
+ //TODO: Try to find content from another source
+ return null;
+ }
+}
+
+````
+
+Return `null` if your source can not find the content, so `ITemplateContentProvider` fallbacks to the next contributor.
+
+### Template Definition Manager
+
+`ITemplateDefinitionManager` service can be used to get the template definitions (created by the template definition providers).
+
+## See Also
+
+* [The source code of the sample application](https://github.com/abpframework/abp-samples/tree/master/TextTemplateDemo) developed and referred through this document.
+* [Localization system](Localization.md).
+* [Virtual File System](Virtual-File-System.md).
diff --git a/docs/en/Text-Templating.md b/docs/en/Text-Templating.md
index bdea7cd00b..c9b03cd940 100644
--- a/docs/en/Text-Templating.md
+++ b/docs/en/Text-Templating.md
@@ -12,549 +12,21 @@ It is very similar to an ASP.NET Core Razor View (or Page):
You can use the rendered output for any purpose, like sending emails or preparing some reports.
-### Example
-
-Here, a simple template:
-
-````
-Hello {%{{{model.name}}}%} :)
-````
-
-You can define a class with a `Name` property to render this template:
-
-````csharp
-public class HelloModel
-{
- public string Name { get; set; }
-}
-````
-
-If you render the template with a `HelloModel` with the `Name` is `John`, the rendered output is will be:
-
-````
-Hello John :)
-````
-
Template rendering engine is very powerful;
-* It is based on the [Scriban](https://github.com/lunet-io/scriban) library, so it supports **conditional logics**, **loops** and much more.
+* It supports **conditional logics**, **loops** and much more.
* Template content **can be localized**.
* You can define **layout templates** to be used as the layout while rendering other templates.
* You can pass arbitrary objects to the template context (beside the model) for advanced scenarios.
-### Source Code
-
-Get [the source code of the sample application](https://github.com/abpframework/abp-samples/tree/master/TextTemplateDemo) developed and referred through this document.
-
-## Installation
-
-It is suggested to use the [ABP CLI](CLI.md) to install this package.
-
-### Using the ABP CLI
-
-Open a command line window in the folder of the project (.csproj file) and type the following command:
-
-````bash
-abp add-package Volo.Abp.TextTemplating
-````
-
-### Manual Installation
-
-If you want to manually install;
-
-1. Add the [Volo.Abp.TextTemplating](https://www.nuget.org/packages/Volo.Abp.TextTemplating) NuGet package to your project:
-
-````
-Install-Package Volo.Abp.TextTemplating
-````
-
-2. Add the `AbpTextTemplatingModule` to the dependency list of your module:
-
-````csharp
-[DependsOn(
- //...other dependencies
- typeof(AbpTextTemplatingModule) //Add the new module dependency
- )]
-public class YourModule : AbpModule
-{
-}
-````
-
-## Defining Templates
-
-Before rendering a template, you should define it. Create a class inheriting from the `TemplateDefinitionProvider` base class:
-
-````csharp
-public class DemoTemplateDefinitionProvider : TemplateDefinitionProvider
-{
- public override void Define(ITemplateDefinitionContext context)
- {
- context.Add(
- new TemplateDefinition("Hello") //template name: "Hello"
- .WithVirtualFilePath(
- "/Demos/Hello/Hello.tpl", //template content path
- isInlineLocalized: true
- )
- );
- }
-}
-````
-
-* `context` object is used to add new templates or get the templates defined by depended modules. Used `context.Add(...)` to define a new template.
-* `TemplateDefinition` is the class represents a template. Each template must have a unique name (that will be used while you are rendering the template).
-* `/Demos/Hello/Hello.tpl` is the path of the template file.
-* `isInlineLocalized` is used to declare if you are using a single template for all languages (`true`) or different templates for each language (`false`). See the Localization section below for more.
-
-### The Template Content
-
-`WithVirtualFilePath` indicates that we are using the [Virtual File System](Virtual-File-System.md) to store the template content. Create a `Hello.tpl` file inside your project and mark it as "**embedded resource**" on the properties window:
-
-
-
-Example `Hello.tpl` content is shown below:
-
-````
-Hello {%{{{model.name}}}%} :)
-````
-
-The [Virtual File System](Virtual-File-System.md) requires to add your files in the `ConfigureServices` method of your [module](Module-Development-Basics.md) class:
-
-````csharp
-Configure(options =>
-{
- options.FileSets.AddEmbedded("TextTemplateDemo");
-});
-````
-
-* `TextTemplateDemoModule` is the module class that you define your template in.
-* `TextTemplateDemo` is the root namespace of your project.
-
-## Rendering the Template
-
-`ITemplateRenderer` service is used to render a template content.
-
-### Example: Rendering a Simple Template
-
-````csharp
-public class HelloDemo : ITransientDependency
-{
- private readonly ITemplateRenderer _templateRenderer;
-
- public HelloDemo(ITemplateRenderer templateRenderer)
- {
- _templateRenderer = templateRenderer;
- }
-
- public async Task RunAsync()
- {
- var result = await _templateRenderer.RenderAsync(
- "Hello", //the template name
- new HelloModel
- {
- Name = "John"
- }
- );
-
- Console.WriteLine(result);
- }
-}
-````
-
-* `HelloDemo` is a simple class that injects the `ITemplateRenderer` in its constructor and uses it inside the `RunAsync` method.
-* `RenderAsync` gets two fundamental parameters:
- * `templateName`: The name of the template to be rendered (`Hello` in this example).
- * `model`: An object that is used as the `model` inside the template (a `HelloModel` object in this example).
-
-The result shown below for this example:
-
-````csharp
-Hello John :)
-````
-
-### Anonymous Model
-
-While it is suggested to create model classes for the templates, it would be practical (and possible) to use anonymous objects for simple cases:
-
-````csharp
-var result = await _templateRenderer.RenderAsync(
- "Hello",
- new
- {
- Name = "John"
- }
-);
-````
-
-In this case, we haven't created a model class, but created an anonymous object as the model.
-
-### PascalCase vs snake_case
-
-PascalCase property names (like `UserName`) is used as snake_case (like `user_name`) in the templates.
-
-## Localization
-
-It is possible to localize a template content based on the current culture. There are two types of localization options described in the following sections.
-
-### Inline localization
-
-Inline localization uses the [localization system](Localization.md) to localize texts inside templates.
-
-#### Example: Reset Password Link
-
-Assuming you need to send an email to a user to reset her/his password. Here, the template content:
-
-````
-{%{{{L "ResetMyPassword" model.name}}}%}
-````
-
-`L` function is used to localize the given key based on the current user culture. You need to define the `ResetMyPassword` key inside your localization file:
-
-````json
-"ResetMyPasswordTitle": "Reset my password",
-"ResetMyPassword": "Hi {0}, Click here to reset your password"
-````
-
-You also need to declare the localization resource to be used with this template, inside your template definition provider class:
-
-````csharp
-context.Add(
- new TemplateDefinition(
- "PasswordReset", //Template name
- typeof(DemoResource) //LOCALIZATION RESOURCE
- ).WithVirtualFilePath(
- "/Demos/PasswordReset/PasswordReset.tpl", //template content path
- isInlineLocalized: true
- )
-);
-````
-
-That's all. When you render this template like that:
-
-````csharp
-var result = await _templateRenderer.RenderAsync(
- "PasswordReset", //the template name
- new PasswordResetModel
- {
- Name = "john",
- Link = "https://abp.io/example-link?userId=123&token=ABC"
- }
-);
-````
-
-You will see the localized result:
-
-````csharp
-Hi john, Click here to reset your password
-````
-
-> If you define the [default localization resource](Localization.md) for your application, then no need to declare the resource type for the template definition.
-
-### Multiple Contents Localization
-
-Instead of a single template that uses the localization system to localize the template, you may want to create different template files for each language. It can be needed if the template should be completely different for a specific culture rather than simple text localizations.
-
-#### Example: Welcome Email Template
-
-Assuming that you want to send a welcome email to your users, but want to define a completely different template based on the user culture.
-
-First, create a folder and put your templates inside it, like `en.tpl`, `tr.tpl`... one for each culture you support:
-
-
-
-Then add your template definition in the template definition provider class:
-
-````csharp
-context.Add(
- new TemplateDefinition(
- name: "WelcomeEmail",
- defaultCultureName: "en"
- )
- .WithVirtualFilePath(
- "/Demos/WelcomeEmail/Templates", //template content folder
- isInlineLocalized: false
- )
-);
-````
-
-* Set **default culture name**, so it fallbacks to the default culture if there is no template for the desired culture.
-* Specify **the template folder** rather than a single template file.
-* Set `isInlineLocalized` to `false` for this case.
-
-That's all, you can render the template for the current culture:
-
-````csharp
-var result = await _templateRenderer.RenderAsync("WelcomeEmail");
-````
-
-> Skipped the modal for this example to keep it simple, but you can use models as just explained before.
+ABP Framework provides two types of engines;
-### Specify the Culture
+* **[Razor](Text-Templating-Razor.md)**
+* **[Scriban](Text-Templating-Scriban.md)**
-`ITemplateRenderer` service uses the current culture (`CultureInfo.CurrentUICulture`) if not specified. If you need, you can specify the culture as the `cultureName` parameter:
+## Source Code
-````csharp
-var result = await _templateRenderer.RenderAsync(
- "WelcomeEmail",
- cultureName: "en"
-);
-````
-
-## Layout Templates
-
-Layout templates are used to create shared layouts among other templates. It is similar to the layout system in the ASP.NET Core MVC / Razor Pages.
-
-### Example: Email HTML Layout Template
-
-For example, you may want to create a single layout for all of your email templates.
-
-First, create a template file just like before:
-
-````xml
-
-
-
-
-
-
- {%{{{content}}}%}
-
-
-````
-
-* A layout template must have a **{%{{{content}}}%}** part as a place holder for the rendered child content.
-
-The register your template in the template definition provider:
-
-````csharp
-context.Add(
- new TemplateDefinition(
- "EmailLayout",
- isLayout: true //SET isLayout!
- ).WithVirtualFilePath(
- "/Demos/EmailLayout/EmailLayout.tpl",
- isInlineLocalized: true
- )
-);
-````
-
-Now, you can use this template as the layout of any other template:
-
-````csharp
-context.Add(
- new TemplateDefinition(
- name: "WelcomeEmail",
- defaultCultureName: "en",
- layout: "EmailLayout" //Set the LAYOUT
- ).WithVirtualFilePath(
- "/Demos/WelcomeEmail/Templates",
- isInlineLocalized: false
- )
-);
-````
-
-## Global Context
-
-ABP passes the `model` that can be used to access to the model inside the template. You can pass more global variables if you need.
-
-An example template content:
-
-````
-A global object value: {%{{{myGlobalObject}}}%}
-````
-
-This template assumes that that is a `myGlobalObject` object in the template rendering context. You can provide it like shown below:
-
-````csharp
-var result = await _templateRenderer.RenderAsync(
- "GlobalContextUsage",
- globalContext: new Dictionary
- {
- {"myGlobalObject", "TEST VALUE"}
- }
-);
-````
-
-The rendering result will be:
-
-````
-A global object value: TEST VALUE
-````
-
-## Replacing the Existing Templates
-
-It is possible to replace a template defined by a module that used in your application. In this way, you can customize the templates based on your requirements without changing the module code.
-
-### Option-1: Using the Virtual File System
-
-The [Virtual File System](Virtual-File-System.md) allows you to override any file by placing the same file into the same path in your project.
-
-#### Example: Replace the Standard Email Layout Template
-
-ABP Framework provides an [email sending system](Emailing.md) that internally uses the text templating to render the email content. It defines a standard email layout template in the `/Volo/Abp/Emailing/Templates/Layout.tpl` path. The unique name of the template is `Abp.StandardEmailTemplates.Layout` and this string is defined as a constant on the `Volo.Abp.Emailing.Templates.StandardEmailTemplates` static class.
-
-Do the following steps to replace the template file with your own;
-
-**1)** Add a new file into the same location (`/Volo/Abp/Emailing/Templates/Layout.tpl`) in your project:
-
-
-
-**2)** Prepare your email layout template:
-
-````html
-
-
-
-
-
-
-
This my header
-
- {%{{{content}}}%}
-
-
-
-
-````
-
-This example simply adds a header and footer to the template and renders the content between them (see the *Layout Templates* section above to understand it).
-
-**3)** Configure the embedded resources in the `.csproj` file
-
-* Add [Microsoft.Extensions.FileProviders.Embedded](https://www.nuget.org/packages/Microsoft.Extensions.FileProviders.Embedded) NuGet package to the project.
-* Add `true` into the `...` section of your `.csproj` file.
-* Add the following code into your `.csproj` file:
-
-````xml
-
-
-
-
-````
-
-This makes the template files "embedded resource".
-
-**4)** Configure the virtual file system
-
-Configure the `AbpVirtualFileSystemOptions` in the `ConfigureServices` method of your [module](Module-Development-Basics.md) to add the embedded files into the virtual file system:
-
-```csharp
-Configure(options =>
-{
- options.FileSets.AddEmbedded();
-});
-```
-
-`BookStoreDomainModule` should be your module name, in this example code.
-
-> Be sure that your module (directly or indirectly) [depends on](Module-Development-Basics.md) the `AbpEmailingModule`. Because the VFS can override files based on the dependency order.
-
-Now, your template will be used when you want to render the email layout template.
-
-### Option-2: Using the Template Definition Provider
-
-You can create a template definition provider class that gets the email layout template and changes the virtual file path for the template.
-
-**Example: Use the `/MyTemplates/EmailLayout.tpl` file instead of the standard template**
-
-```csharp
-using Volo.Abp.DependencyInjection;
-using Volo.Abp.Emailing.Templates;
-using Volo.Abp.TextTemplating;
-
-namespace MyProject
-{
- public class MyTemplateDefinitionProvider
- : TemplateDefinitionProvider, ITransientDependency
- {
- public override void Define(ITemplateDefinitionContext context)
- {
- var emailLayoutTemplate = context.GetOrNull(StandardEmailTemplates.Layout);
-
- emailLayoutTemplate
- .WithVirtualFilePath(
- "/MyTemplates/EmailLayout.tpl",
- isInlineLocalized: true
- );
- }
- }
-}
-```
-
-You should still add the file `/MyTemplates/EmailLayout.tpl` to the virtual file system as explained before. This approach allows you to locate templates in any folder instead of the folder defined by the depended module.
-
-Beside the template content, you can manipulate the template definition properties, like `DisplayName`, `Layout` or `LocalizationSource`.
-
-## Advanced Features
-
-This section covers some internals and more advanced usages of the text templating system.
-
-### Template Content Provider
-
-`ITemplateRenderer` is used to render the template, which is what you want for most of the cases. However, you can use the `ITemplateContentProvider` to get the raw (not rendered) template contents.
-
-> `ITemplateContentProvider` is internally used by the `ITemplateRenderer` to get the raw template contents.
-
-Example:
-
-````csharp
-public class TemplateContentDemo : ITransientDependency
-{
- private readonly ITemplateContentProvider _templateContentProvider;
-
- public TemplateContentDemo(ITemplateContentProvider templateContentProvider)
- {
- _templateContentProvider = templateContentProvider;
- }
-
- public async Task RunAsync()
- {
- var result = await _templateContentProvider
- .GetContentOrNullAsync("Hello");
-
- Console.WriteLine(result);
- }
-}
-````
-
-The result will be the raw template content:
-
-````
-Hello {%{{{model.name}}}%} :)
-````
-
-* `GetContentOrNullAsync` returns `null` if no content defined for the requested template.
-* It can get a `cultureName` parameter that is used if template has different files for different cultures (see Multiple Contents Localization section above).
-
-### Template Content Contributor
-
-`ITemplateContentProvider` service uses `ITemplateContentContributor` implementations to find template contents. There is a single pre-implemented content contributor, `VirtualFileTemplateContentContributor`, which gets template contents from the virtual file system as described above.
-
-You can implement the `ITemplateContentContributor` to read raw template contents from another source.
-
-Example:
-
-````csharp
-public class MyTemplateContentProvider
- : ITemplateContentContributor, ITransientDependency
-{
- public async Task GetOrNullAsync(TemplateContentContributorContext context)
- {
- var templateName = context.TemplateDefinition.Name;
-
- //TODO: Try to find content from another source
- return null;
- }
-}
-
-````
-
-Return `null` if your source can not find the content, so `ITemplateContentProvider` fallbacks to the next contributor.
-
-### Template Definition Manager
-
-`ITemplateDefinitionManager` service can be used to get the template definitions (created by the template definition providers).
+Get [the source code of the sample application](https://github.com/abpframework/abp-samples/tree/master/TextTemplateDemo) developed and referred through this document.
## See Also
diff --git a/docs/en/docs-nav.json b/docs/en/docs-nav.json
index 03c8ff2c4c..0a7722de78 100644
--- a/docs/en/docs-nav.json
+++ b/docs/en/docs-nav.json
@@ -9,7 +9,21 @@
"items": [
{
"text": "Web Application",
- "path": "Getting-Started.md"
+ "path": "Getting-Started.md",
+ "items": [
+ {
+ "text": "1: Setup Your Development Environment",
+ "path": "Getting-Started-Setup-Environment.md"
+ },
+ {
+ "text": "2: Creating a New Solution",
+ "path": "Getting-Started-Create-Solution.md"
+ },
+ {
+ "text": "3: Running the Solution",
+ "path": "Getting-Started-Running-Solution.md"
+ }
+ ]
},
{
"text": "Console Application",
@@ -342,7 +356,17 @@
},
{
"text": "Text Templating",
- "path": "Text-Templating.md"
+ "path": "Text-Templating.md",
+ "items": [
+ {
+ "text": "Razor Integration",
+ "path": "Text-Templating-Razor.md"
+ },
+ {
+ "text": "Scriban Integration",
+ "path": "Text-Templating-Scriban.md"
+ }
+ ]
},
{
"text": "Timing",
@@ -376,9 +400,70 @@
"text": "Customizing/Extending Modules",
"path": "Customizing-Application-Modules-Guide.md"
},
+ {
+ "text": "Customizing/Extending Entities",
+ "path": "Customizing-Application-Modules-Extending-Entities.md"
+ },
+ {
+ "text": "Customizing/Overriding Services",
+ "path": "Customizing-Application-Modules-Overriding-Services.md"
+ },
+ {
+ "text": "Module Entity Extensions",
+ "path": "Module-Entity-Extensions.md"
+ },
{
"text": "Best Practices",
- "path": "Best-Practices/Index.md"
+ "path": "Best-Practices/Index.md",
+ "items": [
+ {
+ "text": "Module Architecture",
+ "path": "Best-Practices/Module-Architecture.md"
+ },
+ {
+ "text": "Domain Layer",
+ "items": [
+ {
+ "text": "Entities",
+ "path": "Best-Practices/Entities.md"
+ },
+ {
+ "text": "Repositories",
+ "path": "Best-Practices/Repositories.md"
+ },
+ {
+ "text": "Domain Services",
+ "path": "Best-Practices/Domain-Services.md"
+ }
+ ]
+ },
+ {
+ "text": "Application Layer",
+ "items": [
+ {
+ "text": "Application Services",
+ "path": "Best-Practices/Application-Services.md"
+ },
+ {
+ "text": "Data Transfer Objects",
+ "path": "Best-Practices/Data-Transfer-Objects.md"
+ }
+ ]
+ },
+ {
+ "text": "Data Access",
+ "items": [
+ {
+ "text": "Entity Framework Core Integration",
+ "path": "Best-Practices/Entity-Framework-Core-Integration.md"
+ },
+ {
+ "text": "MongoDB Integration",
+ "path": "Best-Practices/MongoDB-Integration.md"
+ }
+ ]
+ }
+ ]
}
]
},
@@ -396,6 +481,10 @@
"text": "Entities & Aggregate Roots",
"path": "Entities.md"
},
+ {
+ "text": "Multi Lingual Entities",
+ "path": "Multi-Lingual-Entities.md"
+ },
{
"text": "Value Objects",
"path": "Value-Objects.md"
@@ -623,6 +712,18 @@
{
"text": "Customize/Extend the UI",
"path": "UI/AspNetCore/Customization-User-Interface.md"
+ },
+ {
+ "text": "Entity Action Extensions",
+ "path": "UI/AspNetCore/Entity-Action-Extensions.md"
+ },
+ {
+ "text": "Data Table Column Extensions",
+ "path": "UI/AspNetCore/Data-Table-Column-Extensions.md"
+ },
+ {
+ "text": "Page Toolbar Extensions",
+ "path": "UI/AspNetCore/Page-Toolbar-Extensions.md"
}
]
},
@@ -809,6 +910,10 @@
{
"text": "Multi Tenancy",
"path": "UI/Angular/Multi-Tenancy.md"
+ },
+ {
+ "text": "Account Module",
+ "path": "UI/Angular/Account-Module"
}
]
},
@@ -862,12 +967,36 @@
{
"text": "Ellipsis",
"path": "UI/Angular/Ellipsis-Directive.md"
+ },
+ {
+ "text": "Context Strategy",
+ "path": "UI/Angular/Context-Strategy.md"
+ },
+ {
+ "text": "Cross Origin Strategy",
+ "path": "UI/Angular/Cross-Origin-Strategy.md"
+ },
+ {
+ "text": "Dom Strategy",
+ "path": "UI/Angular/Dom-Strategy.md"
+ },
+ {
+ "text": "Container Strategy",
+ "path": "UI/Angular/Container-Strategy.md"
+ },
+ {
+ "text": "Content Security Strategy",
+ "path": "UI/Angular/Content-Security-Strategy.md"
}
]
},
{
"text": "Customization",
"items": [
+ {
+ "text": "Customization Guide",
+ "path": "UI/Angular/Customization-User-Interface.md"
+ },
{
"text": "Modifying the Menu",
"path": "UI/Angular/Modifying-the-Menu.md"
@@ -926,6 +1055,10 @@
{
"text": "Common",
"items": [
+ {
+ "text": "Overriding the User Interface",
+ "path": "Customizing-Application-Modules-Overriding-User-Interface.md"
+ },
{
"text": "Utilities",
"items": [
@@ -968,7 +1101,17 @@
},
{
"text": "To Oracle",
- "path": "Entity-Framework-Core-Oracle.md"
+ "path": "Entity-Framework-Core-Oracle.md",
+ "items": [
+ {
+ "text": "Oracle",
+ "path": "Entity-Framework-Core-Oracle-Official.md"
+ },
+ {
+ "text": "Oracle Devart",
+ "path": "Entity-Framework-Core-Oracle-Devart.md"
+ }
+ ]
},
{
"text": "To SQLite",
@@ -1096,7 +1239,56 @@
},
{
"text": "Migration Guides",
- "path": "Migration-Guides/Index.md"
+ "path": "Migration-Guides/Index.md",
+ "items": [
+ {
+ "text": "MVC / Razor Pages",
+ "items": [
+ {
+ "text": "3.3 to 4.0",
+ "path": "Migration-Guides/Abp-4_0-MVC-Razor-Pages.md"
+ }
+ ]
+ },
+ {
+ "text": "Blazor",
+ "items": [
+ {
+ "text": "3.2 to 3.3",
+ "path": "Migration-Guides/BlazorUI-3_3.md"
+ },
+ {
+ "text": "3.3 to 4.0",
+ "path": "Migration-Guides/Abp-4_0-Blazor.md"
+ }
+ ]
+ },
+ {
+ "text": "Angular",
+ "items": [
+ {
+ "text": "3.3 to 4.0",
+ "path": "Migration-Guides/Abp-4_0-Angular.md"
+ }
+ ]
+ },
+ {
+ "text": "3.3 to 4.0",
+ "path": "Migration-Guides/Abp-4_0.md"
+ },
+ {
+ "text": "4.2",
+ "path": "Migration-Guides/Abp-4_2.md"
+ },
+ {
+ "text": "4.x to 4.3",
+ "path": "Migration-Guides/Abp-4_3.md"
+ },
+ {
+ "text": "Upgrading the Startup Template",
+ "path": "Migration-Guides/Upgrading-Startup-Template.md"
+ }
+ ]
}
]
},
@@ -1105,7 +1297,17 @@
"items": [
{
"text": "CLI",
- "path": "CLI.md"
+ "path": "CLI.md",
+ "items": [
+ {
+ "text": "Build Command",
+ "path": "CLI-BuildCommand.md"
+ },
+ {
+ "text": "Create Solution Sample Commands",
+ "path": "CLI-New-Command-Samples.md"
+ }
+ ]
}
]
},
diff --git a/docs/en/images/hello-template-razor.png b/docs/en/images/hello-template-razor.png
new file mode 100644
index 0000000000..1e0e39fcec
Binary files /dev/null and b/docs/en/images/hello-template-razor.png differ
diff --git a/docs/en/images/multiple-file-template-razor.png b/docs/en/images/multiple-file-template-razor.png
new file mode 100644
index 0000000000..85f87e84f4
Binary files /dev/null and b/docs/en/images/multiple-file-template-razor.png differ
diff --git a/docs/en/images/replace-email-layout-razor.png b/docs/en/images/replace-email-layout-razor.png
new file mode 100644
index 0000000000..84ff6c1080
Binary files /dev/null and b/docs/en/images/replace-email-layout-razor.png differ
diff --git a/docs/zh-Hans/Value-Objects.md b/docs/zh-Hans/Value-Objects.md
index 6735074fd4..c5dd929ff2 100644
--- a/docs/zh-Hans/Value-Objects.md
+++ b/docs/zh-Hans/Value-Objects.md
@@ -1,3 +1,75 @@
-## Value Objects
+## 值对象
- TODO
\ No newline at end of file
+> 一个对象,表示领域的描述方面,没有概念上的身份被称为 值对象.
+>
+> (Eric Evans)
+
+属性相同但`Id`不同的两个[实体](https://docs.abp.io/zh-Hans/abp/latest/Entities) 被视为不同的实体.但是,值对象没有`Id`
+
+## 值对象的类
+
+值对象是一个抽象类,可以继承它来创建值对象类
+
+**示例: An Address class**
+
+```csharp
+public class Address : ValueObject
+{
+ public Guid CityId { get; private set; }
+
+ public string Street { get; private set; }
+
+ public int Number { get; private set; }
+
+ private Address()
+ {
+
+ }
+
+ public Address(
+ Guid cityId,
+ string street,
+ int number)
+ {
+ CityId = cityId;
+ Street = street;
+ Number = number;
+ }
+
+ protected override IEnumerable
diff --git a/framework/src/Volo.Abp.ExceptionHandling/Volo/Abp/ExceptionHandling/Localization/fi.json b/framework/src/Volo.Abp.ExceptionHandling/Volo/Abp/ExceptionHandling/Localization/fi.json
new file mode 100644
index 0000000000..03f1ce3b5c
--- /dev/null
+++ b/framework/src/Volo.Abp.ExceptionHandling/Volo/Abp/ExceptionHandling/Localization/fi.json
@@ -0,0 +1,25 @@
+{
+ "culture": "fi",
+ "texts": {
+ "InternalServerErrorMessage": "Pyynnön aikana tapahtui sisäinen virhe!",
+ "ValidationErrorMessage": "Pyyntösi ei kelpaa!",
+ "ValidationNarrativeErrorMessageTitle": "Seuraavat virheet havaittiin validoinnin aikana.",
+ "DefaultErrorMessage": "Tapahtui virhe!",
+ "DefaultErrorMessageDetail": "Palvelin ei lähettänyt virhetietoja.",
+ "DefaultErrorMessage401": "Sinua ei ole todennettu!",
+ "DefaultErrorMessage401Detail": "Suorita tämä toimenpide kirjautumalla sisään.",
+ "DefaultErrorMessage403": "Sinulla ei ole valtuuksia!",
+ "DefaultErrorMessage403Detail": "Et saa suorittaa tätä toimintoa!",
+ "DefaultErrorMessage404": "Resurssia ei löydy!",
+ "DefaultErrorMessage404Detail": "Pyydettyä resurssia ei löytynyt palvelimelta!",
+ "EntityNotFoundErrorMessage": "Ei ole olemassa kohdetta {0}, jonka tunnus = {1}!",
+ "Error": "Virhe",
+ "UnhandledException": "Käsittelemätön poikkeus!",
+ "401Message": "Luvaton",
+ "403Message": "Kielletty",
+ "404Message": "Sivua ei löydetty",
+ "500Message": "Sisäinen palvelinvirhe",
+ "403MessageDetail": "Sinulla ei ole oikeutta suorittaa tätä toimintoa!",
+ "404MessageDetail": "Valitettavasti tässä osoitteessa ei ole mitään."
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.ExceptionHandling/Volo/Abp/ExceptionHandling/Localization/fr.json b/framework/src/Volo.Abp.ExceptionHandling/Volo/Abp/ExceptionHandling/Localization/fr.json
index f1c9713e65..cdbd620313 100644
--- a/framework/src/Volo.Abp.ExceptionHandling/Volo/Abp/ExceptionHandling/Localization/fr.json
+++ b/framework/src/Volo.Abp.ExceptionHandling/Volo/Abp/ExceptionHandling/Localization/fr.json
@@ -18,6 +18,8 @@
"401Message": "Non autorisé",
"403Message": "Interdit",
"404Message": "Page introuvable",
- "500Message": "Erreur Interne du Serveur"
+ "500Message": "Erreur Interne du Serveur",
+ "403MessageDetail": "Vous n'êtes pas autorisé à effectuer cette opération!",
+ "404MessageDetail": "Désolé, il n'y a rien à cette adresse."
}
-}
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.Features/Volo/Abp/Features/Localization/FR.json b/framework/src/Volo.Abp.Features/Volo/Abp/Features/Localization/FR.json
new file mode 100644
index 0000000000..19fb314868
--- /dev/null
+++ b/framework/src/Volo.Abp.Features/Volo/Abp/Features/Localization/FR.json
@@ -0,0 +1,8 @@
+{
+ "culture": "fr",
+ "texts": {
+ "Volo.Feature:010001": "La fonctionnalité n'est pas activée: {FeatureName}",
+ "Volo.Feature:010002": "Les fonctionnalités requises ne sont pas activées. Toutes ces fonctionnalités doivent être activées: {FeatureNames}",
+ "Volo.Feature:010003": "Les fonctionnalités requises ne sont pas activées. Au moins une de ces fonctionnalités doit être activée: {FeatureNames}"
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.Features/Volo/Abp/Features/Localization/fi.json b/framework/src/Volo.Abp.Features/Volo/Abp/Features/Localization/fi.json
new file mode 100644
index 0000000000..4bc400d6f9
--- /dev/null
+++ b/framework/src/Volo.Abp.Features/Volo/Abp/Features/Localization/fi.json
@@ -0,0 +1,8 @@
+{
+ "culture": "fi",
+ "texts": {
+ "Volo.Feature:010001": "Ominaisuus ei ole käytössä: {FeatureName}",
+ "Volo.Feature:010002": "Vaaditut ominaisuudet eivät ole käytössä. Kaikkien näiden ominaisuuksien on oltava käytössä: {FeatureNames}",
+ "Volo.Feature:010003": "Vaaditut ominaisuudet eivät ole käytössä. Ainakin yhden näistä ominaisuuksista on oltava käytössä: {FeatureNames}"
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.GlobalFeatures/Volo.Abp.GlobalFeatures.csproj b/framework/src/Volo.Abp.GlobalFeatures/Volo.Abp.GlobalFeatures.csproj
index 456366b542..024ec5ec17 100644
--- a/framework/src/Volo.Abp.GlobalFeatures/Volo.Abp.GlobalFeatures.csproj
+++ b/framework/src/Volo.Abp.GlobalFeatures/Volo.Abp.GlobalFeatures.csproj
@@ -20,7 +20,6 @@
-
diff --git a/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/FR.json b/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/FR.json
new file mode 100644
index 0000000000..8062337807
--- /dev/null
+++ b/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/FR.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fr",
+ "texts": {
+ "Volo.GlobalFeature:010001": "Le service \"{ServiceName}\" doit activer la fonction \"{GlobalFeatureName}\"."
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/fi.json b/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/fi.json
new file mode 100644
index 0000000000..b842a7e3a4
--- /dev/null
+++ b/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/fi.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fi",
+ "texts": {
+ "Volo.GlobalFeature:010001": "Palvelun {ServiceName} on oltava käytössä {GlobalFeatureName} -ominaisuus."
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/nl.json b/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/nl.json
new file mode 100644
index 0000000000..449c09efd6
--- /dev/null
+++ b/framework/src/Volo.Abp.GlobalFeatures/Volo/Abp/GlobalFeatures/Localization/nl.json
@@ -0,0 +1,6 @@
+{
+ "culture": "nl",
+ "texts": {
+ "Volo.GlobalFeature:010001": "De service '{ServiceName}' moet de functie '{GlobalFeatureName}' inschakelen."
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/UrlBuilder.cs b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/UrlBuilder.cs
index 5bee246bd0..4d77423c7f 100644
--- a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/UrlBuilder.cs
+++ b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/UrlBuilder.cs
@@ -15,12 +15,18 @@ namespace Volo.Abp.Http.Client.DynamicProxying
{
public static string GenerateUrlWithParameters(ActionApiDescriptionModel action, IReadOnlyDictionary methodArguments, ApiVersionInfo apiVersion)
{
- var urlBuilder = new StringBuilder(action.Url);
+ // The ASP.NET Core route value provider and query string value provider:
+ // Treat values as invariant culture.
+ // Expect that URLs are culture-invariant.
+ using (CultureHelper.Use(CultureInfo.InvariantCulture))
+ {
+ var urlBuilder = new StringBuilder(action.Url);
- ReplacePathVariables(urlBuilder, action.Parameters, methodArguments, apiVersion);
- AddQueryStringParameters(urlBuilder, action.Parameters, methodArguments, apiVersion);
+ ReplacePathVariables(urlBuilder, action.Parameters, methodArguments, apiVersion);
+ AddQueryStringParameters(urlBuilder, action.Parameters, methodArguments, apiVersion);
- return urlBuilder.ToString();
+ return urlBuilder.ToString();
+ }
}
private static void ReplacePathVariables(StringBuilder urlBuilder, IList actionParameters, IReadOnlyDictionary methodArguments, ApiVersionInfo apiVersion)
@@ -128,15 +134,12 @@ namespace Volo.Abp.Http.Client.DynamicProxying
private static string ConvertValueToString([NotNull] object value)
{
- using (CultureHelper.Use(CultureInfo.InvariantCulture))
+ if (value is DateTime dateTimeValue)
{
- if (value is DateTime dateTimeValue)
- {
- return dateTimeValue.ToUniversalTime().ToString("u");
- }
-
- return value.ToString();
+ return dateTimeValue.ToUniversalTime().ToString("O");
}
+
+ return value.ToString();
}
}
}
diff --git a/framework/src/Volo.Abp.Http/Volo/Abp/Http/ProxyScripting/Generators/ProxyScriptingHelper.cs b/framework/src/Volo.Abp.Http/Volo/Abp/Http/ProxyScripting/Generators/ProxyScriptingHelper.cs
index 700dc4e290..2568ccbb1c 100644
--- a/framework/src/Volo.Abp.Http/Volo/Abp/Http/ProxyScripting/Generators/ProxyScriptingHelper.cs
+++ b/framework/src/Volo.Abp.Http/Volo/Abp/Http/ProxyScripting/Generators/ProxyScriptingHelper.cs
@@ -1,7 +1,9 @@
using System;
using System.Collections.Generic;
+using System.Globalization;
using System.Linq;
using Volo.Abp.Http.Modeling;
+using Volo.Abp.Localization;
namespace Volo.Abp.Http.ProxyScripting.Generators
{
@@ -11,10 +13,16 @@ namespace Volo.Abp.Http.ProxyScripting.Generators
public static string GenerateUrlWithParameters(ActionApiDescriptionModel action)
{
- //TODO: Can be optimized using StringBuilder?
- var url = ReplacePathVariables(action.Url, action.Parameters);
- url = AddQueryStringParameters(url, action.Parameters);
- return url;
+ // The ASP.NET Core route value provider and query string value provider:
+ // Treat values as invariant culture.
+ // Expect that URLs are culture-invariant.
+ using (CultureHelper.Use(CultureInfo.InvariantCulture))
+ {
+ //TODO: Can be optimized using StringBuilder?
+ var url = ReplacePathVariables(action.Url, action.Parameters);
+ url = AddQueryStringParameters(url, action.Parameters);
+ return url;
+ }
}
public static string GenerateHeaders(ActionApiDescriptionModel action, int indent = 0)
diff --git a/framework/src/Volo.Abp.Ldap/Volo/Abp/Ldap/Localization/fi.json b/framework/src/Volo.Abp.Ldap/Volo/Abp/Ldap/Localization/fi.json
new file mode 100644
index 0000000000..953f4491db
--- /dev/null
+++ b/framework/src/Volo.Abp.Ldap/Volo/Abp/Ldap/Localization/fi.json
@@ -0,0 +1,15 @@
+{
+ "culture": "fi",
+ "texts": {
+ "DisplayName:Abp.Ldap.ServerHost": "Palvelimen isäntä",
+ "Description:Abp.Ldap.ServerHost": "Palvelimen isäntä",
+ "DisplayName:Abp.Ldap.ServerPort": "Palvelimen portti",
+ "Description:Abp.Ldap.ServerPort": "Palvelimen portti",
+ "DisplayName:Abp.Ldap.BaseDc": "Perustoimialueen komponentti",
+ "Description:Abp.Ldap.BaseDc": "Perustoimialueen komponentti",
+ "DisplayName:Abp.Ldap.UserName": "Käyttäjätunnus",
+ "Description:Abp.Ldap.UserName": "Käyttäjätunnus",
+ "DisplayName:Abp.Ldap.Password": "Salasana",
+ "Description:Abp.Ldap.Password": "Salasana"
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.Localization/Volo/Abp/Localization/Resources/AbpLocalization/fi.json b/framework/src/Volo.Abp.Localization/Volo/Abp/Localization/Resources/AbpLocalization/fi.json
new file mode 100644
index 0000000000..19b96532de
--- /dev/null
+++ b/framework/src/Volo.Abp.Localization/Volo/Abp/Localization/Resources/AbpLocalization/fi.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fi",
+ "texts": {
+ "DisplayName:Abp.Localization.DefaultLanguage": "Oletuskieli",
+ "Description:Abp.Localization.DefaultLanguage": "Sovelluksen oletuskieli."
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.Localization/Volo/Abp/Localization/VirtualFiles/VirtualFileLocalizationResourceContributorBase.cs b/framework/src/Volo.Abp.Localization/Volo/Abp/Localization/VirtualFiles/VirtualFileLocalizationResourceContributorBase.cs
index c42036d1cd..5667d7de4d 100644
--- a/framework/src/Volo.Abp.Localization/Volo/Abp/Localization/VirtualFiles/VirtualFileLocalizationResourceContributorBase.cs
+++ b/framework/src/Volo.Abp.Localization/Volo/Abp/Localization/VirtualFiles/VirtualFileLocalizationResourceContributorBase.cs
@@ -22,17 +22,17 @@ namespace Volo.Abp.Localization.VirtualFiles
_virtualPath = virtualPath;
}
- public void Initialize(LocalizationResourceInitializationContext context)
+ public virtual void Initialize(LocalizationResourceInitializationContext context)
{
_virtualFileProvider = context.ServiceProvider.GetRequiredService();
}
- public LocalizedString GetOrNull(string cultureName, string name)
+ public virtual LocalizedString GetOrNull(string cultureName, string name)
{
return GetDictionaries().GetOrDefault(cultureName)?.GetOrNull(name);
}
- public void Fill(string cultureName, Dictionary dictionary)
+ public virtual void Fill(string cultureName, Dictionary dictionary)
{
GetDictionaries().GetOrDefault(cultureName)?.Fill(dictionary);
}
@@ -105,4 +105,4 @@ namespace Volo.Abp.Localization.VirtualFiles
protected abstract ILocalizationDictionary CreateDictionaryFromFileContent(string fileContent);
}
-}
\ No newline at end of file
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Abstractions/FodyWeavers.xml b/framework/src/Volo.Abp.TextTemplating.Abstractions/FodyWeavers.xml
new file mode 100644
index 0000000000..be0de3a908
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Abstractions/FodyWeavers.xml
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.TextTemplating.Abstractions/FodyWeavers.xsd b/framework/src/Volo.Abp.TextTemplating.Abstractions/FodyWeavers.xsd
new file mode 100644
index 0000000000..3f3946e282
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Abstractions/FodyWeavers.xsd
@@ -0,0 +1,30 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 'true' to run assembly verification (PEVerify) on the target assembly after all weavers have been executed.
+
+
+
+
+ A comma-separated list of error codes that can be safely ignored in assembly verification.
+
+
+
+
+ 'false' to turn off automatic generation of the XML Schema file.
+
+
+
+
+
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo.Abp.TextTemplating.Abstractions.csproj b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo.Abp.TextTemplating.Abstractions.csproj
new file mode 100644
index 0000000000..649bdc2912
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo.Abp.TextTemplating.Abstractions.csproj
@@ -0,0 +1,16 @@
+
+
+
+
+
+
+ netstandard2.0
+
+
+
+
+
+
+
+
+
diff --git a/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/AbpTextTemplatingAbstractionsModule.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/AbpTextTemplatingAbstractionsModule.cs
new file mode 100644
index 0000000000..bb76b299b2
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/AbpTextTemplatingAbstractionsModule.cs
@@ -0,0 +1,46 @@
+using System;
+using System.Collections.Generic;
+using Microsoft.Extensions.DependencyInjection;
+using Volo.Abp.Localization;
+using Volo.Abp.Modularity;
+using Volo.Abp.VirtualFileSystem;
+
+namespace Volo.Abp.TextTemplating
+{
+ [DependsOn(
+ typeof(AbpVirtualFileSystemModule),
+ typeof(AbpLocalizationAbstractionsModule)
+ )]
+ public class AbpTextTemplatingAbstractionsModule : AbpModule
+ {
+ public override void PreConfigureServices(ServiceConfigurationContext context)
+ {
+ AutoAddProvidersAndContributors(context.Services);
+ }
+
+ private static void AutoAddProvidersAndContributors(IServiceCollection services)
+ {
+ var definitionProviders = new List();
+ var contentContributors = new List();
+
+ services.OnRegistred(context =>
+ {
+ if (typeof(ITemplateDefinitionProvider).IsAssignableFrom(context.ImplementationType))
+ {
+ definitionProviders.Add(context.ImplementationType);
+ }
+
+ if (typeof(ITemplateContentContributor).IsAssignableFrom(context.ImplementationType))
+ {
+ contentContributors.Add(context.ImplementationType);
+ }
+ });
+
+ services.Configure(options =>
+ {
+ options.DefinitionProviders.AddIfNotContains(definitionProviders);
+ options.ContentContributors.AddIfNotContains(contentContributors);
+ });
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/AbpTextTemplatingOptions.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/AbpTextTemplatingOptions.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/AbpTextTemplatingOptions.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/AbpTextTemplatingOptions.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateContentContributor.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateContentContributor.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateContentContributor.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateContentContributor.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateContentProvider.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateContentProvider.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateContentProvider.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateContentProvider.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateDefinitionContext.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateDefinitionContext.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateDefinitionContext.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateDefinitionContext.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateDefinitionManager.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateDefinitionManager.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateDefinitionManager.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateDefinitionManager.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateDefinitionProvider.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateDefinitionProvider.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateDefinitionProvider.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateDefinitionProvider.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateRenderer.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateRenderer.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/ITemplateRenderer.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/ITemplateRenderer.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateContentContributorContext.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateContentContributorContext.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateContentContributorContext.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateContentContributorContext.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateContentProvider.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateContentProvider.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateContentProvider.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateContentProvider.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinition.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinition.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinition.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinition.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionContext.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionContext.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionContext.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionContext.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionExtensions.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionExtensions.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionExtensions.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionExtensions.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionManager.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionManager.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionManager.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionManager.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionProvider.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionProvider.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateDefinitionProvider.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/TemplateDefinitionProvider.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/FileInfoLocalizedTemplateContentReader.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/FileInfoLocalizedTemplateContentReader.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/FileInfoLocalizedTemplateContentReader.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/FileInfoLocalizedTemplateContentReader.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReader.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReader.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReader.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReader.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReaderFactory.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReaderFactory.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReaderFactory.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/ILocalizedTemplateContentReaderFactory.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory.cs
similarity index 96%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory.cs
index 981d1f8819..cc189e1b3b 100644
--- a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory.cs
+++ b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory.cs
@@ -64,7 +64,8 @@ namespace Volo.Abp.TextTemplating.VirtualFiles
if (fileInfo.IsDirectory)
{
- var folderReader = new VirtualFolderLocalizedTemplateContentReader();
+ //TODO: Configure file extensions.
+ var folderReader = new VirtualFolderLocalizedTemplateContentReader(new[] {".tpl", ".cshtml"});
await folderReader.ReadContentsAsync(VirtualFileProvider, virtualPath);
return folderReader;
}
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/NullLocalizedTemplateContentReader.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/NullLocalizedTemplateContentReader.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/NullLocalizedTemplateContentReader.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/NullLocalizedTemplateContentReader.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContentContributor.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContentContributor.cs
similarity index 100%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContentContributor.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContentContributor.cs
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/VirtualFolderLocalizedTemplateContentReader.cs b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/VirtualFolderLocalizedTemplateContentReader.cs
similarity index 80%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/VirtualFolderLocalizedTemplateContentReader.cs
rename to framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/VirtualFolderLocalizedTemplateContentReader.cs
index 6fc7ff4847..27458d02cc 100644
--- a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/VirtualFiles/VirtualFolderLocalizedTemplateContentReader.cs
+++ b/framework/src/Volo.Abp.TextTemplating.Abstractions/Volo/Abp/TextTemplating/VirtualFiles/VirtualFolderLocalizedTemplateContentReader.cs
@@ -9,6 +9,12 @@ namespace Volo.Abp.TextTemplating.VirtualFiles
public class VirtualFolderLocalizedTemplateContentReader : ILocalizedTemplateContentReader
{
private Dictionary _dictionary;
+ private readonly string[] _fileExtension;
+
+ public VirtualFolderLocalizedTemplateContentReader(string[] fileExtension)
+ {
+ _fileExtension = fileExtension;
+ }
public async Task ReadContentsAsync(
IVirtualFileProvider virtualFileProvider,
@@ -29,7 +35,7 @@ namespace Volo.Abp.TextTemplating.VirtualFiles
continue;
}
- _dictionary.Add(file.Name.RemovePostFix(".tpl"), await file.ReadAsStringAsync());
+ _dictionary.Add(file.Name.RemovePostFix(_fileExtension), await file.ReadAsStringAsync());
}
}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/FodyWeavers.xml b/framework/src/Volo.Abp.TextTemplating.Razor/FodyWeavers.xml
new file mode 100644
index 0000000000..be0de3a908
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/FodyWeavers.xml
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/FodyWeavers.xsd b/framework/src/Volo.Abp.TextTemplating.Razor/FodyWeavers.xsd
new file mode 100644
index 0000000000..3f3946e282
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/FodyWeavers.xsd
@@ -0,0 +1,30 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 'true' to run assembly verification (PEVerify) on the target assembly after all weavers have been executed.
+
+
+
+
+ A comma-separated list of error codes that can be safely ignored in assembly verification.
+
+
+
+
+ 'false' to turn off automatic generation of the XML Schema file.
+
+
+
+
+
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo.Abp.TextTemplating.Razor.csproj b/framework/src/Volo.Abp.TextTemplating.Razor/Volo.Abp.TextTemplating.Razor.csproj
new file mode 100644
index 0000000000..f4d16e630c
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo.Abp.TextTemplating.Razor.csproj
@@ -0,0 +1,21 @@
+
+
+
+
+
+
+ netstandard2.0
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpCompiledViewProviderOptions.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpCompiledViewProviderOptions.cs
new file mode 100644
index 0000000000..6da3e8b443
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpCompiledViewProviderOptions.cs
@@ -0,0 +1,15 @@
+using System.Collections.Generic;
+using Microsoft.CodeAnalysis;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class AbpCompiledViewProviderOptions
+ {
+ public Dictionary> TemplateReferences { get; }
+
+ public AbpCompiledViewProviderOptions()
+ {
+ TemplateReferences = new Dictionary>();
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateCSharpCompiler.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateCSharpCompiler.cs
new file mode 100644
index 0000000000..55b0d10f17
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateCSharpCompiler.cs
@@ -0,0 +1,107 @@
+using System;
+using System.Collections.Generic;
+using System.Collections.Immutable;
+using System.IO;
+using System.Linq;
+using System.Reflection;
+using System.Text;
+using Microsoft.CodeAnalysis;
+using Microsoft.CodeAnalysis.CSharp;
+using Microsoft.Extensions.Options;
+using Volo.Abp.DependencyInjection;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class AbpRazorTemplateCSharpCompiler : ISingletonDependency
+ {
+ protected AbpRazorTemplateCSharpCompilerOptions Options { get; }
+
+ public AbpRazorTemplateCSharpCompiler(IOptions options)
+ {
+ Options = options.Value;
+ }
+
+ private static IEnumerable DefaultReferences => new List()
+ {
+ Assembly.Load("netstandard"),
+ Assembly.Load("System.Private.CoreLib"),
+ Assembly.Load("System.Runtime"),
+ Assembly.Load("System.Collections"),
+ Assembly.Load("System.ComponentModel"),
+ Assembly.Load("System.Linq"),
+ Assembly.Load("System.Linq.Expressions"),
+ Assembly.Load("Microsoft.Extensions.DependencyInjection"),
+ Assembly.Load("Microsoft.Extensions.DependencyInjection.Abstractions"),
+ Assembly.Load("Microsoft.Extensions.Localization"),
+ Assembly.Load("Microsoft.Extensions.Localization.Abstractions"),
+
+ typeof(AbpRazorTemplateCSharpCompiler).Assembly
+ }
+ .Select(x => MetadataReference.CreateFromFile(x.Location))
+ .ToImmutableList();
+
+ public virtual Stream CreateAssembly(string code, string assemblyName, List references = null, CSharpCompilationOptions options = null)
+ {
+ var defaultReferences = DefaultReferences.Concat(Options.References);
+ try
+ {
+ var compilation = CSharpCompilation.Create(
+ assemblyName,
+ syntaxTrees: new[] { CreateSyntaxTree(code) },
+ references: references != null ? defaultReferences.Concat(references) : defaultReferences,
+ options ?? GetCompilationOptions());
+
+ using (var memoryStream = new MemoryStream())
+ {
+ var result = compilation.Emit(memoryStream);
+
+ if (!result.Success)
+ {
+ var error = new StringBuilder();
+ error.AppendLine("Build failed");
+ foreach (var diagnostic in result.Diagnostics)
+ {
+ error.AppendLine(diagnostic.ToString());
+ }
+
+ throw new Exception(error.ToString());
+ }
+
+ memoryStream.Seek(0, SeekOrigin.Begin);
+ var assemblyStream = new MemoryStream();
+ memoryStream.CopyTo(assemblyStream);
+
+ return assemblyStream;
+ }
+ }
+ catch (Exception e)
+ {
+ var error = new StringBuilder();
+ error.AppendLine("CreateAssembly failed");
+ error.AppendLine(e.Message);
+ throw new Exception(error.ToString());
+ }
+ }
+
+ protected virtual SyntaxTree CreateSyntaxTree(string code)
+ {
+ return CSharpSyntaxTree.ParseText(code);
+ }
+
+ protected virtual CSharpCompilationOptions GetCompilationOptions()
+ {
+ var csharpCompilationOptions = new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary);
+
+ // Disable 1702 until roslyn turns this off by default
+ csharpCompilationOptions = csharpCompilationOptions.WithSpecificDiagnosticOptions(
+ new Dictionary
+ {
+ {"CS1701", ReportDiagnostic.Suppress}, // Binding redirects
+ {"CS1702", ReportDiagnostic.Suppress},
+ {"CS1705", ReportDiagnostic.Suppress}
+ });
+
+ return csharpCompilationOptions.WithOptimizationLevel(OptimizationLevel.Release);
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateCSharpCompilerOptions.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateCSharpCompilerOptions.cs
new file mode 100644
index 0000000000..20c1e391e0
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateCSharpCompilerOptions.cs
@@ -0,0 +1,15 @@
+using System.Collections.Generic;
+using Microsoft.CodeAnalysis;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class AbpRazorTemplateCSharpCompilerOptions
+ {
+ public List References { get; }
+
+ public AbpRazorTemplateCSharpCompilerOptions()
+ {
+ References = new List();
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateConsts.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateConsts.cs
new file mode 100644
index 0000000000..7701a89630
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpRazorTemplateConsts.cs
@@ -0,0 +1,9 @@
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public static class AbpRazorTemplateConsts
+ {
+ public const string DefaultNameSpace = "Abp.Razor";
+ public const string DefaultClassName = "Template";
+ public const string TypeName = DefaultNameSpace + "." + DefaultClassName;
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpTextTemplatingRazorModule.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpTextTemplatingRazorModule.cs
new file mode 100644
index 0000000000..318a356b93
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/AbpTextTemplatingRazorModule.cs
@@ -0,0 +1,12 @@
+using Volo.Abp.Modularity;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ [DependsOn(
+ typeof(AbpTextTemplatingAbstractionsModule)
+ )]
+ public class AbpTextTemplatingRazorModule : AbpModule
+ {
+
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/DefaultAbpCompiledViewProvider.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/DefaultAbpCompiledViewProvider.cs
new file mode 100644
index 0000000000..c743252456
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/DefaultAbpCompiledViewProvider.cs
@@ -0,0 +1,77 @@
+using System;
+using System.Collections.Concurrent;
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using System.Reflection;
+using System.Threading.Tasks;
+using Microsoft.AspNetCore.Razor.Language;
+using Microsoft.CodeAnalysis;
+using Microsoft.Extensions.Options;
+using Volo.Abp.DependencyInjection;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class DefaultAbpCompiledViewProvider : IAbpCompiledViewProvider, ITransientDependency
+ {
+ private static readonly ConcurrentDictionary CachedAssembles = new ConcurrentDictionary();
+
+ private readonly AbpCompiledViewProviderOptions _options;
+ private readonly AbpRazorTemplateCSharpCompiler _razorTemplateCSharpCompiler;
+ private readonly IAbpRazorProjectEngineFactory _razorProjectEngineFactory;
+ private readonly ITemplateContentProvider _templateContentProvider;
+
+ public DefaultAbpCompiledViewProvider(
+ IOptions options,
+ IAbpRazorProjectEngineFactory razorProjectEngineFactory,
+ AbpRazorTemplateCSharpCompiler razorTemplateCSharpCompiler,
+ ITemplateContentProvider templateContentProvider)
+ {
+ _options = options.Value;
+
+ _razorProjectEngineFactory = razorProjectEngineFactory;
+ _razorTemplateCSharpCompiler = razorTemplateCSharpCompiler;
+ _templateContentProvider = templateContentProvider;
+ }
+
+ public virtual async Task GetAssemblyAsync(TemplateDefinition templateDefinition)
+ {
+ async Task CreateAssembly(string content)
+ {
+ using (var assemblyStream = await GetAssemblyStreamAsync(templateDefinition, content))
+ {
+ return Assembly.Load(await assemblyStream.GetAllBytesAsync());
+ }
+ }
+
+ var templateContent = await _templateContentProvider.GetContentOrNullAsync(templateDefinition);
+ return CachedAssembles.GetOrAdd((templateDefinition.Name + templateContent).ToMd5(), await CreateAssembly(templateContent));
+ }
+
+ protected virtual async Task GetAssemblyStreamAsync(TemplateDefinition templateDefinition, string templateContent)
+ {
+ var razorProjectEngine = await _razorProjectEngineFactory.CreateAsync(builder =>
+ {
+ builder.SetNamespace(AbpRazorTemplateConsts.DefaultNameSpace);
+ builder.ConfigureClass((document, node) =>
+ {
+ node.ClassName = AbpRazorTemplateConsts.DefaultClassName;
+ });
+ });
+
+ var codeDocument = razorProjectEngine.Process(
+ RazorSourceDocument.Create(templateContent, templateDefinition.Name), null,
+ new List(), new List());
+
+ var cSharpDocument = codeDocument.GetCSharpDocument();
+
+ var templateReferences = _options.TemplateReferences
+ .GetOrDefault(templateDefinition.Name)
+ ?.Select(x => x)
+ .Cast()
+ .ToList();
+
+ return _razorTemplateCSharpCompiler.CreateAssembly(cSharpDocument.GeneratedCode, templateDefinition.Name, templateReferences);
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/DefaultAbpRazorProjectEngineFactory.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/DefaultAbpRazorProjectEngineFactory.cs
new file mode 100644
index 0000000000..eb6d851f49
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/DefaultAbpRazorProjectEngineFactory.cs
@@ -0,0 +1,20 @@
+using System;
+using System.Threading.Tasks;
+using Microsoft.AspNetCore.Razor.Language;
+using Volo.Abp.DependencyInjection;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class DefaultAbpRazorProjectEngineFactory : IAbpRazorProjectEngineFactory, ITransientDependency
+ {
+ public virtual async Task CreateAsync(Action configure = null)
+ {
+ return RazorProjectEngine.Create(await CreateRazorConfigurationAsync(), EmptyProjectFileSystem.Empty, configure);
+ }
+
+ protected virtual Task CreateRazorConfigurationAsync()
+ {
+ return Task.FromResult(RazorConfiguration.Default);
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/EmptyProjectFileSystem.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/EmptyProjectFileSystem.cs
new file mode 100644
index 0000000000..496e86e977
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/EmptyProjectFileSystem.cs
@@ -0,0 +1,30 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using Microsoft.AspNetCore.Razor.Language;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ internal class EmptyProjectFileSystem : RazorProjectFileSystem
+ {
+ internal static readonly RazorProjectFileSystem Empty = new EmptyProjectFileSystem();
+
+ public override IEnumerable EnumerateItems(string basePath)
+ {
+ NormalizeAndEnsureValidPath(basePath);
+ return Enumerable.Empty();
+ }
+
+ [Obsolete("Use GetItem(string path, string fileKind) instead.")]
+ public override RazorProjectItem GetItem(string path)
+ {
+ return GetItem(path, fileKind: null);
+ }
+
+ public override RazorProjectItem GetItem(string path, string fileKind)
+ {
+ NormalizeAndEnsureValidPath(path);
+ return new NotFoundProjectItem(string.Empty, path, fileKind);
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IAbpCompiledViewProvider.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IAbpCompiledViewProvider.cs
new file mode 100644
index 0000000000..ba5e96d2fa
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IAbpCompiledViewProvider.cs
@@ -0,0 +1,10 @@
+using System.Reflection;
+using System.Threading.Tasks;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public interface IAbpCompiledViewProvider
+ {
+ Task GetAssemblyAsync(TemplateDefinition templateDefinition);
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IAbpRazorProjectEngineFactory.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IAbpRazorProjectEngineFactory.cs
new file mode 100644
index 0000000000..c96b171f59
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IAbpRazorProjectEngineFactory.cs
@@ -0,0 +1,11 @@
+using System;
+using System.Threading.Tasks;
+using Microsoft.AspNetCore.Razor.Language;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public interface IAbpRazorProjectEngineFactory
+ {
+ Task CreateAsync(Action configure = null);
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IRazorTemplatePage.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IRazorTemplatePage.cs
new file mode 100644
index 0000000000..fc386c3353
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/IRazorTemplatePage.cs
@@ -0,0 +1,34 @@
+using System;
+using System.Collections.Generic;
+using System.Text.Encodings.Web;
+using System.Threading.Tasks;
+using Microsoft.Extensions.Localization;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public interface IRazorTemplatePage : IRazorTemplatePage
+ {
+ TModel Model { get; set; }
+ }
+
+ public interface IRazorTemplatePage
+ {
+ IServiceProvider ServiceProvider { get; set; }
+
+ IStringLocalizer Localizer { get; set; }
+
+ HtmlEncoder HtmlEncoder { get; set; }
+
+ JavaScriptEncoder JavaScriptEncoder { get; set; }
+
+ UrlEncoder UrlEncoder { get; set; }
+
+ Dictionary GlobalContext { get; set; }
+
+ string Body { get; set; }
+
+ Task ExecuteAsync();
+
+ Task GetOutputAsync();
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/NotFoundProjectItem.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/NotFoundProjectItem.cs
new file mode 100644
index 0000000000..735522bd09
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/NotFoundProjectItem.cs
@@ -0,0 +1,28 @@
+using System;
+using System.IO;
+using Microsoft.AspNetCore.Razor.Language;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ internal class NotFoundProjectItem : RazorProjectItem
+ {
+ public NotFoundProjectItem(string basePath, string path, string fileKind)
+ {
+ BasePath = basePath;
+ FilePath = path;
+ FileKind = fileKind ?? FileKinds.GetFileKindFromFilePath(path);
+ }
+
+ public override string BasePath { get; }
+
+ public override string FilePath { get; }
+
+ public override string FileKind { get; }
+
+ public override bool Exists => false;
+
+ public override string PhysicalPath => throw new NotSupportedException();
+
+ public override Stream Read() => throw new NotSupportedException();
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/RazorTemplatePageBase.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/RazorTemplatePageBase.cs
new file mode 100644
index 0000000000..9854775a25
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/RazorTemplatePageBase.cs
@@ -0,0 +1,177 @@
+using System;
+using System.Collections.Generic;
+using System.Text;
+using System.Text.Encodings.Web;
+using System.Threading.Tasks;
+using Microsoft.Extensions.Localization;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public abstract class RazorTemplatePageBase : RazorTemplatePageBase, IRazorTemplatePage
+ {
+ public TModel Model { get; set; }
+ }
+
+ public abstract class RazorTemplatePageBase : IRazorTemplatePage
+ {
+ public Dictionary GlobalContext { get; set; }
+
+ public IServiceProvider ServiceProvider { get; set; }
+
+ public IStringLocalizer Localizer { get; set; }
+
+ public HtmlEncoder HtmlEncoder { get; set; }
+
+ public JavaScriptEncoder JavaScriptEncoder { get; set; }
+
+ public UrlEncoder UrlEncoder { get; set; }
+
+ public string Body { get; set; }
+
+ private readonly StringBuilder _stringBuilder = new StringBuilder();
+
+ private AttributeInfo _attributeInfo;
+
+ public virtual void WriteLiteral(string literal = null)
+ {
+ _stringBuilder.Append(literal);
+ }
+
+ public virtual void Write(object obj = null)
+ {
+ _stringBuilder.Append(obj);
+ }
+
+ public virtual void BeginWriteAttribute(string name, string prefix, int prefixOffset, string suffix, int suffixOffset, int attributeValuesCount)
+ {
+ _attributeInfo = new AttributeInfo(name, prefix, prefixOffset, suffix, suffixOffset, attributeValuesCount);
+ WriteLiteral(prefix);
+ }
+
+ public virtual void WriteAttributeValue(string prefix, int prefixOffset, object value, int valueOffset, int valueLength, bool isLiteral)
+ {
+ if (_attributeInfo.AttributeValuesCount == 1)
+ {
+ if (IsBoolFalseOrNullValue(prefix, value))
+ {
+ // Value is either null or the bool 'false' with no prefix; don't render the attribute.
+ _attributeInfo.Suppressed = true;
+ return;
+ }
+
+ // We are not omitting the attribute. Write the prefix.
+ WriteLiteral(_attributeInfo.Prefix);
+
+ if (IsBoolTrueWithEmptyPrefixValue(prefix, value))
+ {
+ // The value is just the bool 'true', write the attribute name instead of the string 'True'.
+ value = _attributeInfo.Name;
+ }
+ }
+
+ // This block handles two cases.
+ // 1. Single value with prefix.
+ // 2. Multiple values with or without prefix.
+ if (value != null)
+ {
+ if (!string.IsNullOrEmpty(prefix))
+ {
+ WriteLiteral(prefix);
+ }
+
+ WriteUnprefixedAttributeValue(value, isLiteral);
+ }
+
+ _stringBuilder.Append(prefix);
+ _stringBuilder.Append(value);
+ }
+
+ public virtual void EndWriteAttribute()
+ {
+ if (!_attributeInfo.Suppressed)
+ {
+ WriteLiteral(_attributeInfo.Suffix);
+ }
+ }
+
+ public virtual Task ExecuteAsync()
+ {
+ return Task.CompletedTask;
+ }
+
+ public virtual Task GetOutputAsync()
+ {
+ return Task.FromResult(_stringBuilder.ToString());
+ }
+
+ private void WriteUnprefixedAttributeValue(object value, bool isLiteral)
+ {
+ var stringValue = value as string;
+
+ // The extra branching here is to ensure that we call the Write*To(string) overload where possible.
+ if (isLiteral && stringValue != null)
+ {
+ WriteLiteral(stringValue);
+ }
+ else if (isLiteral)
+ {
+ //WriteLiteral(value);
+ _stringBuilder.Append(value);
+ }
+ else if (stringValue != null)
+ {
+ Write(stringValue);
+ }
+ else
+ {
+ Write(value);
+ }
+ }
+
+ private bool IsBoolFalseOrNullValue(string prefix, object value)
+ {
+ return string.IsNullOrEmpty(prefix) && (value == null || (value is bool b && !b));
+ }
+
+ private bool IsBoolTrueWithEmptyPrefixValue(string prefix, object value)
+ {
+ // If the value is just the bool 'true', use the attribute name as the value.
+ return string.IsNullOrEmpty(prefix) && (value is bool b && b);
+ }
+
+ private struct AttributeInfo
+ {
+ public AttributeInfo(
+ string name,
+ string prefix,
+ int prefixOffset,
+ string suffix,
+ int suffixOffset,
+ int attributeValuesCount)
+ {
+ Name = name;
+ Prefix = prefix;
+ PrefixOffset = prefixOffset;
+ Suffix = suffix;
+ SuffixOffset = suffixOffset;
+ AttributeValuesCount = attributeValuesCount;
+
+ Suppressed = false;
+ }
+
+ public int AttributeValuesCount { get; }
+
+ public string Name { get; }
+
+ public string Prefix { get; }
+
+ public int PrefixOffset { get; }
+
+ public string Suffix { get; }
+
+ public int SuffixOffset { get; }
+
+ public bool Suppressed { get; set; }
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/RazorTemplateRenderer.cs b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/RazorTemplateRenderer.cs
new file mode 100644
index 0000000000..db06e67d4a
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Razor/Volo/Abp/TextTemplating/Razor/RazorTemplateRenderer.cs
@@ -0,0 +1,168 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Reflection;
+using System.Text.Encodings.Web;
+using System.Threading.Tasks;
+using JetBrains.Annotations;
+using Microsoft.Extensions.DependencyInjection;
+using Microsoft.Extensions.Localization;
+using Volo.Abp.DependencyInjection;
+using Volo.Abp.Localization;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ [Dependency(ReplaceServices = true)]
+ public class RazorTemplateRenderer : ITemplateRenderer, ITransientDependency
+ {
+ private readonly IServiceScopeFactory _serviceScopeFactory;
+ private readonly IAbpCompiledViewProvider _abpCompiledViewProvider;
+ private readonly ITemplateDefinitionManager _templateDefinitionManager;
+ private readonly IStringLocalizerFactory _stringLocalizerFactory;
+
+ public RazorTemplateRenderer(
+ IServiceScopeFactory serviceScopeFactory,
+ IAbpCompiledViewProvider abpCompiledViewProvider,
+ ITemplateDefinitionManager templateDefinitionManager,
+ IStringLocalizerFactory stringLocalizerFactory)
+ {
+ _serviceScopeFactory = serviceScopeFactory;
+ _templateDefinitionManager = templateDefinitionManager;
+ _stringLocalizerFactory = stringLocalizerFactory;
+ _abpCompiledViewProvider = abpCompiledViewProvider;
+ }
+
+ public virtual async Task RenderAsync(
+ [NotNull] string templateName,
+ [CanBeNull] object model = null,
+ [CanBeNull] string cultureName = null,
+ [CanBeNull] Dictionary globalContext = null)
+ {
+ Check.NotNullOrWhiteSpace(templateName, nameof(templateName));
+
+ if (globalContext == null)
+ {
+ globalContext = new Dictionary();
+ }
+
+ if (cultureName == null)
+ {
+ return await RenderInternalAsync(
+ templateName,
+ null,
+ globalContext,
+ model
+ );
+ }
+ else
+ {
+ using (CultureHelper.Use(cultureName))
+ {
+ return await RenderInternalAsync(
+ templateName,
+ null,
+ globalContext,
+ model
+ );
+ }
+ }
+ }
+
+ protected virtual async Task RenderInternalAsync(
+ string templateName,
+ string body,
+ Dictionary globalContext,
+ object model = null)
+ {
+ var templateDefinition = _templateDefinitionManager.Get(templateName);
+
+ var renderedContent = await RenderSingleTemplateAsync(
+ templateDefinition,
+ body,
+ globalContext,
+ model
+ );
+
+ if (templateDefinition.Layout != null)
+ {
+ renderedContent = await RenderInternalAsync(
+ templateDefinition.Layout,
+ renderedContent,
+ globalContext
+ );
+ }
+
+ return renderedContent;
+ }
+
+ protected virtual async Task RenderSingleTemplateAsync(
+ TemplateDefinition templateDefinition,
+ string body,
+ Dictionary globalContext,
+ object model = null)
+ {
+ return await RenderTemplateContentWithRazorAsync(
+ templateDefinition,
+ body,
+ globalContext,
+ model
+ );
+ }
+
+ protected virtual async Task RenderTemplateContentWithRazorAsync(
+ TemplateDefinition templateDefinition,
+ string body,
+ Dictionary globalContext,
+ object model = null)
+ {
+ var assembly = await _abpCompiledViewProvider.GetAssemblyAsync(templateDefinition);
+ var templateType = assembly.GetType(AbpRazorTemplateConsts.TypeName);
+ var template = (IRazorTemplatePage) Activator.CreateInstance(templateType);
+
+ using (var scope = _serviceScopeFactory.CreateScope())
+ {
+ var modelType = templateType
+ .GetInterfaces()
+ .Where(x => x.IsGenericType && x.GetGenericTypeDefinition() == typeof(IRazorTemplatePage<>))
+ .Select(x => x.GenericTypeArguments.FirstOrDefault())
+ .FirstOrDefault();
+
+ if (modelType != null)
+ {
+ GetType().GetMethod(nameof(SetModel), BindingFlags.Instance | BindingFlags.NonPublic)
+ ?.MakeGenericMethod(modelType).Invoke(this, new[] {template, model});
+ }
+
+ template.ServiceProvider = scope.ServiceProvider;
+ template.Localizer = GetLocalizerOrNull(templateDefinition);
+ template.HtmlEncoder = scope.ServiceProvider.GetService();
+ template.JavaScriptEncoder = scope.ServiceProvider.GetService();
+ template.UrlEncoder = scope.ServiceProvider.GetService();
+ template.Body = body;
+ template.GlobalContext = globalContext;
+
+ await template.ExecuteAsync();
+
+ return await template.GetOutputAsync();
+ }
+ }
+
+ private void SetModel(IRazorTemplatePage razorTemplatePage, object model = null)
+ {
+ if (razorTemplatePage is IRazorTemplatePage razorTemplatePageWithModel)
+ {
+ razorTemplatePageWithModel.Model = (TModel)model;
+ }
+ }
+
+ private IStringLocalizer GetLocalizerOrNull(TemplateDefinition templateDefinition)
+ {
+ if (templateDefinition.LocalizationResource != null)
+ {
+ return _stringLocalizerFactory.Create(templateDefinition.LocalizationResource);
+ }
+
+ return _stringLocalizerFactory.CreateDefaultOrNull();
+ }
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating.Scriban/FodyWeavers.xml b/framework/src/Volo.Abp.TextTemplating.Scriban/FodyWeavers.xml
new file mode 100644
index 0000000000..be0de3a908
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Scriban/FodyWeavers.xml
@@ -0,0 +1,3 @@
+
+
+
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.TextTemplating.Scriban/FodyWeavers.xsd b/framework/src/Volo.Abp.TextTemplating.Scriban/FodyWeavers.xsd
new file mode 100644
index 0000000000..3f3946e282
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Scriban/FodyWeavers.xsd
@@ -0,0 +1,30 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 'true' to run assembly verification (PEVerify) on the target assembly after all weavers have been executed.
+
+
+
+
+ A comma-separated list of error codes that can be safely ignored in assembly verification.
+
+
+
+
+ 'false' to turn off automatic generation of the XML Schema file.
+
+
+
+
+
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.TextTemplating.Scriban/Volo.Abp.TextTemplating.Scriban.csproj b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo.Abp.TextTemplating.Scriban.csproj
new file mode 100644
index 0000000000..b4b0d66789
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo.Abp.TextTemplating.Scriban.csproj
@@ -0,0 +1,19 @@
+
+
+
+
+
+
+ netstandard2.0
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/AbpTextTemplatingScribanModule.cs b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/AbpTextTemplatingScribanModule.cs
new file mode 100644
index 0000000000..789b9e376f
--- /dev/null
+++ b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/AbpTextTemplatingScribanModule.cs
@@ -0,0 +1,12 @@
+using Volo.Abp.Modularity;
+
+namespace Volo.Abp.TextTemplating.Scriban
+{
+ [DependsOn(
+ typeof(AbpTextTemplatingAbstractionsModule)
+ )]
+ public class AbpTextTemplatingScribanModule : AbpModule
+ {
+
+ }
+}
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateLocalizer.cs b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/ScribanTemplateLocalizer.cs
similarity index 91%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateLocalizer.cs
rename to framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/ScribanTemplateLocalizer.cs
index 4e4a1e4833..e280cd194a 100644
--- a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateLocalizer.cs
+++ b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/ScribanTemplateLocalizer.cs
@@ -7,13 +7,13 @@ using Scriban;
using Scriban.Runtime;
using Scriban.Syntax;
-namespace Volo.Abp.TextTemplating
+namespace Volo.Abp.TextTemplating.Scriban
{
- public class TemplateLocalizer : IScriptCustomFunction
+ public class ScribanTemplateLocalizer : IScriptCustomFunction
{
private readonly IStringLocalizer _localizer;
- public TemplateLocalizer(IStringLocalizer localizer)
+ public ScribanTemplateLocalizer(IStringLocalizer localizer)
{
_localizer = localizer;
}
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateRenderer.cs b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/ScribanTemplateRenderer.cs
similarity index 94%
rename from framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateRenderer.cs
rename to framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/ScribanTemplateRenderer.cs
index 1d89dc3b68..3fdd667a82 100644
--- a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/TemplateRenderer.cs
+++ b/framework/src/Volo.Abp.TextTemplating.Scriban/Volo/Abp/TextTemplating/Scriban/ScribanTemplateRenderer.cs
@@ -1,23 +1,21 @@
-using System;
-using System.Collections.Generic;
+using System.Collections.Generic;
using System.Threading.Tasks;
using JetBrains.Annotations;
using Microsoft.Extensions.Localization;
-using Microsoft.Extensions.Options;
using Scriban;
using Scriban.Runtime;
using Volo.Abp.DependencyInjection;
using Volo.Abp.Localization;
-namespace Volo.Abp.TextTemplating
+namespace Volo.Abp.TextTemplating.Scriban
{
- public class TemplateRenderer : ITemplateRenderer, ITransientDependency
+ public class ScribanTemplateRenderer : ITemplateRenderer, ITransientDependency
{
private readonly ITemplateContentProvider _templateContentProvider;
private readonly ITemplateDefinitionManager _templateDefinitionManager;
private readonly IStringLocalizerFactory _stringLocalizerFactory;
- public TemplateRenderer(
+ public ScribanTemplateRenderer(
ITemplateContentProvider templateContentProvider,
ITemplateDefinitionManager templateDefinitionManager,
IStringLocalizerFactory stringLocalizerFactory)
@@ -140,7 +138,7 @@ namespace Volo.Abp.TextTemplating
var localizer = GetLocalizerOrNull(templateDefinition);
if (localizer != null)
{
- scriptObject.SetValue("L", new TemplateLocalizer(localizer), true);
+ scriptObject.SetValue("L", new ScribanTemplateLocalizer(localizer), true);
}
context.PushGlobal(scriptObject);
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo.Abp.TextTemplating.csproj b/framework/src/Volo.Abp.TextTemplating/Volo.Abp.TextTemplating.csproj
index 506b086a88..3af0ba984d 100644
--- a/framework/src/Volo.Abp.TextTemplating/Volo.Abp.TextTemplating.csproj
+++ b/framework/src/Volo.Abp.TextTemplating/Volo.Abp.TextTemplating.csproj
@@ -5,22 +5,11 @@
netstandard2.0
- Volo.Abp.TextTemplating
- Volo.Abp.TextTemplating
- $(AssetTargetFallback);portable-net45+win8+wp8+wpa81;
- false
- false
- false
-
-
-
-
-
-
+
diff --git a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/AbpTextTemplatingModule.cs b/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/AbpTextTemplatingModule.cs
index 8d76391829..d5817d0d33 100644
--- a/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/AbpTextTemplatingModule.cs
+++ b/framework/src/Volo.Abp.TextTemplating/Volo/Abp/TextTemplating/AbpTextTemplatingModule.cs
@@ -1,46 +1,13 @@
-using System;
-using System.Collections.Generic;
-using Microsoft.Extensions.DependencyInjection;
-using Volo.Abp.Localization;
-using Volo.Abp.Modularity;
-using Volo.Abp.VirtualFileSystem;
+using Volo.Abp.Modularity;
+using Volo.Abp.TextTemplating.Scriban;
namespace Volo.Abp.TextTemplating
{
[DependsOn(
- typeof(AbpVirtualFileSystemModule),
- typeof(AbpLocalizationAbstractionsModule)
- )]
+ typeof(AbpTextTemplatingScribanModule)
+ )]
public class AbpTextTemplatingModule : AbpModule
{
- public override void PreConfigureServices(ServiceConfigurationContext context)
- {
- AutoAddProvidersAndContributors(context.Services);
- }
- private static void AutoAddProvidersAndContributors(IServiceCollection services)
- {
- var definitionProviders = new List();
- var contentContributors = new List();
-
- services.OnRegistred(context =>
- {
- if (typeof(ITemplateDefinitionProvider).IsAssignableFrom(context.ImplementationType))
- {
- definitionProviders.Add(context.ImplementationType);
- }
-
- if (typeof(ITemplateContentContributor).IsAssignableFrom(context.ImplementationType))
- {
- contentContributors.Add(context.ImplementationType);
- }
- });
-
- services.Configure(options =>
- {
- options.DefinitionProviders.AddIfNotContains(definitionProviders);
- options.ContentContributors.AddIfNotContains(contentContributors);
- });
- }
}
}
diff --git a/framework/src/Volo.Abp.Timing/Volo/Abp/Timing/Localization/fi.json b/framework/src/Volo.Abp.Timing/Volo/Abp/Timing/Localization/fi.json
new file mode 100644
index 0000000000..f7301d9c31
--- /dev/null
+++ b/framework/src/Volo.Abp.Timing/Volo/Abp/Timing/Localization/fi.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fi",
+ "texts": {
+ "DisplayName:Abp.Timing.Timezone": "Aikavyöhyke",
+ "Description:Abp.Timing.Timezone": "Levityksen aikavyöhyke"
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.UI.Navigation/Volo/Abp/Ui/Navigation/Localization/Resource/fi.json b/framework/src/Volo.Abp.UI.Navigation/Volo/Abp/Ui/Navigation/Localization/Resource/fi.json
new file mode 100644
index 0000000000..41a34208e5
--- /dev/null
+++ b/framework/src/Volo.Abp.UI.Navigation/Volo/Abp/Ui/Navigation/Localization/Resource/fi.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fi",
+ "texts": {
+ "Menu:Administration": "Hallinto"
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/fi.json b/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/fi.json
new file mode 100644
index 0000000000..844752e093
--- /dev/null
+++ b/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/fi.json
@@ -0,0 +1,52 @@
+{
+ "culture": "fi",
+ "texts": {
+ "Languages": "Kieli (kielet",
+ "AreYouSure": "Oletko varma?",
+ "Cancel": "Peruuttaa",
+ "Clear": "Asia selvä",
+ "Yes": "Joo",
+ "No": "Ei",
+ "Ok": "Ok",
+ "Close": "kiinni",
+ "Save": "Tallentaa",
+ "SavingWithThreeDot": "Tallentaa...",
+ "Actions": "Toiminnot",
+ "Delete": "Poistaa",
+ "Edit": "Muokata",
+ "Refresh": "virkistää",
+ "Language": "Kieli",
+ "LoadMore": "Lataa lisää",
+ "ProcessingWithThreeDot": "Käsitellään ...",
+ "LoadingWithThreeDot": "Ladataan...",
+ "Welcome": "Tervetuloa",
+ "Login": "Kirjaudu sisään",
+ "Register": "Rekisteröidy",
+ "Logout": "Kirjautua ulos",
+ "Submit": "Lähetä",
+ "Back": "Takaisin",
+ "PagerSearch": "Hae",
+ "PagerNext": "Seuraava",
+ "PagerPrevious": "Edellinen",
+ "PagerFirst": "Ensimmäinen",
+ "PagerLast": "Kestää",
+ "PagerInfo": "Näytetään _START_ - _END_ / _TOTAL_ merkinnästä",
+ "PagerInfo{0}{1}{2}": "Näytetään {0} - {1} / {2} merkinnästä",
+ "PagerInfoEmpty": "Näytetään 0 - 0/0 merkinnästä",
+ "PagerInfoFiltered": "(suodatettu _MAX_ merkinnän kokonaismäärästä)",
+ "NoDataAvailableInDatatable": "Tietoja ei ole käytettävissä",
+ "Total": "kaikki yhteensä",
+ "Selected": "valittu",
+ "PagerShowMenuEntries": "Näytä _MENU_ merkinnät",
+ "DatatableActionDropdownDefaultText": "Toiminnot",
+ "ChangePassword": "Vaihda salasana",
+ "PersonalInfo": "Profiilini",
+ "AreYouSureYouWantToCancelEditingWarningMessage": "Sinulla on tallentamattomia muutoksia.",
+ "GoHomePage": "Siirry kotisivulle",
+ "GoBack": "Mene takaisin",
+ "Search": "Hae",
+ "ItemWillBeDeletedMessageWithFormat": "{0} poistetaan!",
+ "ItemWillBeDeletedMessage": "Tämä kohde poistetaan!",
+ "ManageYourAccount": "Hallitse tiliäsi"
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/fr.json b/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/fr.json
index 8084270436..c27708ee09 100644
--- a/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/fr.json
+++ b/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/fr.json
@@ -46,6 +46,7 @@
"GoBack": "Retour",
"Search": "Recherche",
"ItemWillBeDeletedMessageWithFormat": "{0} sera supprimé!",
- "ItemWillBeDeletedMessage": "Cet objet va être supprimé!"
+ "ItemWillBeDeletedMessage": "Cet objet va être supprimé!",
+ "ManageYourAccount": "Gérer votre compte"
}
-}
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/nl.json b/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/nl.json
index 69ad662be1..a85e443bdf 100644
--- a/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/nl.json
+++ b/framework/src/Volo.Abp.UI/Localization/Resources/AbpUi/nl.json
@@ -7,7 +7,7 @@
"Clear": "Wissen",
"Yes": "Ja",
"No": "Nee",
- "Ok": "Ok",
+ "Ok": "Oké",
"Close": "Sluiten",
"Save": "Opslaan",
"SavingWithThreeDot": "Opslaan...",
diff --git a/framework/src/Volo.Abp.Validation/Volo/Abp/Validation/Localization/fi.json b/framework/src/Volo.Abp.Validation/Volo/Abp/Validation/Localization/fi.json
new file mode 100644
index 0000000000..f4501b87cf
--- /dev/null
+++ b/framework/src/Volo.Abp.Validation/Volo/Abp/Validation/Localization/fi.json
@@ -0,0 +1,34 @@
+{
+ "culture": "fi",
+ "texts": {
+ "'{0}' and '{1}' do not match.": "{0} ja {1} eivät täsmää.",
+ "The {0} field is not a valid credit card number.": "Kenttä {0} ei ole kelvollinen luottokortin numero.",
+ "{0} is not valid.": "{0} ei kelpaa.",
+ "The {0} field is not a valid e-mail address.": "{0} -kenttä ei ole kelvollinen sähköpostiosoite.",
+ "The {0} field only accepts files with the following extensions: {1}": "{0} -kenttä hyväksyy vain tiedostot, joilla on seuraavat laajennukset: {1}",
+ "The field {0} must be a string or array type with a maximum length of '{1}'.": "Kentän {0} on oltava merkkijono- tai taulukotyyppi, jonka enimmäispituus on {1}.",
+ "The field {0} must be a string or array type with a minimum length of '{1}'.": "Kentän {0} on oltava merkkijono tai matriisityyppi, jonka vähimmäispituus on {1}.",
+ "The {0} field is not a valid phone number.": "{0} -kenttä ei ole kelvollinen puhelinnumero.",
+ "The field {0} must be between {1} and {2}.": "Kentän {0} on oltava välillä {1} - {2}.",
+ "The field {0} must match the regular expression '{1}'.": "Kenttä {0} ei vastaa pyydettyä muotoa.",
+ "The {0} field is required.": "{0} -kenttä on pakollinen.",
+ "The field {0} must be a string with a maximum length of {1}.": "Kentän {0} on oltava merkkijono, jonka enimmäispituus on {1}.",
+ "The field {0} must be a string with a minimum length of {2} and a maximum length of {1}.": "Kentän {0} on oltava merkkijono, jonka vähimmäispituus on {2} ja enimmäispituus {1}.",
+ "The {0} field is not a valid fully-qualified http, https, or ftp URL.": "{0} -kenttä ei ole kelvollinen täysin hyväksytty http, https tai ftp URL.",
+ "The field {0} is invalid.": "Kenttä {0} on virheellinen.",
+ "ThisFieldIsNotAValidCreditCardNumber.": "Tämä kenttä ei ole kelvollinen luottokortin numero.",
+ "ThisFieldIsNotValid.": "Tämä kenttä ei kelpaa.",
+ "ThisFieldIsNotAValidEmailAddress.": "Tämä kenttä ei ole kelvollinen sähköpostiosoite.",
+ "ThisFieldOnlyAcceptsFilesWithTheFollowingExtensions:{0}": "Tämä kenttä hyväksyy vain tiedostot, joilla on seuraavat laajennukset: {0}",
+ "ThisFieldMustBeAStringOrArrayTypeWithAMaximumLengthOf{0}": "Tämän kentän on oltava merkkijono- tai taulukotyyppi, jonka enimmäispituus on {0}.",
+ "ThisFieldMustBeAStringOrArrayTypeWithAMinimumLengthOf{0}": "Tämän kentän on oltava merkkijono- tai taulukotyyppi, jonka vähimmäispituus on {0}.",
+ "ThisFieldIsNotAValidPhoneNumber.": "Tämä kenttä ei ole kelvollinen puhelinnumero.",
+ "ThisFieldMustBeBetween{0}And{1}": "Tämän kentän on oltava välillä {0} - {1}.",
+ "ThisFieldMustMatchTheRegularExpression{0}": "Tämän kentän on vastattava säännöllistä lauseketta {0}.",
+ "ThisFieldIsRequired.": "Tämä kenttä pitää täyttää.",
+ "ThisFieldMustBeAStringWithAMaximumLengthOf{0}": "Tämän kentän on oltava merkkijono, jonka enimmäispituus on {0}.",
+ "ThisFieldMustBeAStringWithAMinimumLengthOf{1}AndAMaximumLengthOf{0}": "Tämän kentän on oltava merkkijono, jonka vähimmäispituus on {1} ja enimmäispituus {0}.",
+ "ThisFieldIsNotAValidFullyQualifiedHttpHttpsOrFtpUrl": "Tämä kenttä ei ole kelvollinen täysin hyväksytty http, https tai ftp URL.",
+ "ThisFieldIsInvalid.": "Tämä kenttä on virheellinen."
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.AspNetCore.Mvc.Tests/Volo/Abp/AspNetCore/Mvc/Localization/Resource/fi.json b/framework/test/Volo.Abp.AspNetCore.Mvc.Tests/Volo/Abp/AspNetCore/Mvc/Localization/Resource/fi.json
new file mode 100644
index 0000000000..9e0ba9d1ed
--- /dev/null
+++ b/framework/test/Volo.Abp.AspNetCore.Mvc.Tests/Volo/Abp/AspNetCore/Mvc/Localization/Resource/fi.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fi",
+ "texts": {
+ "BirthDate": "Syntymäpäivä",
+ "Value1": "Arvo Yksi"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Emailing.Tests/Volo/Abp/Emailing/Localization/fi.json b/framework/test/Volo.Abp.Emailing.Tests/Volo/Abp/Emailing/Localization/fi.json
new file mode 100644
index 0000000000..5c1c8eac8d
--- /dev/null
+++ b/framework/test/Volo.Abp.Emailing.Tests/Volo/Abp/Emailing/Localization/fi.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fi",
+ "texts": {
+ "hello": "Hei"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/Localization/FR.json b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/Localization/FR.json
new file mode 100644
index 0000000000..b4da8701fe
--- /dev/null
+++ b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/Localization/FR.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fr",
+ "texts": {
+ "Volo.Abp.Http.DynamicProxying:10001": "Exception commerciale avec données: {0}"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/Localization/fi.json b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/Localization/fi.json
new file mode 100644
index 0000000000..2549056076
--- /dev/null
+++ b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/Localization/fi.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fi",
+ "texts": {
+ "Volo.Abp.Http.DynamicProxying:10001": "Yrityspoikkeus ja tiedot: {0}"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/CountryNames/FR.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/CountryNames/FR.json
new file mode 100644
index 0000000000..f1cc5293ca
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/CountryNames/FR.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fr",
+ "texts": {
+ "USA": "les états-unis d'Amérique",
+ "Brazil": "Brésil"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/CountryNames/fi.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/CountryNames/fi.json
new file mode 100644
index 0000000000..7a031f8c41
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/CountryNames/fi.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fi",
+ "texts": {
+ "USA": "Yhdysvallat",
+ "Brazil": "Brasilia"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/Validation/FR.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/Validation/FR.json
new file mode 100644
index 0000000000..32fac7bf02
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/Validation/FR.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fr",
+ "texts": {
+ "ThisFieldIsRequired": "Ce champ est requis",
+ "MaxLenghtErrorMessage": "Ce champ peut contenir au maximum \"{0}\" caractères"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/Validation/fi.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/Validation/fi.json
new file mode 100644
index 0000000000..b2d289d3c7
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Base/Validation/fi.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fi",
+ "texts": {
+ "ThisFieldIsRequired": "Tämä kenttä pitää täyttää",
+ "MaxLenghtErrorMessage": "Tämä kenttä voi olla enintään {0} merkkiä"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Source/FR.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Source/FR.json
new file mode 100644
index 0000000000..ab5f63ad78
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Source/FR.json
@@ -0,0 +1,11 @@
+{
+ "culture": "fr",
+ "texts": {
+ "Hello {0}.": "Bonjour {0} .",
+ "Car": "Voiture",
+ "CarPlural": "Voitures",
+ "MaxLenghtErrorMessage": "La longueur de ce champ peut être au maximum de \"{0}\" caractères",
+ "Universe": "Univers",
+ "FortyTwo": "Quarante-deux"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Source/fi.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Source/fi.json
new file mode 100644
index 0000000000..74eabcd747
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/Source/fi.json
@@ -0,0 +1,11 @@
+{
+ "culture": "fi",
+ "texts": {
+ "Hello {0}.": "Hei {0} .",
+ "Car": "Auto",
+ "CarPlural": "Autot",
+ "MaxLenghtErrorMessage": "Tämän kentän pituus voi olla enintään {0} merkkiä",
+ "Universe": "Maailmankaikkeus",
+ "FortyTwo": "Neljäkymmentäkaksi"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/SourceExt/FR.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/SourceExt/FR.json
new file mode 100644
index 0000000000..e74ca8ed74
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/SourceExt/FR.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fr",
+ "texts": {
+ "SeeYou": "À bientôt"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/SourceExt/fi.json b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/SourceExt/fi.json
new file mode 100644
index 0000000000..456c75dfa8
--- /dev/null
+++ b/framework/test/Volo.Abp.Localization.Tests/Volo/Abp/Localization/TestResources/SourceExt/fi.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fi",
+ "texts": {
+ "SeeYou": "Nähdään"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo.Abp.TextTemplating.Razor.Tests.csproj b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo.Abp.TextTemplating.Razor.Tests.csproj
new file mode 100644
index 0000000000..ca26b60a4b
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo.Abp.TextTemplating.Razor.Tests.csproj
@@ -0,0 +1,23 @@
+
+
+
+
+
+ net5.0
+
+
+
+
+
+ Always
+
+
+
+
+
+
+
+
+
+
+
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/AbpCompiledViewProviderOptions_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/AbpCompiledViewProviderOptions_Tests.cs
new file mode 100644
index 0000000000..9f4cd3ecaa
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/AbpCompiledViewProviderOptions_Tests.cs
@@ -0,0 +1,48 @@
+using System.Collections.Generic;
+using System.Linq;
+using System.Reflection;
+using System.Threading.Tasks;
+using Microsoft.CodeAnalysis;
+using Microsoft.Extensions.DependencyInjection;
+using Shouldly;
+using Volo.Abp.TextTemplating.Razor.SampleTemplates;
+using Xunit;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class AbpCompiledViewProviderOptions_Tests : TemplateDefinitionTests
+ {
+ private readonly IAbpCompiledViewProvider _compiledViewProvider;
+ private readonly ITemplateDefinitionManager _templateDefinitionManager;
+
+ public AbpCompiledViewProviderOptions_Tests()
+ {
+ _templateDefinitionManager = GetRequiredService();
+ _compiledViewProvider = GetRequiredService();
+ }
+
+ protected override void AfterAddApplication(IServiceCollection services)
+ {
+ services.Configure(options =>
+ {
+ options.TemplateReferences.Add(RazorTestTemplates.TestTemplate, new List()
+ {
+ Assembly.Load("Microsoft.Extensions.Logging.Abstractions")
+ }
+ .Select(x => MetadataReference.CreateFromFile(x.Location))
+ .ToList());
+ });
+ base.AfterAddApplication(services);
+ }
+
+ [Fact]
+ public async Task Custom_TemplateReferences_Test()
+ {
+ var templateDefinition = _templateDefinitionManager.GetOrNull(RazorTestTemplates.TestTemplate);
+
+ var assembly = await _compiledViewProvider.GetAssemblyAsync(templateDefinition);
+
+ assembly.GetReferencedAssemblies().ShouldContain(x => x.Name == "Microsoft.Extensions.Logging.Abstractions");
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorLocalizedTemplateContentReaderFactory_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorLocalizedTemplateContentReaderFactory_Tests.cs
new file mode 100644
index 0000000000..53eaf0fc80
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorLocalizedTemplateContentReaderFactory_Tests.cs
@@ -0,0 +1,26 @@
+using System;
+using System.IO;
+using Microsoft.Extensions.FileProviders;
+using Volo.Abp.TextTemplating.VirtualFiles;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class RazorLocalizedTemplateContentReaderFactory_Tests : LocalizedTemplateContentReaderFactory_Tests
+ {
+ public RazorLocalizedTemplateContentReaderFactory_Tests()
+ {
+ LocalizedTemplateContentReaderFactory = new LocalizedTemplateContentReaderFactory(
+ new PhysicalFileVirtualFileProvider(
+ new PhysicalFileProvider(Path.Combine(Directory.GetCurrentDirectory(),
+ "Volo", "Abp", "TextTemplating", "Razor"))));
+
+ WelcomeEmailEnglishContent = "@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase" +
+ Environment.NewLine +
+ "Welcome @Model.Name to the abp.io!";
+
+ WelcomeEmailTurkishContent = "@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase" +
+ Environment.NewLine +
+ "Merhaba @Model.Name, abp.io'ya hoşgeldiniz!";
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTemplateDefinitionTests.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTemplateDefinitionTests.cs
new file mode 100644
index 0000000000..53c40b5ea8
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTemplateDefinitionTests.cs
@@ -0,0 +1,7 @@
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class RazorTemplateDefinitionTests : TemplateDefinitionTests
+ {
+
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTemplateRenderer_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTemplateRenderer_Tests.cs
new file mode 100644
index 0000000000..07da572c1e
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTemplateRenderer_Tests.cs
@@ -0,0 +1,145 @@
+using System.Threading.Tasks;
+using Shouldly;
+using Xunit;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class RazorTemplateRenderer_Tests : AbpTextTemplatingTestBase
+ {
+ private readonly ITemplateRenderer _templateRenderer;
+
+ public RazorTemplateRenderer_Tests()
+ {
+ _templateRenderer = GetRequiredService();
+ }
+
+ [Fact]
+ public async Task Should_Get_Rendered_Localized_Template_Content_With_Different_Cultures()
+ {
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.WelcomeEmail,
+ model: new WelcomeEmailModel()
+ {
+ Name = "John"
+ },
+ cultureName: "en"
+ )).ShouldBe("Welcome John to the abp.io!");
+
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.WelcomeEmail,
+ model: new WelcomeEmailModel()
+ {
+ Name = "John"
+ },
+ cultureName: "tr"
+ )).ShouldBe("Merhaba John, abp.io'ya hoşgeldiniz!");
+
+ //"en-US" fallbacks to "en" since "en-US" doesn't exists and "en" is the fallback culture
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.WelcomeEmail,
+ model: new WelcomeEmailModel()
+ {
+ Name = "John"
+ },
+ cultureName: "en-US"
+ )).ShouldBe("Welcome John to the abp.io!");
+
+ //"fr" fallbacks to "en" since "fr" doesn't exists and "en" is the default culture
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.WelcomeEmail,
+ model: new WelcomeEmailModel()
+ {
+ Name = "John"
+ },
+ cultureName: "fr"
+ )).ShouldBe("Welcome John to the abp.io!");
+ }
+
+ [Fact]
+ public async Task Should_Get_Rendered_Localized_Template_Content_With_Stronly_Typed_Model()
+ {
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.WelcomeEmail,
+ model: new WelcomeEmailModel("John"),
+ cultureName: "en"
+ )).ShouldBe("Welcome John to the abp.io!");
+ }
+
+ [Fact]
+ public async Task Should_Get_Rendered_Inline_Localized_Template()
+ {
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.ForgotPasswordEmail,
+ new ForgotPasswordEmailModel("John"),
+ cultureName: "en"
+ )).ShouldBe("*BEGIN*Hello John, how are you?. Please click to the following link to get an email to reset your password!*END*");
+
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.ForgotPasswordEmail,
+ new ForgotPasswordEmailModel("John"),
+ cultureName: "tr"
+ )).ShouldBe("*BEGIN*Merhaba John, nasılsın?. Please click to the following link to get an email to reset your password!*END*");
+ }
+
+ [Fact]
+ public async Task Should_Get_Localized_Numbers()
+ {
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.ShowDecimalNumber,
+ new ShowDecimalNumberModel(123.45M),
+ cultureName: "en"
+ )).ShouldBe("*BEGIN*123.45*END*");
+
+ (await _templateRenderer.RenderAsync(
+ TestTemplates.ShowDecimalNumber,
+ new ShowDecimalNumberModel(123.45M),
+ cultureName: "de"
+ )).ShouldBe("*BEGIN*123,45*END*");
+ }
+
+ public class WelcomeEmailModel
+ {
+ public string Name { get; set; }
+
+ public WelcomeEmailModel()
+ {
+
+ }
+
+ public WelcomeEmailModel(string name)
+ {
+ Name = name;
+ }
+ }
+
+ public class ForgotPasswordEmailModel
+ {
+ public string Name { get; set; }
+
+ public ForgotPasswordEmailModel()
+ {
+
+ }
+
+ public ForgotPasswordEmailModel(string name)
+ {
+ Name = name;
+ }
+ }
+
+ public class ShowDecimalNumberModel
+ {
+ public decimal Amount { get; set; }
+
+ public ShowDecimalNumberModel()
+ {
+
+ }
+
+ public ShowDecimalNumberModel(decimal amount)
+ {
+ Amount = amount;
+ }
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTestTemplateDefinitionProvider.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTestTemplateDefinitionProvider.cs
new file mode 100644
index 0000000000..3bb1d7f831
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTestTemplateDefinitionProvider.cs
@@ -0,0 +1,24 @@
+using Volo.Abp.TextTemplating.Razor.SampleTemplates;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class RazorTestTemplateDefinitionProvider : TemplateDefinitionProvider
+ {
+ public override void Define(ITemplateDefinitionContext context)
+ {
+ context.GetOrNull(TestTemplates.WelcomeEmail)?
+ .WithVirtualFilePath("/SampleTemplates/WelcomeEmail", false);
+
+ context.GetOrNull(TestTemplates.ForgotPasswordEmail)?
+ .WithVirtualFilePath("/SampleTemplates/ForgotPasswordEmail.cshtml", true);
+
+ context.GetOrNull(TestTemplates.TestTemplateLayout1)?
+ .WithVirtualFilePath("/SampleTemplates/TestTemplateLayout1.cshtml", true);
+
+ context.GetOrNull(TestTemplates.ShowDecimalNumber)?
+ .WithVirtualFilePath("/SampleTemplates/ShowDecimalNumber.cshtml", true);
+
+ context.Add(new TemplateDefinition(RazorTestTemplates.TestTemplate).WithVirtualFilePath("/SampleTemplates/TestTemplate.cshtml", true));
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTextTemplatingTestModule.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTextTemplatingTestModule.cs
new file mode 100644
index 0000000000..df2f8b3b3d
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorTextTemplatingTestModule.cs
@@ -0,0 +1,26 @@
+using Microsoft.CodeAnalysis;
+using Volo.Abp.Modularity;
+using Volo.Abp.VirtualFileSystem;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ [DependsOn(
+ typeof(AbpTextTemplatingRazorModule),
+ typeof(AbpTextTemplatingTestModule)
+ )]
+ public class RazorTextTemplatingTestModule : AbpModule
+ {
+ public override void ConfigureServices(ServiceConfigurationContext context)
+ {
+ Configure(options =>
+ {
+ options.FileSets.AddEmbedded("Volo.Abp.TextTemplating.Razor");
+ });
+
+ Configure(options =>
+ {
+ options.References.Add(MetadataReference.CreateFromFile(typeof(RazorTextTemplatingTestModule).Assembly.Location));
+ });
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorVirtualFileTemplateContributor_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorVirtualFileTemplateContributor_Tests.cs
new file mode 100644
index 0000000000..5634ccfa2c
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/RazorVirtualFileTemplateContributor_Tests.cs
@@ -0,0 +1,23 @@
+using System;
+using Volo.Abp.TextTemplating.VirtualFiles;
+
+namespace Volo.Abp.TextTemplating.Razor
+{
+ public class RazorVirtualFileTemplateContributor_Tests : VirtualFileTemplateContributor_Tests
+ {
+ public RazorVirtualFileTemplateContributor_Tests()
+ {
+ WelcomeEmailEnglishContent = "@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase" +
+ Environment.NewLine +
+ "Welcome @Model.Name to the abp.io!";
+
+ WelcomeEmailTurkishContent = "@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase" +
+ Environment.NewLine +
+ "Merhaba @Model.Name, abp.io'ya hoşgeldiniz!";
+
+ ForgotPasswordEmailEnglishContent = "@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase" +
+ Environment.NewLine +
+ "@Localizer[\"HelloText\", Model.Name], @Localizer[\"HowAreYou\"]. Please click to the following link to get an email to reset your password!";
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/ForgotPasswordEmail.cshtml b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/ForgotPasswordEmail.cshtml
new file mode 100644
index 0000000000..e02bc728ab
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/ForgotPasswordEmail.cshtml
@@ -0,0 +1,2 @@
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+@Localizer["HelloText", Model.Name], @Localizer["HowAreYou"]. Please click to the following link to get an email to reset your password!
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/RazorTestTemplates.cs b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/RazorTestTemplates.cs
new file mode 100644
index 0000000000..3fa87f7d90
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/RazorTestTemplates.cs
@@ -0,0 +1,7 @@
+namespace Volo.Abp.TextTemplating.Razor.SampleTemplates
+{
+ public static class RazorTestTemplates
+ {
+ public const string TestTemplate = "TestTemplate";
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/ShowDecimalNumber.cshtml b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/ShowDecimalNumber.cshtml
new file mode 100644
index 0000000000..d309dc5a2b
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/ShowDecimalNumber.cshtml
@@ -0,0 +1,2 @@
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+@Model.Amount
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/TestTemplate.cshtml b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/TestTemplate.cshtml
new file mode 100644
index 0000000000..3de93e1e40
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/TestTemplate.cshtml
@@ -0,0 +1,6 @@
+@using Microsoft.Extensions.DependencyInjection
+@using Microsoft.Extensions.Logging
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+@{
+ var loggerFactory = ServiceProvider.GetService();
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/TestTemplateLayout1.cshtml b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/TestTemplateLayout1.cshtml
new file mode 100644
index 0000000000..4d3bc7b7c2
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/TestTemplateLayout1.cshtml
@@ -0,0 +1,2 @@
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+*BEGIN*@Body*END*
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/WelcomeEmail/en.cshtml b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/WelcomeEmail/en.cshtml
new file mode 100644
index 0000000000..410bfe1240
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/WelcomeEmail/en.cshtml
@@ -0,0 +1,2 @@
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+Welcome @Model.Name to the abp.io!
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/WelcomeEmail/tr.cshtml b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/WelcomeEmail/tr.cshtml
new file mode 100644
index 0000000000..7ea3c8802a
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Razor.Tests/Volo/Abp/TextTemplating/Razor/SampleTemplates/WelcomeEmail/tr.cshtml
@@ -0,0 +1,2 @@
+@inherits Volo.Abp.TextTemplating.Razor.RazorTemplatePageBase
+Merhaba @Model.Name, abp.io'ya hoşgeldiniz!
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo.Abp.TextTemplating.Scriban.Tests.csproj b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo.Abp.TextTemplating.Scriban.Tests.csproj
new file mode 100644
index 0000000000..d17a4d46fa
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo.Abp.TextTemplating.Scriban.Tests.csproj
@@ -0,0 +1,23 @@
+
+
+
+
+
+ net5.0
+
+
+
+
+
+ Always
+
+
+
+
+
+
+
+
+
+
+
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/ForgotPasswordEmail.tpl b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/ForgotPasswordEmail.tpl
similarity index 100%
rename from framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/ForgotPasswordEmail.tpl
rename to framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/ForgotPasswordEmail.tpl
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/ShowDecimalNumber.tpl b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/ShowDecimalNumber.tpl
similarity index 100%
rename from framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/ShowDecimalNumber.tpl
rename to framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/ShowDecimalNumber.tpl
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/TestTemplateLayout1.tpl b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/TestTemplateLayout1.tpl
similarity index 100%
rename from framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/TestTemplateLayout1.tpl
rename to framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/TestTemplateLayout1.tpl
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/WelcomeEmail/en.tpl b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/WelcomeEmail/en.tpl
similarity index 100%
rename from framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/WelcomeEmail/en.tpl
rename to framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/WelcomeEmail/en.tpl
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/WelcomeEmail/tr.tpl b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/WelcomeEmail/tr.tpl
similarity index 100%
rename from framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/SampleTemplates/WelcomeEmail/tr.tpl
rename to framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/SampleTemplates/WelcomeEmail/tr.tpl
diff --git a/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanLocalizedTemplateContentReaderFactory_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanLocalizedTemplateContentReaderFactory_Tests.cs
new file mode 100644
index 0000000000..1757bdaa17
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanLocalizedTemplateContentReaderFactory_Tests.cs
@@ -0,0 +1,20 @@
+using System.IO;
+using Microsoft.Extensions.FileProviders;
+using Volo.Abp.TextTemplating.VirtualFiles;
+
+namespace Volo.Abp.TextTemplating.Scriban
+{
+ public class ScribanLocalizedTemplateContentReaderFactory_Tests : LocalizedTemplateContentReaderFactory_Tests
+ {
+ public ScribanLocalizedTemplateContentReaderFactory_Tests()
+ {
+ LocalizedTemplateContentReaderFactory = new LocalizedTemplateContentReaderFactory(
+ new PhysicalFileVirtualFileProvider(
+ new PhysicalFileProvider(Path.Combine(Directory.GetCurrentDirectory(),
+ "Volo", "Abp", "TextTemplating", "Scriban"))));
+
+ WelcomeEmailEnglishContent = "Welcome {{model.name}} to the abp.io!";
+ WelcomeEmailTurkishContent = "Merhaba {{model.name}}, abp.io'ya hoşgeldiniz!";
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTemplateDefinitionTests.cs b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTemplateDefinitionTests.cs
new file mode 100644
index 0000000000..e454af36ef
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTemplateDefinitionTests.cs
@@ -0,0 +1,7 @@
+namespace Volo.Abp.TextTemplating.Scriban
+{
+ public class ScribanTemplateDefinitionTests : TemplateDefinitionTests
+ {
+
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TemplateRenderer_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTemplateRenderer_Tests.cs
similarity index 95%
rename from framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TemplateRenderer_Tests.cs
rename to framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTemplateRenderer_Tests.cs
index ca4ae3e3fd..a076301919 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TemplateRenderer_Tests.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTemplateRenderer_Tests.cs
@@ -3,13 +3,13 @@ using System.Threading.Tasks;
using Shouldly;
using Xunit;
-namespace Volo.Abp.TextTemplating
+namespace Volo.Abp.TextTemplating.Scriban
{
- public class TemplateRenderer_Tests : AbpTextTemplatingTestBase
+ public class ScribanTemplateRenderer_Tests : AbpTextTemplatingTestBase
{
private readonly ITemplateRenderer _templateRenderer;
- public TemplateRenderer_Tests()
+ public ScribanTemplateRenderer_Tests()
{
_templateRenderer = GetRequiredService();
}
@@ -91,7 +91,7 @@ namespace Volo.Abp.TextTemplating
cultureName: "tr"
)).ShouldBe("*BEGIN*Merhaba John, nasılsın?. Please click to the following link to get an email to reset your password!*END*");
}
-
+
[Fact]
public async Task Should_Get_Localized_Numbers()
{
diff --git a/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTestTemplateDefinitionProvider.cs b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTestTemplateDefinitionProvider.cs
new file mode 100644
index 0000000000..956c376fd0
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTestTemplateDefinitionProvider.cs
@@ -0,0 +1,20 @@
+namespace Volo.Abp.TextTemplating.Scriban
+{
+ public class ScribanTestTemplateDefinitionProvider : TemplateDefinitionProvider
+ {
+ public override void Define(ITemplateDefinitionContext context)
+ {
+ context.GetOrNull(TestTemplates.WelcomeEmail)?
+ .WithVirtualFilePath("/SampleTemplates/WelcomeEmail", false);
+
+ context.GetOrNull(TestTemplates.ForgotPasswordEmail)?
+ .WithVirtualFilePath("/SampleTemplates/ForgotPasswordEmail.tpl", true);
+
+ context.GetOrNull(TestTemplates.TestTemplateLayout1)?
+ .WithVirtualFilePath("/SampleTemplates/TestTemplateLayout1.tpl", true);
+
+ context.GetOrNull(TestTemplates.ShowDecimalNumber)?
+ .WithVirtualFilePath("/SampleTemplates/ShowDecimalNumber.tpl", true);
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTextTemplatingTestModule.cs b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTextTemplatingTestModule.cs
new file mode 100644
index 0000000000..2ed2e51b56
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanTextTemplatingTestModule.cs
@@ -0,0 +1,20 @@
+using Volo.Abp.Modularity;
+using Volo.Abp.VirtualFileSystem;
+
+namespace Volo.Abp.TextTemplating.Scriban
+{
+ [DependsOn(
+ typeof(AbpTextTemplatingScribanModule),
+ typeof(AbpTextTemplatingTestModule)
+ )]
+ public class ScribanTextTemplatingTestModule : AbpModule
+ {
+ public override void ConfigureServices(ServiceConfigurationContext context)
+ {
+ Configure(options =>
+ {
+ options.FileSets.AddEmbedded("Volo.Abp.TextTemplating.Scriban");
+ });
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanVirtualFileTemplateContributor_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanVirtualFileTemplateContributor_Tests.cs
new file mode 100644
index 0000000000..026b8abe59
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Scriban.Tests/Volo/Abp/TextTemplating/Scriban/ScribanVirtualFileTemplateContributor_Tests.cs
@@ -0,0 +1,14 @@
+using Volo.Abp.TextTemplating.VirtualFiles;
+
+namespace Volo.Abp.TextTemplating.Scriban
+{
+ public class ScribanVirtualFileTemplateContributor_Tests : VirtualFileTemplateContributor_Tests
+ {
+ public ScribanVirtualFileTemplateContributor_Tests()
+ {
+ WelcomeEmailEnglishContent = "Welcome {{model.name}} to the abp.io!";
+ WelcomeEmailTurkishContent = "Merhaba {{model.name}}, abp.io'ya hoşgeldiniz!";
+ ForgotPasswordEmailEnglishContent = "{{L \"HelloText\" model.name}}, {{L \"HowAreYou\" }}. Please click to the following link to get an email to reset your password!";
+ }
+ }
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo.Abp.TextTemplating.Tests.csproj b/framework/test/Volo.Abp.TextTemplating.Tests/Volo.Abp.TextTemplating.Tests.csproj
index 555c2934b0..65983605be 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo.Abp.TextTemplating.Tests.csproj
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo.Abp.TextTemplating.Tests.csproj
@@ -10,15 +10,12 @@
-
- Always
-
-
+
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingOptions_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingOptions_Tests.cs
index 294596c693..591f6c9caf 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingOptions_Tests.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingOptions_Tests.cs
@@ -4,7 +4,7 @@ using Xunit;
namespace Volo.Abp.TextTemplating
{
- public class AbpTextTemplatingOptions_Tests : AbpTextTemplatingTestBase
+ public class AbpTextTemplatingOptions_Tests : AbpTextTemplatingTestBase
{
private readonly AbpTextTemplatingOptions _options;
@@ -21,4 +21,4 @@ namespace Volo.Abp.TextTemplating
.ShouldContain(typeof(TestTemplateDefinitionProvider));
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestBase.cs b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestBase.cs
index ca5dc20445..24f8bad395 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestBase.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestBase.cs
@@ -1,8 +1,10 @@
-using Volo.Abp.Testing;
+using Volo.Abp.Modularity;
+using Volo.Abp.Testing;
namespace Volo.Abp.TextTemplating
{
- public abstract class AbpTextTemplatingTestBase : AbpIntegratedTest
+ public abstract class AbpTextTemplatingTestBase : AbpIntegratedTest
+ where TStartupModule : IAbpModule
{
protected override void SetAbpApplicationCreationOptions(AbpApplicationCreationOptions options)
{
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestModule.cs b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestModule.cs
index 8e8b69e116..74174ea690 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestModule.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/AbpTextTemplatingTestModule.cs
@@ -7,7 +7,7 @@ using Volo.Abp.VirtualFileSystem;
namespace Volo.Abp.TextTemplating
{
[DependsOn(
- typeof(AbpTextTemplatingModule),
+ typeof(AbpTextTemplatingAbstractionsModule),
typeof(AbpTestBaseModule),
typeof(AbpAutofacModule),
typeof(AbpLocalizationModule)
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/Localization/fi.json b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/Localization/fi.json
new file mode 100644
index 0000000000..a1938e1f84
--- /dev/null
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/Localization/fi.json
@@ -0,0 +1,7 @@
+{
+ "culture": "fi",
+ "texts": {
+ "HelloText": "Hei {0}",
+ "HowAreYou": "mitä kuuluu?"
+ }
+}
\ No newline at end of file
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TemplateDefinitionTests.cs b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TemplateDefinitionTests.cs
index 51c78cc13e..6fd97cd2d4 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TemplateDefinitionTests.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TemplateDefinitionTests.cs
@@ -1,25 +1,27 @@
using Shouldly;
+using Volo.Abp.Modularity;
using Xunit;
namespace Volo.Abp.TextTemplating
{
- public class TemplateDefinitionTests : AbpTextTemplatingTestBase
+ public abstract class TemplateDefinitionTests : AbpTextTemplatingTestBase
+ where TStartupModule : IAbpModule
{
- private readonly ITemplateDefinitionManager _templateDefinitionManager;
+ protected readonly ITemplateDefinitionManager TemplateDefinitionManager;
- public TemplateDefinitionTests()
+ protected TemplateDefinitionTests()
{
- _templateDefinitionManager = GetRequiredService();
+ TemplateDefinitionManager = GetRequiredService();
}
[Fact]
public void Should_Retrieve_Template_Definition_By_Name()
{
- var welcomeEmailTemplate = _templateDefinitionManager.Get(TestTemplates.WelcomeEmail);
+ var welcomeEmailTemplate = TemplateDefinitionManager.Get(TestTemplates.WelcomeEmail);
welcomeEmailTemplate.Name.ShouldBe(TestTemplates.WelcomeEmail);
welcomeEmailTemplate.IsInlineLocalized.ShouldBeFalse();
- var forgotPasswordEmailTemplate = _templateDefinitionManager.Get(TestTemplates.ForgotPasswordEmail);
+ var forgotPasswordEmailTemplate = TemplateDefinitionManager.Get(TestTemplates.ForgotPasswordEmail);
forgotPasswordEmailTemplate.Name.ShouldBe(TestTemplates.ForgotPasswordEmail);
forgotPasswordEmailTemplate.IsInlineLocalized.ShouldBeTrue();
}
@@ -27,15 +29,15 @@ namespace Volo.Abp.TextTemplating
[Fact]
public void Should_Get_Null_If_Template_Not_Found()
{
- var definition = _templateDefinitionManager.GetOrNull("undefined-template");
+ var definition = TemplateDefinitionManager.GetOrNull("undefined-template");
definition.ShouldBeNull();
}
[Fact]
public void Should_Retrieve_All_Template_Definitions()
{
- var definitions = _templateDefinitionManager.GetAll();
+ var definitions = TemplateDefinitionManager.GetAll();
definitions.Count.ShouldBeGreaterThan(1);
}
}
-}
\ No newline at end of file
+}
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TestTemplateDefinitionProvider.cs b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TestTemplateDefinitionProvider.cs
index 9db711d70c..289bf1b04f 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TestTemplateDefinitionProvider.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/TestTemplateDefinitionProvider.cs
@@ -10,7 +10,7 @@ namespace Volo.Abp.TextTemplating
new TemplateDefinition(
TestTemplates.WelcomeEmail,
defaultCultureName: "en"
- ).WithVirtualFilePath("/SampleTemplates/WelcomeEmail", false)
+ )
);
context.Add(
@@ -18,22 +18,22 @@ namespace Volo.Abp.TextTemplating
TestTemplates.ForgotPasswordEmail,
localizationResource: typeof(TestLocalizationSource),
layout: TestTemplates.TestTemplateLayout1
- ).WithVirtualFilePath("/SampleTemplates/ForgotPasswordEmail.tpl", true)
+ )
);
context.Add(
new TemplateDefinition(
TestTemplates.TestTemplateLayout1,
isLayout: true
- ).WithVirtualFilePath("/SampleTemplates/TestTemplateLayout1.tpl", true)
+ )
);
-
+
context.Add(
new TemplateDefinition(
TestTemplates.ShowDecimalNumber,
localizationResource: typeof(TestLocalizationSource),
layout: TestTemplates.TestTemplateLayout1
- ).WithVirtualFilePath("/SampleTemplates/ShowDecimalNumber.tpl", true)
+ )
);
}
}
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory_Tests.cs
index 63e526ac1d..7a56997031 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory_Tests.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/LocalizedTemplateContentReaderFactory_Tests.cs
@@ -1,37 +1,36 @@
-using System.IO;
-using System.Threading.Tasks;
+using System.Threading.Tasks;
using Microsoft.Extensions.FileProviders;
using Microsoft.Extensions.Primitives;
using Shouldly;
+using Volo.Abp.Modularity;
using Volo.Abp.VirtualFileSystem;
using Xunit;
namespace Volo.Abp.TextTemplating.VirtualFiles
{
- public class LocalizedTemplateContentReaderFactory_Tests: AbpTextTemplatingTestBase
+ public abstract class LocalizedTemplateContentReaderFactory_Tests : AbpTextTemplatingTestBase
+ where TStartupModule : IAbpModule
{
- private readonly ITemplateDefinitionManager _templateDefinitionManager;
+ protected readonly ITemplateDefinitionManager TemplateDefinitionManager;
+ protected LocalizedTemplateContentReaderFactory LocalizedTemplateContentReaderFactory;
+ protected string WelcomeEmailEnglishContent;
+ protected string WelcomeEmailTurkishContent;
- public LocalizedTemplateContentReaderFactory_Tests()
+ protected LocalizedTemplateContentReaderFactory_Tests()
{
- _templateDefinitionManager = GetRequiredService();
+ TemplateDefinitionManager = GetRequiredService();
}
[Fact]
public async Task Create_Should_Work_With_PhysicalFileProvider()
{
- var localizedTemplateContentReaderFactory = new LocalizedTemplateContentReaderFactory(
- new PhysicalFileVirtualFileProvider(
- new PhysicalFileProvider(Path.Combine(Directory.GetCurrentDirectory(),
- "Volo", "Abp", "TextTemplating"))));
+ var reader = await LocalizedTemplateContentReaderFactory.CreateAsync(TemplateDefinitionManager.Get(TestTemplates.WelcomeEmail));
- var reader = await localizedTemplateContentReaderFactory.CreateAsync(_templateDefinitionManager.Get(TestTemplates.WelcomeEmail));
-
- reader.GetContentOrNull("en").ShouldBe("Welcome {{model.name}} to the abp.io!");
- reader.GetContentOrNull("tr").ShouldBe("Merhaba {{model.name}}, abp.io'ya hoşgeldiniz!");
+ reader.GetContentOrNull("en").ShouldBe(WelcomeEmailEnglishContent);
+ reader.GetContentOrNull("tr").ShouldBe(WelcomeEmailTurkishContent);
}
- class PhysicalFileVirtualFileProvider : IVirtualFileProvider
+ public class PhysicalFileVirtualFileProvider : IVirtualFileProvider
{
private readonly PhysicalFileProvider _physicalFileProvider;
diff --git a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContributor_Tests.cs b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContributor_Tests.cs
index e5c7138118..4862e61203 100644
--- a/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContributor_Tests.cs
+++ b/framework/test/Volo.Abp.TextTemplating.Tests/Volo/Abp/TextTemplating/VirtualFiles/VirtualFileTemplateContributor_Tests.cs
@@ -1,45 +1,50 @@
using System.Threading.Tasks;
using Shouldly;
+using Volo.Abp.Modularity;
using Xunit;
namespace Volo.Abp.TextTemplating.VirtualFiles
{
- public class VirtualFileTemplateContributor_Tests : AbpTextTemplatingTestBase
+ public abstract class VirtualFileTemplateContributor_Tests : AbpTextTemplatingTestBase
+ where TStartupModule : IAbpModule
{
- private readonly ITemplateDefinitionManager _templateDefinitionManager;
- private readonly VirtualFileTemplateContentContributor _virtualFileTemplateContentContributor;
+ protected readonly ITemplateDefinitionManager TemplateDefinitionManager;
+ protected readonly VirtualFileTemplateContentContributor VirtualFileTemplateContentContributor;
+ protected string WelcomeEmailEnglishContent;
+ protected string WelcomeEmailTurkishContent;
+ protected string ForgotPasswordEmailEnglishContent;
- public VirtualFileTemplateContributor_Tests()
+ protected VirtualFileTemplateContributor_Tests()
{
- _templateDefinitionManager = GetRequiredService();
- _virtualFileTemplateContentContributor = GetRequiredService();
+ TemplateDefinitionManager = GetRequiredService();
+ VirtualFileTemplateContentContributor = GetRequiredService();
}
[Fact]
public async Task Should_Get_Localized_Content_By_Culture()
{
- (await _virtualFileTemplateContentContributor.GetOrNullAsync(
- new TemplateContentContributorContext(_templateDefinitionManager.Get(TestTemplates.WelcomeEmail),
+ (await VirtualFileTemplateContentContributor.GetOrNullAsync(
+ new TemplateContentContributorContext(TemplateDefinitionManager.Get(TestTemplates.WelcomeEmail),
ServiceProvider,
"en")))
- .ShouldBe("Welcome {{model.name}} to the abp.io!");
+ .ShouldBe(WelcomeEmailEnglishContent);
- (await _virtualFileTemplateContentContributor.GetOrNullAsync(
- new TemplateContentContributorContext(_templateDefinitionManager.Get(TestTemplates.WelcomeEmail),
+ (await VirtualFileTemplateContentContributor.GetOrNullAsync(
+ new TemplateContentContributorContext(TemplateDefinitionManager.Get(TestTemplates.WelcomeEmail),
ServiceProvider,
"tr")))
- .ShouldBe("Merhaba {{model.name}}, abp.io'ya hoşgeldiniz!");
+ .ShouldBe(WelcomeEmailTurkishContent);
}
[Fact]
public async Task Should_Get_Non_Localized_Template_Content()
{
- (await _virtualFileTemplateContentContributor.GetOrNullAsync(
+ (await VirtualFileTemplateContentContributor.GetOrNullAsync(
new TemplateContentContributorContext(
- _templateDefinitionManager.Get(TestTemplates.ForgotPasswordEmail),
+ TemplateDefinitionManager.Get(TestTemplates.ForgotPasswordEmail),
ServiceProvider,
null)))
- .ShouldBe("{{L \"HelloText\" model.name}}, {{L \"HowAreYou\" }}. Please click to the following link to get an email to reset your password!");
+ .ShouldBe(ForgotPasswordEmailEnglishContent);
}
}
}
diff --git a/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/fi.json b/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/fi.json
new file mode 100644
index 0000000000..9b4a2ff5df
--- /dev/null
+++ b/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/fi.json
@@ -0,0 +1,67 @@
+{
+ "culture": "fi",
+ "texts": {
+ "UserName": "Käyttäjätunnus",
+ "EmailAddress": "Sähköpostiosoite",
+ "UserNameOrEmailAddress": "Käyttäjänimi tai Sähköpostiosoite",
+ "Password": "Salasana",
+ "RememberMe": "Muista minut",
+ "UseAnotherServiceToLogin": "Käytä toista palvelua kirjautumiseen",
+ "UserLockedOutMessage": "Käyttäjätili on lukittu virheellisten kirjautumisyritysten vuoksi. Odota hetki ja yritä uudelleen.",
+ "InvalidUserNameOrPassword": "Väärä käyttäjänimi tai salasana!",
+ "LoginIsNotAllowed": "Et saa kirjautua sisään! Sinun on vahvistettava sähköpostiosoitteesi / puhelinnumerosi.",
+ "SelfRegistrationDisabledMessage": "Itserekisteröinti on poistettu käytöstä tälle sovellukselle. Rekisteröi uusi käyttäjä ottamalla yhteyttä sovelluksen järjestelmänvalvojaan.",
+ "LocalLoginDisabledMessage": "Paikallinen sisäänkirjautuminen on poistettu käytöstä tälle sovellukselle.",
+ "Login": "Kirjaudu sisään",
+ "Cancel": "Peruuttaa",
+ "Register": "Rekisteröidy",
+ "AreYouANewUser": "Oletko uusi käyttäjä?",
+ "AlreadyRegistered": "Jo rekisteröity?",
+ "InvalidLoginRequest": "Virheellinen kirjautumispyyntö",
+ "ThereAreNoLoginSchemesConfiguredForThisClient": "Tälle asiakkaalle ei ole määritetty kirjautumismalleja.",
+ "LogInUsingYourProviderAccount": "Kirjaudu sisään {0} -tililläsi",
+ "DisplayName:CurrentPassword": "Nykyinen salasana",
+ "DisplayName:NewPassword": "Uusi salasana",
+ "DisplayName:NewPasswordConfirm": "Vahvista uusi salasana",
+ "PasswordChangedMessage": "Salasanasi on vaihdettu onnistuneesti.",
+ "DisplayName:UserName": "Käyttäjätunnus",
+ "DisplayName:Email": "Sähköposti",
+ "DisplayName:Name": "Nimi",
+ "DisplayName:Surname": "Sukunimi",
+ "DisplayName:Password": "Salasana",
+ "DisplayName:EmailAddress": "Sähköpostiosoite",
+ "DisplayName:PhoneNumber": "Puhelinnumero",
+ "PersonalSettings": "Henkilökohtaiset asetukset",
+ "PersonalSettingsSaved": "Henkilökohtaiset asetukset tallennettu",
+ "PasswordChanged": "Salasana vaihdettu",
+ "NewPasswordConfirmFailed": "Vahvista uusi salasana.",
+ "Manage": "Hallitse",
+ "ManageYourProfile": "Hallitse profiiliasi",
+ "DisplayName:Abp.Account.IsSelfRegistrationEnabled": "Onko itserekisteröinti käytössä",
+ "Description:Abp.Account.IsSelfRegistrationEnabled": "Voiko käyttäjä rekisteröidä tilin itse.",
+ "DisplayName:Abp.Account.EnableLocalLogin": "Todennus paikallisella tilillä",
+ "Description:Abp.Account.EnableLocalLogin": "Ilmaisee, salliko palvelin käyttäjien todentamisen paikallisella tilillä.",
+ "LoggedOutTitle": "Kirjautui ulos",
+ "LoggedOutText": "Sinut on kirjautunut ulos ja sinut ohjataan pian.",
+ "ReturnToText": "Napsauta tätä ja ohjaa uudelleen osoitteeseen {0}",
+ "OrLoginWith": "Tai kirjaudu sisään:",
+ "ForgotPassword": "Unohtuiko salasana?",
+ "SendPasswordResetLink_Information": "Salasanan vaihtamislinkki lähetetään sähköpostiisi salasanan vaihtamiseksi. Jos et saa sähköpostia muutamassa minuutissa, yritä uudelleen.",
+ "PasswordResetMailSentMessage": "Tilin palauttamisen sähköposti lähetetään sähköpostiosoitteeseesi. Jos et näe tätä sähköpostia postilaatikossa 15 minuutin kuluessa, etsi se roskapostikansiostasi. Jos löydät sen sieltä, merkitse se nimellä -Ei roskaa.",
+ "ResetPassword": "Nollaa salasana",
+ "ConfirmPassword": "Vahvista (toista) salasana",
+ "ResetPassword_Information": "Anna uusi salasanasi.",
+ "YourPasswordIsSuccessfullyReset": "Salasanasi on palautettu.",
+ "GoToTheApplication": "Siirry sovellukseen",
+ "BackToLogin": "Takaisin sisäänkirjautumiseen",
+ "ProfileTab:Password": "Vaihda salasana",
+ "ProfileTab:PersonalInfo": "Henkilökohtaiset tiedot",
+ "ReturnToApplication": "Palaa sovellukseen",
+ "Volo.Account:InvalidEmailAddress": "Annettua sähköpostiosoitetta ei löydy: {0}",
+ "PasswordReset": "Salasanan nollaus",
+ "PasswordResetInfoInEmail": "Saimme tilin palautuspyynnön! Jos aloitit tämän pyynnön, voit nollata salasanasi napsauttamalla seuraavaa linkkiä.",
+ "ResetMyPassword": "Vaihda salasanani",
+ "AccessDenied": "Pääsy evätty!",
+ "AccessDeniedMessage": "Sinulla ei ole pääsyä tähän resurssiin."
+ }
+}
\ No newline at end of file
diff --git a/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/fr.json b/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/fr.json
index 1469c7619d..eb42bbaa0d 100644
--- a/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/fr.json
+++ b/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/fr.json
@@ -1,6 +1,6 @@
{
- "culture": "fr",
- "texts": {
+ "culture": "fr",
+ "texts": {
"UserName": "Nom d'utilisateur",
"EmailAddress": "Adresse e-mail",
"UserNameOrEmailAddress": "Nom d'utilisateur ou adresse e-mail",
@@ -11,50 +11,57 @@
"InvalidUserNameOrPassword": "Nom d'utilisateur ou mot de passe invalide!",
"LoginIsNotAllowed": "Vous n'êtes pas autorisé à vous connecter! Vous devez confirmer votre e-mail / numéro de téléphone.",
"SelfRegistrationDisabledMessage": "L'auto-inscription est désactivée pour cette application. Veuillez contacter l'administrateur de l'application pour enregistrer un nouvel utilisateur.",
- "LocalLoginDisabledMessage": "La connexion locale est désactivée pour cette application.",
- "Login": "Connectez-vous",
- "Cancel": "Annuler",
- "Register": "S'inscrire",
- "AreYouANewUser": "Êtes-vous un nouvel utilisateur?",
- "AlreadyRegistered": "Déjà inscrit?",
- "InvalidLoginRequest": "Demande de connexion non valide",
- "ThereAreNoLoginSchemesConfiguredForThisClient": "Aucun schéma de connexion n'est configuré pour ce client.",
- "LogInUsingYourProviderAccount": "Connectez-vous à l'aide de votre compte {0}",
- "DisplayName:CurrentPassword": "Mot de passe actuel",
- "DisplayName:NewPassword": "Nouveau mot de passe",
- "DisplayName:NewPasswordConfirm": "Confirmer le nouveau mot de passe",
- "PasswordChangedMessage": "Votre mot de passe a été changé avec succès.",
- "DisplayName:UserName": "Nom d'utilisateur",
- "DisplayName:Email": "Email",
- "DisplayName:Name": "Nom",
- "DisplayName:Surname": "Nom de famille",
- "DisplayName:Password": "Mot de passe",
- "DisplayName:EmailAddress": "Adresse e-mail",
- "DisplayName:PhoneNumber": "Numéro de téléphone",
- "PersonalSettings": "Paramètres personnels",
- "PersonalSettingsSaved": "Paramètres personnels enregistrés",
- "PasswordChanged": "Mot de passe changé",
- "NewPasswordConfirmFailed": "Veuillez confirmer le nouveau mot de passe.",
- "Manage": "Gérer",
- "ManageYourProfile": "Gérez votre profil",
- "DisplayName:Abp.Account.IsSelfRegistrationEnabled": "L’auto-inscription est-elle activée",
- "Description:Abp.Account.IsSelfRegistrationEnabled": "Si un utilisateur peut enregistrer le compte par lui-même.",
- "DisplayName:Abp.Account.EnableLocalLogin": "Authentifier avec un compte local",
- "Description:Abp.Account.EnableLocalLogin": "Indique si le serveur permettra aux utilisateurs de s’authentifier avec un compte local.",
- "LoggedOutTitle": "Se déconnecter",
- "LoggedOutText": "Vous avez été déconnecté et vous serez bientôt redirigé.",
- "ReturnToText": "Cliquez ici pour rediriger vers {0}",
- "OrLoginWith": "Ou connectez-vous avec:",
- "ForgotPassword": "Vous avez oublié le mot de passe ?",
- "SendPasswordResetLink_Information": "Un lien de réinitialisation du mot de passe sera envoyé à votre adresse e-mail pour réinitialiser votre mot de passe. Si vous ne recevez pas d'e-mail dans quelques minutes, veuillez réessayer.",
- "PasswordResetMailSentMessage": "E-mail de récupération de compte envoyé à votre adresse e-mail. Si vous ne voyez pas cet e-mail dans votre boîte de réception dans les 15 minutes, recherchez-le dans votre dossier de courrier indésirable. Si vous le trouvez là-bas, veuillez le marquer comme -Pas indésirable-.",
- "ResetPassword": "Réinitialiser le mot de passe",
- "ConfirmPassword": "Confirmer (répéter) le mot de passe",
- "ResetPassword_Information": "Entrez votre nouveau mot de passe.",
- "YourPasswordIsSuccessfullyReset": "Votre mot de passe a été réinitialisé avec succès.",
- "GoToTheApplication": "Accédez à l’application",
- "BackToLogin": "Retour à la connexion",
- "ProfileTab:Password": "Changer le mot de passe",
- "ProfileTab:PersonalInfo": "Informations personnelles"
- }
-}
+ "LocalLoginDisabledMessage": "La connexion locale est désactivée pour cette application.",
+ "Login": "Connectez-vous",
+ "Cancel": "Annuler",
+ "Register": "S'inscrire",
+ "AreYouANewUser": "Êtes-vous un nouvel utilisateur?",
+ "AlreadyRegistered": "Déjà inscrit?",
+ "InvalidLoginRequest": "Demande de connexion non valide",
+ "ThereAreNoLoginSchemesConfiguredForThisClient": "Aucun schéma de connexion n'est configuré pour ce client.",
+ "LogInUsingYourProviderAccount": "Connectez-vous à l'aide de votre compte {0}",
+ "DisplayName:CurrentPassword": "Mot de passe actuel",
+ "DisplayName:NewPassword": "Nouveau mot de passe",
+ "DisplayName:NewPasswordConfirm": "Confirmer le nouveau mot de passe",
+ "PasswordChangedMessage": "Votre mot de passe a été changé avec succès.",
+ "DisplayName:UserName": "Nom d'utilisateur",
+ "DisplayName:Email": "Email",
+ "DisplayName:Name": "Nom",
+ "DisplayName:Surname": "Nom de famille",
+ "DisplayName:Password": "Mot de passe",
+ "DisplayName:EmailAddress": "Adresse e-mail",
+ "DisplayName:PhoneNumber": "Numéro de téléphone",
+ "PersonalSettings": "Paramètres personnels",
+ "PersonalSettingsSaved": "Paramètres personnels enregistrés",
+ "PasswordChanged": "Mot de passe changé",
+ "NewPasswordConfirmFailed": "Veuillez confirmer le nouveau mot de passe.",
+ "Manage": "Gérer",
+ "ManageYourProfile": "Gérez votre profil",
+ "DisplayName:Abp.Account.IsSelfRegistrationEnabled": "L’auto-inscription est-elle activée",
+ "Description:Abp.Account.IsSelfRegistrationEnabled": "Si un utilisateur peut enregistrer le compte par lui-même.",
+ "DisplayName:Abp.Account.EnableLocalLogin": "Authentifier avec un compte local",
+ "Description:Abp.Account.EnableLocalLogin": "Indique si le serveur permettra aux utilisateurs de s’authentifier avec un compte local.",
+ "LoggedOutTitle": "Se déconnecter",
+ "LoggedOutText": "Vous avez été déconnecté et vous serez bientôt redirigé.",
+ "ReturnToText": "Cliquez ici pour rediriger vers {0}",
+ "OrLoginWith": "Ou connectez-vous avec:",
+ "ForgotPassword": "Vous avez oublié le mot de passe ?",
+ "SendPasswordResetLink_Information": "Un lien de réinitialisation du mot de passe sera envoyé à votre adresse e-mail pour réinitialiser votre mot de passe. Si vous ne recevez pas d'e-mail dans quelques minutes, veuillez réessayer.",
+ "PasswordResetMailSentMessage": "E-mail de récupération de compte envoyé à votre adresse e-mail. Si vous ne voyez pas cet e-mail dans votre boîte de réception dans les 15 minutes, recherchez-le dans votre dossier de courrier indésirable. Si vous le trouvez là-bas, veuillez le marquer comme -Pas indésirable-.",
+ "ResetPassword": "Réinitialiser le mot de passe",
+ "ConfirmPassword": "Confirmer (répéter) le mot de passe",
+ "ResetPassword_Information": "Entrez votre nouveau mot de passe.",
+ "YourPasswordIsSuccessfullyReset": "Votre mot de passe a été réinitialisé avec succès.",
+ "GoToTheApplication": "Accédez à l’application",
+ "BackToLogin": "Retour à la connexion",
+ "ProfileTab:Password": "Changer le mot de passe",
+ "ProfileTab:PersonalInfo": "Informations personnelles",
+ "ReturnToApplication": "Revenir à l'application",
+ "Volo.Account:InvalidEmailAddress": "Impossible de trouver l'adresse e-mail indiquée: {0}",
+ "PasswordReset": "Réinitialisation du mot de passe",
+ "PasswordResetInfoInEmail": "Nous avons reçu une demande de récupération de compte! Si vous avez lancé cette demande, cliquez sur le lien suivant pour réinitialiser votre mot de passe.",
+ "ResetMyPassword": "Réinitialiser mon mot de passe",
+ "AccessDenied": "Accès refusé!",
+ "AccessDeniedMessage": "Vous n'avez pas accès à cette ressource."
+ }
+}
\ No newline at end of file
diff --git a/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/nl.json b/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/nl.json
index efd751859e..7b17f1076b 100644
--- a/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/nl.json
+++ b/modules/account/src/Volo.Abp.Account.Application.Contracts/Volo/Abp/Account/Localization/Resources/nl.json
@@ -60,6 +60,8 @@
"Volo.Account:InvalidEmailAddress": "Kan het opgegeven e-mailadres '{0}' niet vinden",
"PasswordReset": "Wachtwoord opnieuw instellen",
"PasswordResetInfoInEmail": "We hebben een verzoek ontvangen om uw wachtwoord opnieuw in te stellen. Als u dit verzoek heeft ingediend, klikt u op de volgende link om een nieuw wachtwoord in te stellen.",
- "ResetMyPassword": "Reset mijn wachtwoord"
+ "ResetMyPassword": "Reset mijn wachtwoord",
+ "AccessDenied": "Toegang geweigerd!",
+ "AccessDeniedMessage": "U heeft geen toegang tot deze bron."
}
}
\ No newline at end of file
diff --git a/modules/account/src/Volo.Abp.Account.Web.IdentityServer/Pages/Account/IdentityServerSupportedLoginModel.cs b/modules/account/src/Volo.Abp.Account.Web.IdentityServer/Pages/Account/IdentityServerSupportedLoginModel.cs
index 4dab1f9761..2bdadc6ebd 100644
--- a/modules/account/src/Volo.Abp.Account.Web.IdentityServer/Pages/Account/IdentityServerSupportedLoginModel.cs
+++ b/modules/account/src/Volo.Abp.Account.Web.IdentityServer/Pages/Account/IdentityServerSupportedLoginModel.cs
@@ -189,46 +189,32 @@ namespace Volo.Abp.Account.Web.Pages.Account
return await base.OnPostExternalLogin(provider);
}
- private async Task ProcessWindowsLoginAsync()
+ protected virtual async Task ProcessWindowsLoginAsync()
{
var result = await HttpContext.AuthenticateAsync(AccountOptions.WindowsAuthenticationSchemeName);
- if (!(result?.Principal is WindowsPrincipal windowsPrincipal))
+ if (result.Succeeded)
{
- return Challenge(AccountOptions.WindowsAuthenticationSchemeName);
- }
-
- var props = new AuthenticationProperties
- {
- RedirectUri = Url.Page("./Login", pageHandler: "ExternalLoginCallback", values: new { ReturnUrl, ReturnUrlHash }),
- Items =
+ var props = new AuthenticationProperties()
{
- {"scheme", AccountOptions.WindowsAuthenticationSchemeName},
- }
- };
-
- var identity = new ClaimsIdentity(AccountOptions.WindowsAuthenticationSchemeName);
- identity.AddClaim(new Claim(JwtClaimTypes.Subject, windowsPrincipal.Identity.Name));
- identity.AddClaim(new Claim(JwtClaimTypes.Name, windowsPrincipal.Identity.Name));
-
- //TODO: Consider to add Windows groups the the identity
- //if (_accountOptions.IncludeWindowsGroups)
- //{
- // var windowsIdentity = windowsPrincipal.Identity as WindowsIdentity;
- // if (windowsIdentity != null)
- // {
- // var groups = windowsIdentity.Groups?.Translate(typeof(NTAccount));
- // var roles = groups.Select(x => new Claim(JwtClaimTypes.Role, x.Value));
- // identity.AddClaims(roles);
- // }
- //}
-
- await HttpContext.SignInAsync(
- IdentityServer4.IdentityServerConstants.ExternalCookieAuthenticationScheme,
- new ClaimsPrincipal(identity),
- props
- );
+ RedirectUri = Url.Page("./Login", pageHandler: "ExternalLoginCallback", values: new {ReturnUrl, ReturnUrlHash}),
+ Items =
+ {
+ {
+ "LoginProvider", AccountOptions.WindowsAuthenticationSchemeName
+ },
+ }
+ };
+
+ var id = new ClaimsIdentity(AccountOptions.WindowsAuthenticationSchemeName);
+ id.AddClaim(new Claim(ClaimTypes.NameIdentifier, result.Principal.FindFirstValue(ClaimTypes.PrimarySid)));
+ id.AddClaim(new Claim(ClaimTypes.Name, result.Principal.FindFirstValue(ClaimTypes.Name)));
+
+ await HttpContext.SignInAsync(IdentityConstants.ExternalScheme, new ClaimsPrincipal(id), props);
+
+ return Redirect(props.RedirectUri);
+ }
- return RedirectSafely(props.RedirectUri);
+ return Challenge(AccountOptions.WindowsAuthenticationSchemeName);
}
}
}
diff --git a/modules/blob-storing-database/src/Volo.Abp.BlobStoring.Database.Domain.Shared/Volo/Abp/BlobStoring/Database/Localization/fi.json b/modules/blob-storing-database/src/Volo.Abp.BlobStoring.Database.Domain.Shared/Volo/Abp/BlobStoring/Database/Localization/fi.json
new file mode 100644
index 0000000000..b2bf5f1fc3
--- /dev/null
+++ b/modules/blob-storing-database/src/Volo.Abp.BlobStoring.Database.Domain.Shared/Volo/Abp/BlobStoring/Database/Localization/fi.json
@@ -0,0 +1,6 @@
+{
+ "culture": "fi",
+ "texts": {
+ "ManageYourProfile": "Hallitse profiiliasi"
+ }
+}
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/BloggingTestAppModule.cs b/modules/blogging/app/Volo.BloggingTestApp/BloggingTestAppModule.cs
index 2a31e4aa74..09bb9ca36b 100644
--- a/modules/blogging/app/Volo.BloggingTestApp/BloggingTestAppModule.cs
+++ b/modules/blogging/app/Volo.BloggingTestApp/BloggingTestAppModule.cs
@@ -165,6 +165,7 @@ namespace Volo.BloggingTestApp
});
app.UseAuthentication();
+ app.UseAuthorization();
app.UseAbpRequestLocalization();
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ar.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ar.js
new file mode 100644
index 0000000000..fe036163c7
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ar.js
@@ -0,0 +1,171 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 1);
+/******/ })
+/************************************************************************/
+/******/ ([
+/* 0 */
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+/* 1 */
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Arabic
+ * @author Amira Salah
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage('ar', {
+ Markdown: 'لغة ترميز',
+ WYSIWYG: 'ما تراه هو ما تحصل عليه',
+ Write: 'يكتب',
+ Preview: 'عرض مسبق',
+ Headings: 'العناوين',
+ Paragraph: 'فقرة',
+ Bold: 'خط عريض',
+ Italic: 'خط مائل',
+ Strike: 'إضراب',
+ Code: 'رمز',
+ Line: 'خط',
+ Blockquote: 'فقرة مقتبسة',
+ 'Unordered list': 'قائمة غير مرتبة',
+ 'Ordered list': 'قائمة مرتبة',
+ Task: 'مهمة',
+ Indent: 'المسافة البادئة',
+ Outdent: 'المسافة الخارجة',
+ 'Insert link': 'أدخل الرابط',
+ 'Insert CodeBlock': 'أدخل الكود',
+ 'Insert table': 'أدخل جدول',
+ 'Insert image': 'أدخل صورة',
+ Heading: 'عنوان',
+ 'Image URL': 'رابط الصورة',
+ 'Select image file': 'حدد ملف الصورة',
+ Description: 'وصف',
+ OK: 'موافقة',
+ More: 'أكثر',
+ Cancel: 'إلغاء',
+ File: 'ملف',
+ URL: 'رابط',
+ 'Link text': 'نص الرابط',
+ 'Add row': 'ضف سطر',
+ 'Add col': 'ضف عمود',
+ 'Remove row': 'حذف سطر',
+ 'Remove col': 'حذف عمود',
+ 'Align left': 'محاذاة اليسار',
+ 'Align center': 'محاذاة الوسط',
+ 'Align right': 'محاذاة اليمين',
+ 'Remove table': 'حذف الجدول',
+ 'Would you like to paste as table?': 'هل تريد اللصق كجدول',
+ 'Text color': 'لون النص',
+ 'Auto scroll enabled': 'التحريك التلقائي ممكّن',
+ 'Auto scroll disabled': 'التحريك التلقائي معطّل',
+ 'Choose language': 'اختر اللغة'
+});
+
+/***/ })
+/******/ ]);
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/cs-cz.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/cs-cz.js
new file mode 100644
index 0000000000..03056d947c
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/cs-cz.js
@@ -0,0 +1,172 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 2);
+/******/ })
+/************************************************************************/
+/******/ ([
+/* 0 */
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+/* 1 */,
+/* 2 */
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Czech
+ * @author Dmitrij Tkačenko
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['cs', 'cs-CZ'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Napsat',
+ Preview: 'Náhled',
+ Headings: 'Nadpisy',
+ Paragraph: 'Odstavec',
+ Bold: 'Tučné',
+ Italic: 'Kurzíva',
+ Strike: 'Přeškrtnuté',
+ Code: 'Kód',
+ Line: 'Vodorovná čára',
+ Blockquote: 'Citace',
+ 'Unordered list': 'Seznam s odrážkami',
+ 'Ordered list': 'Číslovaný seznam',
+ Task: 'Úkol',
+ Indent: 'Zvětšit odsazení',
+ Outdent: 'Zmenšit odsazení',
+ 'Insert link': 'Vložit odkaz',
+ 'Insert CodeBlock': 'Vložit blok kódu',
+ 'Insert table': 'Vložit tabulku',
+ 'Insert image': 'Vložit obrázek',
+ Heading: 'Nadpis',
+ 'Image URL': 'URL obrázku',
+ 'Select image file': 'Vybrat obrázek',
+ Description: 'Popis',
+ OK: 'OK',
+ More: 'Více',
+ Cancel: 'Zrušit',
+ File: 'Soubor',
+ URL: 'URL',
+ 'Link text': 'Text odkazu',
+ 'Add row': 'Přidat řádek',
+ 'Add col': 'Přidat sloupec',
+ 'Remove row': 'Odebrat řádek',
+ 'Remove col': 'Odebrat sloupec',
+ 'Align left': 'Zarovnat vlevo',
+ 'Align center': 'Zarovnat na střed',
+ 'Align right': 'Zarovnat vpravo',
+ 'Remove table': 'Odstranit tabulku',
+ 'Would you like to paste as table?': 'Chcete vložit jako tabulku?',
+ 'Text color': 'Barva textu',
+ 'Auto scroll enabled': 'Automatické rolování zapnuto',
+ 'Auto scroll disabled': 'Automatické rolování vypnuto',
+ 'Choose language': 'Vybrat jazyk'
+});
+
+/***/ })
+/******/ ]);
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/de-de.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/de-de.js
new file mode 100644
index 0000000000..5f325f3af5
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/de-de.js
@@ -0,0 +1,173 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 3);
+/******/ })
+/************************************************************************/
+/******/ ([
+/* 0 */
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+/* 1 */,
+/* 2 */,
+/* 3 */
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for German
+ * @author Jann-Niklas Kiepert
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['de', 'de-DE'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Verfassen',
+ Preview: 'Vorschau',
+ Headings: 'Überschriften',
+ Paragraph: 'Text',
+ Bold: 'Fett',
+ Italic: 'Kursiv',
+ Strike: 'Durchgestrichen',
+ Code: 'Code',
+ Line: 'Trennlinie',
+ Blockquote: 'Blocktext',
+ 'Unordered list': 'Aufzählung',
+ 'Ordered list': 'Nummerierte Aufzählung',
+ Task: 'Aufgabe',
+ Indent: 'Einrücken',
+ Outdent: 'Ausrücken',
+ 'Insert link': 'Link einfügen',
+ 'Insert CodeBlock': 'Codeblock einfügen',
+ 'Insert table': 'Tabelle einfügen',
+ 'Insert image': 'Grafik einfügen',
+ Heading: 'Titel',
+ 'Image URL': 'Bild URL',
+ 'Select image file': 'Grafik auswählen',
+ Description: 'Beschreibung',
+ OK: 'OK',
+ More: 'Mehr',
+ Cancel: 'Abbrechen',
+ File: 'Datei',
+ URL: 'URL',
+ 'Link text': 'Anzuzeigender Text',
+ 'Add row': 'Zeile hinzufügen',
+ 'Add col': 'Spalte hinzufügen',
+ 'Remove row': 'Zeile entfernen',
+ 'Remove col': 'Spalte entfernen',
+ 'Align left': 'Links ausrichten',
+ 'Align center': 'Zentrieren',
+ 'Align right': 'Rechts ausrichten',
+ 'Remove table': 'Tabelle entfernen',
+ 'Would you like to paste as table?': 'Möchten Sie eine Tabelle einfügen?',
+ 'Text color': 'Textfarbe',
+ 'Auto scroll enabled': 'Autoscrollen aktiviert',
+ 'Auto scroll disabled': 'Autoscrollen deaktiviert',
+ 'Choose language': 'Sprache auswählen'
+});
+
+/***/ })
+/******/ ]);
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/es-es.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/es-es.js
new file mode 100644
index 0000000000..eb76415e73
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/es-es.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 4);
+/******/ })
+/************************************************************************/
+/******/ ([
+/* 0 */
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+/* 1 */,
+/* 2 */,
+/* 3 */,
+/* 4 */
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Spanish
+ * @author Enrico Lamperti
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['es', 'es-ES'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Escribir',
+ Preview: 'Vista previa',
+ Headings: 'Encabezados',
+ Paragraph: 'Párrafo',
+ Bold: 'Negrita',
+ Italic: 'Itálica',
+ Strike: 'Tachado',
+ Code: 'Código',
+ Line: 'Línea',
+ Blockquote: 'Cita',
+ 'Unordered list': 'Lista desordenada',
+ 'Ordered list': 'Lista ordenada',
+ Task: 'Tarea',
+ Indent: 'Sangría',
+ Outdent: 'Saliendo',
+ 'Insert link': 'Insertar enlace',
+ 'Insert CodeBlock': 'Insertar bloque de código',
+ 'Insert table': 'Insertar tabla',
+ 'Insert image': 'Insertar imagen',
+ Heading: 'Encabezado',
+ 'Image URL': 'URL de la imagen',
+ 'Select image file': 'Seleccionar archivo de imagen',
+ Description: 'Descripción',
+ OK: 'Aceptar',
+ More: 'Más',
+ Cancel: 'Cancelar',
+ File: 'Archivo',
+ URL: 'URL',
+ 'Link text': 'Texto del enlace',
+ 'Add row': 'Agregar fila',
+ 'Add col': 'Agregar columna',
+ 'Remove row': 'Eliminar fila',
+ 'Remove col': 'Eliminar columna',
+ 'Align left': 'Alinear a la izquierda',
+ 'Align center': 'Centrar',
+ 'Align right': 'Alinear a la derecha',
+ 'Remove table': 'Eliminar tabla',
+ 'Would you like to paste as table?': '¿Desea pegar como tabla?',
+ 'Text color': 'Color del texto',
+ 'Auto scroll enabled': 'Desplazamiento automático habilitado',
+ 'Auto scroll disabled': 'Desplazamiento automático deshabilitado',
+ 'Choose language': 'Elegir idioma'
+});
+
+/***/ })
+/******/ ]);
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/fi-fi.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/fi-fi.js
new file mode 100644
index 0000000000..89b02e3d55
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/fi-fi.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 5);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 5:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Finnish
+ * @author Tomi Mynttinen
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['fi', 'fi-FI'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Kirjoita',
+ Preview: 'Esikatselu',
+ Headings: 'Otsikot',
+ Paragraph: 'Kappale',
+ Bold: 'Lihavointi',
+ Italic: 'Kursivointi',
+ Strike: 'Yliviivaus',
+ Code: 'Koodi',
+ Line: 'Vaakaviiva',
+ Blockquote: 'Lainaus',
+ 'Unordered list': 'Luettelo',
+ 'Ordered list': 'Numeroitu luettelo',
+ Task: 'Tehtävä',
+ Indent: 'Suurenna sisennystä',
+ Outdent: 'Pienennä sisennystä',
+ 'Insert link': 'Lisää linkki',
+ 'Insert CodeBlock': 'Lisää koodia',
+ 'Insert table': 'Lisää taulukko',
+ 'Insert image': 'Lisää kuva',
+ Heading: 'Otsikko',
+ 'Image URL': 'Kuvan URL',
+ 'Select image file': 'Valitse kuvatiedosto',
+ Description: 'Kuvaus',
+ OK: 'OK',
+ More: 'Lisää',
+ Cancel: 'Peruuta',
+ File: 'Tiedosto',
+ URL: 'URL',
+ 'Link text': 'Linkkiteksti',
+ 'Add row': 'Lisää rivi',
+ 'Add col': 'Lisää sarake',
+ 'Remove row': 'Poista rivi',
+ 'Remove col': 'Poista sarake',
+ 'Align left': 'Tasaus vasemmalle',
+ 'Align center': 'Keskitä',
+ 'Align right': 'Tasaus oikealle',
+ 'Remove table': 'Poista taulukko',
+ 'Would you like to paste as table?': 'Haluatko liittää taulukkomuodossa?',
+ 'Text color': 'Tekstin väri',
+ 'Auto scroll enabled': 'Automaattinen skrollaus käytössä',
+ 'Auto scroll disabled': 'Automaattinen skrollaus pois käytöstä',
+ 'Choose language': 'Valitse kieli'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/fr-fr.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/fr-fr.js
new file mode 100644
index 0000000000..59177f18d0
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/fr-fr.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 6);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 6:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for French
+ * @author Stanislas Michalak
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['fr', 'fr-FR'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Écrire',
+ Preview: 'Aperçu',
+ Headings: 'En-têtes',
+ Paragraph: 'Paragraphe',
+ Bold: 'Gras',
+ Italic: 'Italique',
+ Strike: 'Barré',
+ Code: 'Code en ligne',
+ Line: 'Ligne',
+ Blockquote: 'Citation',
+ 'Unordered list': 'Liste non-ordonnée',
+ 'Ordered list': 'Liste ordonnée',
+ Task: 'Tâche',
+ Indent: 'Retrait',
+ Outdent: 'Sortir',
+ 'Insert link': 'Insérer un lien',
+ 'Insert CodeBlock': 'Insérer un bloc de code',
+ 'Insert table': 'Insérer un tableau',
+ 'Insert image': 'Insérer une image',
+ Heading: 'En-tête',
+ 'Image URL': "URL de l'image",
+ 'Select image file': 'Sélectionnez un fichier image',
+ Description: 'Description',
+ OK: 'OK',
+ More: 'de plus',
+ Cancel: 'Annuler',
+ File: 'Fichier',
+ URL: 'URL',
+ 'Link text': 'Texte du lien',
+ 'Add row': 'Ajouter une ligne',
+ 'Add col': 'Ajouter une colonne',
+ 'Remove row': 'Supprimer une ligne',
+ 'Remove col': 'Supprimer une colonne',
+ 'Align left': 'Aligner à gauche',
+ 'Align center': 'Aligner au centre',
+ 'Align right': 'Aligner à droite',
+ 'Remove table': 'Supprimer le tableau',
+ 'Would you like to paste as table?': 'Voulez-vous coller ce contenu en tant que tableau ?',
+ 'Text color': 'Couleur du texte',
+ 'Auto scroll enabled': 'Défilement automatique activé',
+ 'Auto scroll disabled': 'Défilement automatique désactivé',
+ 'Choose language': 'Choix de la langue'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/gl-es.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/gl-es.js
new file mode 100644
index 0000000000..ad0bdc156f
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/gl-es.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 7);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 7:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Spanish
+ * @author Aida Vidal
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['gl', 'gl-ES'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Escribir',
+ Preview: 'Vista previa',
+ Headings: 'Encabezados',
+ Paragraph: 'Parágrafo',
+ Bold: 'Negriña',
+ Italic: 'Cursiva',
+ Strike: 'Riscado',
+ Code: 'Código',
+ Line: 'Liña',
+ Blockquote: 'Cita',
+ 'Unordered list': 'Lista desordenada',
+ 'Ordered list': 'Lista ordenada',
+ Task: 'Tarefa',
+ Indent: 'Sangría',
+ Outdent: 'Anular sangría',
+ 'Insert link': 'Inserir enlace',
+ 'Insert CodeBlock': 'Inserir bloque de código',
+ 'Insert table': 'Inserir táboa',
+ 'Insert image': 'Inserir imaxe',
+ Heading: 'Encabezado',
+ 'Image URL': 'URL da imaxe',
+ 'Select image file': 'Seleccionar arquivo da imaxe',
+ Description: 'Descrición',
+ OK: 'Aceptar',
+ More: 'Máis',
+ Cancel: 'Cancelar',
+ File: 'Arquivo',
+ URL: 'URL',
+ 'Link text': 'Texto do enlace',
+ 'Add row': 'Agregar fila',
+ 'Add col': 'Agregar columna',
+ 'Remove row': 'Eliminar fila',
+ 'Remove col': 'Eliminar columna',
+ 'Align left': 'Aliñar á esquerda',
+ 'Align center': 'Centrar',
+ 'Align right': 'Aliñar á dereita',
+ 'Remove table': 'Eliminar táboa',
+ 'Would you like to paste as table?': 'Desexa pegar como táboa?',
+ 'Text color': 'Cor do texto',
+ 'Auto scroll enabled': 'Desprazamento automático habilitado',
+ 'Auto scroll disabled': 'Desprazamento automático deshabilitado',
+ 'Choose language': 'Elixir idioma'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/hr-hr.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/hr-hr.js
new file mode 100644
index 0000000000..d934dd03cf
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/hr-hr.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 8);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 8:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Croatian
+ * @author Hrvoje A.
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['hr', 'hr-HR'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Piši',
+ Preview: 'Pregled',
+ Headings: 'Naslovi',
+ Paragraph: 'Paragraf',
+ Bold: 'podebljano',
+ Italic: 'kurziv',
+ Strike: 'prcrtano',
+ Code: 'Uklopljeni kôd',
+ Line: 'Linija',
+ Blockquote: 'Blok citat',
+ 'Unordered list': 'Neporedana lista',
+ 'Ordered list': 'Poredana lista',
+ Task: 'Task',
+ Indent: 'Povećaj uvlaku',
+ Outdent: 'Smanji uvlaku',
+ 'Insert link': 'Umetni link',
+ 'Insert CodeBlock': 'Umetni blok kôda',
+ 'Insert table': 'Umetni tablicu',
+ 'Insert image': 'Umetni sliku',
+ Heading: 'Naslov',
+ 'Image URL': 'URL slike',
+ 'Select image file': 'Odaberi slikovnu datoteku',
+ Description: 'Opis',
+ OK: 'OK',
+ More: 'Više',
+ Cancel: 'Odustani',
+ File: 'Datoteka',
+ URL: 'URL',
+ 'Link text': 'Tekst linka',
+ 'Add row': 'Dodaj redak',
+ 'Add col': 'Dodaj stupac',
+ 'Remove row': 'Ukloni redak',
+ 'Remove col': 'Remove stupac',
+ 'Align left': 'Poravnaj lijevo',
+ 'Align center': 'Poravnaj centrirano',
+ 'Align right': 'Poravnaj desno',
+ 'Remove table': 'Ukloni tablicu',
+ 'Would you like to paste as table?': 'Zalite li zalijepiti kao tablicu?',
+ 'Text color': 'Boja teksta',
+ 'Auto scroll enabled': 'Omogući auto klizanje',
+ 'Auto scroll disabled': 'Onemogući auto klizanje',
+ 'Choose language': 'Odabir jezika'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/it-it.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/it-it.js
new file mode 100644
index 0000000000..a73129c92f
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/it-it.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 9);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 9:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Italian
+ * @author Massimo Redaelli
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['it', 'it-IT'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Scrivere',
+ Preview: 'Anteprima',
+ Headings: 'Intestazioni',
+ Paragraph: 'Paragrafo',
+ Bold: 'Grassetto',
+ Italic: 'Corsivo',
+ Strike: 'Barrato',
+ Code: 'Codice',
+ Line: 'Linea',
+ Blockquote: 'Blocco citazione',
+ 'Unordered list': 'Lista puntata',
+ 'Ordered list': 'Lista numerata',
+ Task: 'Attività',
+ Indent: 'Aggiungi indentazione',
+ Outdent: 'Rimuovi indentazione',
+ 'Insert link': 'Inserisci link',
+ 'Insert CodeBlock': 'Inserisci blocco di codice',
+ 'Insert table': 'Inserisci tabella',
+ 'Insert image': 'Inserisci immagine',
+ Heading: 'Intestazione',
+ 'Image URL': 'URL immagine',
+ 'Select image file': 'Seleziona file immagine',
+ Description: 'Descrizione',
+ OK: 'OK',
+ More: 'Più',
+ Cancel: 'Cancella',
+ File: 'File',
+ URL: 'URL',
+ 'Link text': 'Testo del collegamento',
+ 'Add row': 'Aggiungi riga',
+ 'Add col': 'Aggiungi colonna',
+ 'Remove row': 'Rimuovi riga',
+ 'Remove col': 'Rimuovi colonna',
+ 'Align left': 'Allinea a sinistra',
+ 'Align center': 'Allinea al centro',
+ 'Align right': 'Allinea a destra',
+ 'Remove table': 'Rimuovi tabella',
+ 'Would you like to paste as table?': 'Desideri incollare sotto forma di tabella?',
+ 'Text color': 'Colore del testo',
+ 'Auto scroll enabled': 'Scrolling automatico abilitato',
+ 'Auto scroll disabled': 'Scrolling automatico disabilitato',
+ 'Choose language': 'Scegli la lingua'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ja-jp.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ja-jp.js
new file mode 100644
index 0000000000..5408e65ad8
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ja-jp.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 10);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 10:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Japanese
+ * @author NHN FE Development Lab
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['ja', 'ja-JP'], {
+ Markdown: 'マークダウン',
+ WYSIWYG: 'WYSIWYG',
+ Write: '編集する',
+ Preview: 'プレビュー',
+ Headings: '見出し',
+ Paragraph: '本文',
+ Bold: '太字',
+ Italic: 'イタリック',
+ Strike: 'ストライク',
+ Code: 'インラインコード',
+ Line: 'ライン',
+ Blockquote: '引用',
+ 'Unordered list': '番号なしリスト',
+ 'Ordered list': '順序付きリスト',
+ Task: 'タスク',
+ Indent: 'インデント',
+ Outdent: 'アウトデント',
+ 'Insert link': 'リンク挿入',
+ 'Insert CodeBlock': 'コードブロック挿入',
+ 'Insert table': 'テーブル挿入',
+ 'Insert image': '画像挿入',
+ Heading: '見出し',
+ 'Image URL': 'イメージURL',
+ 'Select image file': '画像ファイル選択',
+ Description: 'ディスクリプション ',
+ OK: 'はい',
+ More: 'もっと',
+ Cancel: 'キャンセル',
+ File: 'ファイル',
+ URL: 'URL',
+ 'Link text': 'リンクテキスト',
+ 'Add row': '行追加',
+ 'Add col': '列追加',
+ 'Remove row': '行削除',
+ 'Remove col': '列削除',
+ 'Align left': '左揃え',
+ 'Align center': '中央揃え',
+ 'Align right': '右揃え',
+ 'Remove table': 'テーブル削除',
+ 'Would you like to paste as table?': 'テーブルを貼り付けますか?',
+ 'Text color': '文字色相',
+ 'Auto scroll enabled': '自動スクロールが有効',
+ 'Auto scroll disabled': '自動スクロールを無効に',
+ 'Choose language': '言語選択'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ko-kr.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ko-kr.js
new file mode 100644
index 0000000000..cb43e2ec04
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ko-kr.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 11);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 11:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Korean
+ * @author NHN FE Development Lab
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['ko', 'ko-KR'], {
+ Markdown: '마크다운',
+ WYSIWYG: '위지윅',
+ Write: '편집하기',
+ Preview: '미리보기',
+ Headings: '제목크기',
+ Paragraph: '본문',
+ Bold: '굵게',
+ Italic: '기울임꼴',
+ Strike: '취소선',
+ Code: '인라인 코드',
+ Line: '문단나눔',
+ Blockquote: '인용구',
+ 'Unordered list': '글머리 기호',
+ 'Ordered list': '번호 매기기',
+ Task: '체크박스',
+ Indent: '들여쓰기',
+ Outdent: '내어쓰기',
+ 'Insert link': '링크 삽입',
+ 'Insert CodeBlock': '코드블럭 삽입',
+ 'Insert table': '표 삽입',
+ 'Insert image': '이미지 삽입',
+ Heading: '제목',
+ 'Image URL': '이미지 주소',
+ 'Select image file': '이미지 파일을 선택하세요.',
+ Description: '설명',
+ OK: '확인',
+ More: '더 보기',
+ Cancel: '취소',
+ File: '파일',
+ URL: '주소',
+ 'Link text': '링크 텍스트',
+ 'Add row': '행 추가',
+ 'Add col': '열 추가',
+ 'Remove row': '행 삭제',
+ 'Remove col': '열 삭제',
+ 'Align left': '왼쪽 정렬',
+ 'Align center': '가운데 정렬',
+ 'Align right': '오른쪽 정렬',
+ 'Remove table': '표 삭제',
+ 'Would you like to paste as table?': '표형태로 붙여 넣겠습니까?',
+ 'Text color': '글자 색상',
+ 'Auto scroll enabled': '자동 스크롤 켜짐',
+ 'Auto scroll disabled': '자동 스크롤 꺼짐',
+ 'Choose language': '언어 선택'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/nb-no.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/nb-no.js
new file mode 100644
index 0000000000..3be393c544
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/nb-no.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 12);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 12:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Norwegian
+ * @author Anton Reytarovskiy
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['nb', 'nb-NO'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Skriv',
+ Preview: 'Forhåndsvis',
+ Headings: 'Overskrifter',
+ Paragraph: 'Avsnitt',
+ Bold: 'Fet skrift',
+ Italic: 'Kursiv',
+ Strike: 'Gjennomstrek',
+ Code: 'Kode',
+ Line: 'Linje',
+ Blockquote: 'Blokksitat',
+ 'Unordered list': 'Usortert liste',
+ 'Ordered list': 'Sortert liste',
+ Task: 'Task',
+ Indent: 'Indent',
+ Outdent: 'Outdent',
+ 'Insert link': 'Sett inn lenke',
+ 'Insert CodeBlock': 'Sett inn CodeStreng',
+ 'Insert table': 'Sett inn diagram',
+ 'Insert image': 'Sett inn bilde',
+ Heading: 'Overskrift',
+ 'Image URL': 'BildeURL',
+ 'Select image file': 'Velg bildefil',
+ Description: 'Beskrivelse',
+ OK: 'OK',
+ More: 'Mer',
+ Cancel: 'Angre',
+ File: 'Fil',
+ URL: 'URL',
+ 'Link text': 'Lenketekst',
+ 'Add row': 'Legg til rad',
+ 'Add col': 'Legg til kolonne',
+ 'Remove row': 'Fjern rad',
+ 'Remove col': 'Fjern kolonne',
+ 'Align left': 'Venstreorienter',
+ 'Align center': 'Senterorienter',
+ 'Align right': 'Høyreorienter',
+ 'Remove table': 'Fjern diagram',
+ 'Would you like to paste as table?': 'Ønsker du å lime inn som en tabell?',
+ 'Text color': 'Tekstfarge',
+ 'Auto scroll enabled': 'Auto-scroll aktivert',
+ 'Auto scroll disabled': 'Auto-scroll deaktivert',
+ 'Choose language': 'Velg språk'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/nl-nl.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/nl-nl.js
new file mode 100644
index 0000000000..b55d49c628
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/nl-nl.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 13);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 13:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Dutch
+ * @author NHN FE Development Lab
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['nl', 'nl-NL'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Opslaan',
+ Preview: 'Voorbeeld',
+ Headings: 'Koppen',
+ Paragraph: 'Alinea',
+ Bold: 'Vet',
+ Italic: 'Cursief',
+ Strike: 'Doorhalen',
+ Code: 'Inline code',
+ Line: 'Regel',
+ Blockquote: 'Citaatblok',
+ 'Unordered list': 'Opsomming',
+ 'Ordered list': 'Genummerde opsomming',
+ Task: 'Taak',
+ Indent: 'Niveau verhogen',
+ Outdent: 'Niveau verlagen',
+ 'Insert link': 'Link invoegen',
+ 'Insert CodeBlock': 'Codeblok toevoegen',
+ 'Insert table': 'Tabel invoegen',
+ 'Insert image': 'Afbeelding invoegen',
+ Heading: 'Kop',
+ 'Image URL': 'Afbeelding URL',
+ 'Select image file': 'Selecteer een afbeelding',
+ Description: 'Omschrijving',
+ OK: 'OK',
+ More: 'Meer',
+ Cancel: 'Annuleren',
+ File: 'Bestand',
+ URL: 'URL',
+ 'Link text': 'Link tekst',
+ 'Add row': 'Rij toevoegen',
+ 'Add col': 'Kolom toevoegen',
+ 'Remove row': 'Rij verwijderen',
+ 'Remove col': 'Kolom verwijderen',
+ 'Align left': 'Links uitlijnen',
+ 'Align center': 'Centreren',
+ 'Align right': 'Rechts uitlijnen',
+ 'Remove table': 'Verwijder tabel',
+ 'Would you like to paste as table?': 'Wil je dit als tabel plakken?',
+ 'Text color': 'Tekstkleur',
+ 'Auto scroll enabled': 'Autoscroll ingeschakeld',
+ 'Auto scroll disabled': 'Autoscroll uitgeschakeld',
+ 'Choose language': 'Kies een taal'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/pl-pl.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/pl-pl.js
new file mode 100644
index 0000000000..acb8b23ad1
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/pl-pl.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 14);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 14:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Polish
+ * @author Marcin Mikołajczak
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['pl', 'pl-PL'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Napisz',
+ Preview: 'Podgląd',
+ Headings: 'Nagłówki',
+ Paragraph: 'Akapit',
+ Bold: 'Pogrubienie',
+ Italic: 'Kursywa',
+ Strike: 'Przekreślenie',
+ Code: 'Fragment kodu',
+ Line: 'Linia',
+ Blockquote: 'Cytat',
+ 'Unordered list': 'Lista nieuporządkowana',
+ 'Ordered list': 'Lista uporządkowana',
+ Task: 'Zadanie',
+ Indent: 'Utwórz wcięcie',
+ Outdent: 'Usuń wcięcie',
+ 'Insert link': 'Umieść odnośnik',
+ 'Insert CodeBlock': 'Umieść blok kodu',
+ 'Insert table': 'Umieść tabelę',
+ 'Insert image': 'Umieść obraz',
+ Heading: 'Nagłówek',
+ 'Image URL': 'Adres URL obrazu',
+ 'Select image file': 'Wybierz plik obrazu',
+ Description: 'Opis',
+ OK: 'OK',
+ More: 'Więcej',
+ Cancel: 'Anuluj',
+ File: 'Plik',
+ URL: 'URL',
+ 'Link text': 'Tekst odnośnika',
+ 'Add row': 'Dodaj rząd',
+ 'Add col': 'Dodaj kolumnę',
+ 'Remove row': 'Usuń rząd',
+ 'Remove col': 'Usuń kolumnę',
+ 'Align left': 'Wyrównaj do lewej',
+ 'Align center': 'Wyśrodkuj',
+ 'Align right': 'Wyrównaj do prawej',
+ 'Remove table': 'Usuń tabelę',
+ 'Would you like to paste as table?': 'Czy chcesz wkleić tekst jako tabelę?',
+ 'Text color': 'Kolor tekstu',
+ 'Auto scroll enabled': 'Włączono automatyczne przewijanie',
+ 'Auto scroll disabled': 'Wyłączono automatyczne przewijanie',
+ 'Choose language': 'Wybierz język'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/pt-br.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/pt-br.js
new file mode 100644
index 0000000000..c5eb97233b
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/pt-br.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 15);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 15:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Português
+ * @author Nícolas Huber
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['pt', 'pt-BR'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Escrever',
+ Preview: 'Pré-visualizar',
+ Headings: 'Cabeçalhos',
+ Paragraph: 'Parágrafo',
+ Bold: 'Negrito',
+ Italic: 'Itálico',
+ Strike: 'Traçado',
+ Code: 'Código',
+ Line: 'Linha',
+ Blockquote: 'Bloco de citação',
+ 'Unordered list': 'Lista não ordenada',
+ 'Ordered list': 'Lista ordenada',
+ Task: 'Tarefa',
+ Indent: 'Recuo à esquerda',
+ Outdent: 'Recuo à direita',
+ 'Insert link': 'Inserir link',
+ 'Insert CodeBlock': 'Inserir bloco de código',
+ 'Insert table': 'Inserir tabela',
+ 'Insert image': 'Inserir imagem',
+ Heading: 'Título',
+ 'Image URL': 'URL da imagem',
+ 'Select image file': 'Selecione um arquivo de imagem',
+ Description: 'Descrição',
+ OK: 'OK',
+ More: 'Mais',
+ Cancel: 'Cancelar',
+ File: 'Arquivo',
+ URL: 'URL',
+ 'Link text': 'Link de texto',
+ 'Add row': 'Adicionar linha',
+ 'Add col': 'Adicionar coluna',
+ 'Remove row': 'Remover linha',
+ 'Remove col': 'Remover coluna',
+ 'Align left': 'Alinhar à esquerda',
+ 'Align center': 'Alinhar ao centro',
+ 'Align right': 'Alinhar à direita',
+ 'Remove table': 'Remover tabela',
+ 'Would you like to paste as table?': 'Você gostaria de colar como mesa?',
+ 'Text color': 'Cor do texto',
+ 'Auto scroll enabled': 'Rolagem automática habilitada',
+ 'Auto scroll disabled': 'Rolagem automática desabilitada',
+ 'Choose language': 'Escolher linguagem'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ru-ru.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ru-ru.js
new file mode 100644
index 0000000000..dd2a5462f5
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/ru-ru.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 16);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 16:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Russian
+ * @author Stepan Samko
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['ru', 'ru-RU'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Написать',
+ Preview: 'Предварительный просмотр',
+ Headings: 'Заголовки',
+ Paragraph: 'Абзац',
+ Bold: 'Жирный',
+ Italic: 'Курсив',
+ Strike: 'Зачеркнутый',
+ Code: 'Встроенный код',
+ Line: 'Строка',
+ Blockquote: 'Блок цитирования',
+ 'Unordered list': 'Неупорядоченный список',
+ 'Ordered list': 'Упорядоченный список',
+ Task: 'Задача',
+ Indent: 'отступ',
+ Outdent: 'Выступ',
+ 'Insert link': 'Вставить ссылку',
+ 'Insert CodeBlock': 'Вставить код',
+ 'Insert table': 'Вставить таблицу',
+ 'Insert image': 'Вставить изображение',
+ Heading: 'Заголовок',
+ 'Image URL': 'URL изображения',
+ 'Select image file': 'Выбрать файл изображения',
+ Description: 'Описание',
+ OK: 'Хорошо',
+ More: 'еще',
+ Cancel: 'Отмена',
+ File: 'Файл',
+ URL: 'URL',
+ 'Link text': 'Текст ссылки',
+ 'Add row': 'Добавить ряд',
+ 'Add col': 'Добавить столбец',
+ 'Remove row': 'Удалить ряд',
+ 'Remove col': 'Удалить столбец',
+ 'Align left': 'Выровнять по левому краю',
+ 'Align center': 'Выровнять по центру',
+ 'Align right': 'Выровнять по правому краю',
+ 'Remove table': 'Удалить таблицу',
+ 'Would you like to paste as table?': 'Вы хотите вставить в виде таблицы?',
+ 'Text color': 'Цвет текста',
+ 'Auto scroll enabled': 'Автоматическая прокрутка включена',
+ 'Auto scroll disabled': 'Автоматическая прокрутка отключена',
+ 'Choose language': 'Выбрать язык'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/sv-se.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/sv-se.js
new file mode 100644
index 0000000000..8a69357086
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/sv-se.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 17);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 17:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Swedish
+ * @author Magnus Aspling
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['sv', 'sv-SE'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Skriv',
+ Preview: 'Förhandsgranska',
+ Headings: 'Överskrifter',
+ Paragraph: 'Paragraf',
+ Bold: 'Fet',
+ Italic: 'Kursiv',
+ Strike: 'Genomstruken',
+ Code: 'Kodrad',
+ Line: 'Linje',
+ Blockquote: 'Citatblock',
+ 'Unordered list': 'Punktlista',
+ 'Ordered list': 'Numrerad lista',
+ Task: 'Att göra',
+ Indent: 'Öka indrag',
+ Outdent: 'Minska indrag',
+ 'Insert link': 'Infoga länk',
+ 'Insert CodeBlock': 'Infoga kodblock',
+ 'Insert table': 'Infoga tabell',
+ 'Insert image': 'Infoga bild',
+ Heading: 'Överskrift',
+ 'Image URL': 'Bildadress',
+ 'Select image file': 'Välj en bildfil',
+ Description: 'Beskrivning',
+ OK: 'OK',
+ More: 'Mer',
+ Cancel: 'Avbryt',
+ File: 'Fil',
+ URL: 'Adress',
+ 'Link text': 'Länktext',
+ 'Add row': 'Infoga rad',
+ 'Add col': 'Infoga kolumn',
+ 'Remove row': 'Radera rad',
+ 'Remove col': 'Radera kolumn',
+ 'Align left': 'Vänsterjustera',
+ 'Align center': 'Centrera',
+ 'Align right': 'Högerjustera',
+ 'Remove table': 'Radera tabell',
+ 'Would you like to paste as table?': 'Vill du klistra in som en tabell?',
+ 'Text color': 'Textfärg',
+ 'Auto scroll enabled': 'Automatisk scroll aktiverad',
+ 'Auto scroll disabled': 'Automatisk scroll inaktiverad',
+ 'Choose language': 'Välj språk'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/tr-tr.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/tr-tr.js
new file mode 100644
index 0000000000..5a6e877f6f
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/tr-tr.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 18);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 18:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Turkish
+ * @author Mesut Gölcük
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['tr', 'tr-TR'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Düzenle',
+ Preview: 'Ön izleme',
+ Headings: 'Başlıklar',
+ Paragraph: 'Paragraf',
+ Bold: 'Kalın',
+ Italic: 'İtalik',
+ Strike: 'Altı çizgili',
+ Code: 'Satır içi kod',
+ Line: 'Çizgi',
+ Blockquote: 'Alıntı',
+ 'Unordered list': 'Sıralanmamış liste',
+ 'Ordered list': 'Sıralı liste',
+ Task: 'Görev kutusu',
+ Indent: 'Girintiyi arttır',
+ Outdent: 'Girintiyi azalt',
+ 'Insert link': 'Bağlantı ekle',
+ 'Insert CodeBlock': 'Kod bloku ekle',
+ 'Insert table': 'Tablo ekle',
+ 'Insert image': 'İmaj ekle',
+ Heading: 'Başlık',
+ 'Image URL': 'İmaj URL',
+ 'Select image file': 'İmaj dosyası seç',
+ Description: 'Açıklama',
+ OK: 'Onay',
+ More: 'Daha Fazla',
+ Cancel: 'İptal',
+ File: 'Dosya',
+ URL: 'URL',
+ 'Link text': 'Bağlantı yazısı',
+ 'Add row': 'Satır ekle',
+ 'Add col': 'Sütun ekle',
+ 'Remove row': 'Satır sil',
+ 'Remove col': 'Sütun sil',
+ 'Align left': 'Sola hizala',
+ 'Align center': 'Merkeze hizala',
+ 'Align right': 'Sağa hizala',
+ 'Remove table': 'Tabloyu kaldır',
+ 'Would you like to paste as table?': 'Tablo olarak yapıştırmak ister misiniz?',
+ 'Text color': 'Metin rengi',
+ 'Auto scroll enabled': 'Otomatik kaydırma açık',
+ 'Auto scroll disabled': 'Otomatik kaydırma kapalı',
+ 'Choose language': 'Dil seçiniz'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/uk-ua.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/uk-ua.js
new file mode 100644
index 0000000000..e42497035e
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/uk-ua.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 19);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 19:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Ukrainian
+ * @author Nikolya
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage(['uk', 'uk-UA'], {
+ Markdown: 'Markdown',
+ WYSIWYG: 'WYSIWYG',
+ Write: 'Написати',
+ Preview: 'Попередній перегляд',
+ Headings: 'Заголовки',
+ Paragraph: 'Абзац',
+ Bold: 'Жирний',
+ Italic: 'Курсив',
+ Strike: 'Закреслений',
+ Code: 'Вбудований код',
+ Line: 'Лінія',
+ Blockquote: 'Блок цитування',
+ 'Unordered list': 'Невпорядкований список',
+ 'Ordered list': 'Упорядкований список',
+ Task: 'Завдання',
+ Indent: 'відступ',
+ Outdent: 'застарілий',
+ 'Insert link': 'Вставити посилання',
+ 'Insert CodeBlock': 'Вставити код',
+ 'Insert table': 'Вставити таблицю',
+ 'Insert image': 'Вставити зображення',
+ Heading: 'Заголовок',
+ 'Image URL': 'URL зображення',
+ 'Select image file': 'Вибрати файл зображення',
+ Description: 'Опис',
+ OK: 'OK',
+ More: 'ще',
+ Cancel: 'Скасувати',
+ File: 'Файл',
+ URL: 'URL',
+ 'Link text': 'Текст посилання',
+ 'Add row': 'Додати ряд',
+ 'Add col': 'Додати стовпчик',
+ 'Remove row': 'Видалити ряд',
+ 'Remove col': 'Видалити стовпчик',
+ 'Align left': 'Вирівняти по лівому краю',
+ 'Align center': 'Вирівняти по центру',
+ 'Align right': 'Вирівняти по правому краю',
+ 'Remove table': 'Видалити таблицю',
+ 'Would you like to paste as table?': 'Ви хочете вставити у вигляді таблиці?',
+ 'Text color': 'Колір тексту',
+ 'Auto scroll enabled': 'Автоматична прокрутка включена',
+ 'Auto scroll disabled': 'Автоматична прокрутка відключена',
+ 'Choose language': 'Вибрати мову'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/zh-cn.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/zh-cn.js
new file mode 100644
index 0000000000..5dd3332e70
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/zh-cn.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 20);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 20:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Chinese
+ * @author NHN FE Development Lab
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage('zh-CN', {
+ Markdown: 'Markdown',
+ WYSIWYG: '所见即所得',
+ Write: '编辑',
+ Preview: '预览',
+ Headings: '标题',
+ Paragraph: '文本',
+ Bold: '加粗',
+ Italic: '斜体字',
+ Strike: '删除线',
+ Code: '内嵌代码',
+ Line: '水平线',
+ Blockquote: '引用块',
+ 'Unordered list': '无序列表',
+ 'Ordered list': '有序列表',
+ Task: '任务',
+ Indent: '缩进',
+ Outdent: '减少缩进',
+ 'Insert link': '插入链接',
+ 'Insert CodeBlock': '插入代码块',
+ 'Insert table': '插入表格',
+ 'Insert image': '插入图片',
+ Heading: '标题',
+ 'Image URL': '图片网址',
+ 'Select image file': '选择图片文件',
+ Description: '说明',
+ OK: '确认',
+ More: '更多',
+ Cancel: '取消',
+ File: '文件',
+ URL: 'URL',
+ 'Link text': '链接文本',
+ 'Add row': '添加行',
+ 'Add col': '添加列',
+ 'Remove row': '删除行',
+ 'Remove col': '删除列',
+ 'Align left': '左对齐',
+ 'Align center': '居中对齐',
+ 'Align right': '右对齐',
+ 'Remove table': '删除表格',
+ 'Would you like to paste as table?': '需要粘贴为表格吗?',
+ 'Text color': '文字颜色',
+ 'Auto scroll enabled': '自动滚动已启用',
+ 'Auto scroll disabled': '自动滚动已禁用',
+ 'Choose language': '选择语言'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/zh-tw.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/zh-tw.js
new file mode 100644
index 0000000000..684b689d8a
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/i18n/zh-tw.js
@@ -0,0 +1,174 @@
+/*!
+ * TOAST UI Editor : i18n
+ * @version 2.5.1
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("@toast-ui/editor"));
+ else if(typeof define === 'function' && define.amd)
+ define(["@toast-ui/editor"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("@toast-ui/editor")) : factory(root["toastui"]["Editor"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(window, function(__WEBPACK_EXTERNAL_MODULE__0__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 21);
+/******/ })
+/************************************************************************/
+/******/ ({
+
+/***/ 0:
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE__0__;
+
+/***/ }),
+
+/***/ 21:
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+__webpack_require__.r(__webpack_exports__);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(0);
+/* harmony import */ var _editor__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_editor__WEBPACK_IMPORTED_MODULE_0__);
+/**
+ * @fileoverview I18N for Traditional Chinese
+ * @author Tzu-Ray Su
+ */
+
+_editor__WEBPACK_IMPORTED_MODULE_0___default.a.setLanguage('zh-TW', {
+ Markdown: 'Markdown',
+ WYSIWYG: '所見即所得',
+ Write: '編輯',
+ Preview: '預覽',
+ Headings: '標題',
+ Paragraph: '內文',
+ Bold: '粗體',
+ Italic: '斜體',
+ Strike: '刪除線',
+ Code: '內嵌程式碼',
+ Line: '分隔線',
+ Blockquote: '引言',
+ 'Unordered list': '項目符號清單',
+ 'Ordered list': '編號清單',
+ Task: '核取方塊清單',
+ Indent: '增加縮排',
+ Outdent: '減少縮排',
+ 'Insert link': '插入超連結',
+ 'Insert CodeBlock': '插入程式碼區塊',
+ 'Insert table': '插入表格',
+ 'Insert image': '插入圖片',
+ Heading: '標題',
+ 'Image URL': '圖片網址',
+ 'Select image file': '選擇圖片檔案',
+ Description: '描述',
+ OK: '確認',
+ More: '更多',
+ Cancel: '取消',
+ File: '檔案',
+ URL: 'URL',
+ 'Link text': '超連結文字',
+ 'Add row': '增加行',
+ 'Add col': '增加列',
+ 'Remove row': '刪除行',
+ 'Remove col': '刪除列',
+ 'Align left': '靠左對齊',
+ 'Align center': '置中',
+ 'Align right': '靠右對齊',
+ 'Remove table': '刪除表格',
+ 'Would you like to paste as table?': '您要以表格貼上嗎?',
+ 'Text color': '文字顏色',
+ 'Auto scroll enabled': '已啟用自動滾動',
+ 'Auto scroll disabled': '已停用自動滾動',
+ 'Choose language': '選擇語言'
+});
+
+/***/ })
+
+/******/ });
+});
\ No newline at end of file
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-old.css b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-old.css
new file mode 100644
index 0000000000..6499cff350
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-old.css
@@ -0,0 +1,1649 @@
+/*!
+ * @toast-ui/editor
+ * @version 2.5.1 | Tue Nov 24 2020
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+/* height */
+.auto-height,
+.auto-height .tui-editor-defaultUI {
+ height: auto;
+}
+
+.auto-height .tui-editor {
+ position: relative;
+}
+
+:not(.auto-height) > .tui-editor-defaultUI,
+:not(.auto-height) > .tui-editor-defaultUI > .te-editor-section {
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-direction: column;
+ flex-direction: column;
+}
+
+:not(.auto-height) > .tui-editor-defaultUI > .te-editor-section {
+ -ms-flex: 1;
+ flex: 1;
+}
+
+/* tui editor */
+.tui-editor:after,
+.tui-editor-defaultUI-toolbar:after {
+ content: '';
+ display: block;
+ height: 0;
+ clear: both;
+}
+
+.tui-editor {
+ position: absolute;
+ line-height: 1;
+ color: #181818;
+ width: 100%;
+ height: inherit;
+}
+
+.te-editor-section {
+ min-height: 0px;
+ position: relative;
+ height: inherit;
+}
+
+.te-md-container {
+ display: none;
+ overflow: hidden;
+ height: 100%;
+}
+
+.te-md-container .te-editor {
+ line-height: 1.5;
+}
+
+.te-md-container .te-editor,
+.te-md-container .te-preview {
+ box-sizing: border-box;
+ padding: 0;
+ height: inherit;
+}
+
+.te-md-container .CodeMirror {
+ font-size: 13px;
+ height: inherit;
+}
+
+.te-md-container .te-preview {
+ overflow: auto;
+ padding: 0 25px;
+ height: 100%;
+}
+
+.te-md-container .te-preview > p:first-child {
+ margin-top: 0 !important;
+}
+
+.te-md-container .te-preview .tui-editor-contents {
+ padding-top: 8px;
+}
+
+.tui-editor .te-preview-style-tab > .te-editor,
+.tui-editor .te-preview-style-tab > .te-preview {
+ float: left;
+ width: 100%;
+ display: none;
+}
+
+.tui-editor .te-preview-style-tab > .te-tab-active {
+ display: block;
+}
+
+.tui-editor .te-preview-style-vertical > .te-tab-section {
+ display: none;
+}
+
+.tui-editor .te-preview-style-tab > .te-tab-section {
+ display: block;
+}
+
+.tui-editor .te-preview-style-vertical .te-editor {
+ float: left;
+ width: 50%;
+}
+
+.tui-editor .te-preview-style-vertical .te-preview {
+ float: left;
+ width: 50%;
+}
+
+.tui-editor .te-md-splitter {
+ display: none;
+ position: absolute;
+ left: 50%;
+ top: 0;
+ height: 100%;
+ width: 1px;
+ border-left: 1px solid #e5e5e5;
+}
+
+.tui-editor .te-preview-style-vertical .te-md-splitter {
+ display: block;
+}
+
+.te-ww-container {
+ display: none;
+ overflow: hidden;
+ z-index: 10;
+ height: inherit;
+ background-color: #fff;
+}
+
+.te-ww-container > .te-editor {
+ overflow: auto;
+ height: inherit;
+}
+
+.te-ww-container .tui-editor-contents:focus {
+ outline: none;
+}
+
+.te-ww-container .tui-editor-contents {
+ padding: 0 25px;
+}
+
+.te-ww-container .tui-editor-contents:first-child {
+ box-sizing: border-box;
+ margin: 0px;
+ padding: 16px 25px 0px 25px;
+ height: inherit;
+}
+
+.te-ww-container .tui-editor-contents:last-child {
+ margin-bottom: 16px;
+}
+
+.te-md-mode .te-md-container {
+ display: block;
+ z-index: 100;
+}
+
+.te-ww-mode .te-ww-container {
+ display: block;
+ z-index: 100;
+}
+
+.tui-editor.te-hide,
+.tui-editor-defaultUI.te-hide {
+ display: none;
+}
+
+.tui-editor-defaultUI .CodeMirror-lines {
+ padding-top: 18px;
+ padding-bottom: 18px;
+}
+
+.tui-editor-defaultUI pre.CodeMirror-line {
+ padding-left: 25px;
+ padding-right: 25px;
+}
+
+.tui-editor-defaultUI .CodeMirror pre.CodeMirror-placeholder {
+ margin: 0;
+ padding-left: 25px;
+ color: grey;
+}
+
+.tui-editor-defaultUI .CodeMirror-scroll {
+ cursor: text;
+}
+
+/* Essential element style */
+.tui-editor-contents td.te-cell-selected {
+ background-color: #d8dfec;
+}
+.tui-editor-contents td.te-cell-selected::selection {
+ background-color: #d8dfec;
+}
+.tui-editor-contents th.te-cell-selected {
+ background-color: #908f8f;
+}
+.tui-editor-contents th.te-cell-selected::selection {
+ background-color: #908f8f;
+}
+
+/* default UI Styles */
+.tui-editor-defaultUI {
+ position: relative;
+ border: 1px solid #e5e5e5;
+ height: 100%;
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+}
+
+.tui-editor-defaultUI button {
+ color: #fff;
+ padding: 0px 14px 0px 15px;
+ height: 28px;
+ font-size: 12px;
+ border: none;
+ cursor: pointer;
+ outline: none;
+}
+.tui-editor-defaultUI button.te-ok-button {
+ background-color: #4b96e6;
+}
+.tui-editor-defaultUI button.te-close-button {
+ background-color: #777;
+}
+
+.tui-editor-defaultUI-toolbar {
+ padding: 0 25px;
+ height: 31px;
+ background-color: #fff;
+ border: 0;
+ overflow: hidden;
+}
+
+.tui-toolbar-divider {
+ float: left;
+ display: inline-block;
+ width: 1px;
+ height: 14px;
+ background-color: #ddd;
+ margin: 9px 6px;
+}
+
+.tui-toolbar-button-group {
+ height: 28px;
+ border-right: 1px solid #d9d9d9;
+ float: left;
+}
+
+.te-toolbar-section {
+ height: 32px;
+ box-sizing: border-box;
+ border-bottom: 1px solid #e5e5e5;
+}
+
+.tui-editor-defaultUI-toolbar button {
+ float: left;
+ box-sizing: border-box;
+ outline: none;
+ cursor: pointer;
+ background-color: #fff;
+ width: 22px;
+ height: 22px;
+ padding: 3px;
+ border-radius: 0;
+ margin: 5px 3px;
+ border: 1px solid #fff;
+}
+
+.tui-editor-defaultUI-toolbar button:hover,
+.tui-editor-defaultUI-toolbar button:active,
+.tui-editor-defaultUI-toolbar button.active {
+ border: 1px solid #aaa;
+ background-color: #fff;
+}
+
+.tui-editor-defaultUI-toolbar button:first-child {
+ margin-left: 0;
+}
+
+.tui-editor-defaultUI-toolbar button:last-child {
+ margin-right: 0;
+}
+
+.tui-editor-defaultUI-toolbar button.tui-scrollsync {
+ width: auto;
+ color: #777777;
+ border: 0;
+}
+
+.tui-editor-defaultUI button.tui-scrollsync:after {
+ content: 'Scroll off';
+}
+
+.tui-editor-defaultUI button.tui-scrollsync.active {
+ color: #125de6;
+ font-weight: bold;
+}
+
+.tui-editor-defaultUI button.tui-scrollsync.active:after {
+ content: 'Scroll on';
+}
+
+.tui-editor-defaultUI .te-mode-switch-section {
+ background-color: #f9f9f9;
+ border-top: 1px solid #e5e5e5;
+ height: 20px;
+ font-size: 12px;
+}
+
+.tui-editor-defaultUI .te-mode-switch {
+ float: right;
+ height: 100%;
+}
+
+.tui-editor-defaultUI .te-switch-button {
+ width: 92px;
+ height: inherit;
+ background: #e5e5e5;
+ outline: 0;
+ color: #a0aabf;
+ cursor: pointer;
+ border: 0;
+ border-left: 1px solid #ddd;
+ border-right: 1px solid #ddd;
+}
+
+.tui-editor-defaultUI .te-switch-button.active {
+ background-color: #fff;
+ color: #000;
+}
+
+.tui-editor-defaultUI .te-markdown-tab-section {
+ float: left;
+ height: 31px;
+ background: #fff;
+}
+
+.te-markdown-tab-section .te-tab {
+ margin: 0 -7px 0 24px;
+ background: #fff;
+}
+
+.tui-editor-defaultUI .te-tab button {
+ box-sizing: border-box;
+ line-height: 100%;
+ position: relative;
+ cursor: pointer;
+ z-index: 1;
+ font-size: 13px;
+ background-color: #f9f9f9;
+ border: solid 1px #e5e5e5;
+ border-top: 0;
+ padding: 0 9px;
+ color: #777;
+ border-radius: 0;
+ outline: 0;
+}
+
+.te-markdown-tab-section .te-tab button:last-child {
+ margin-left: -1px;
+}
+
+.te-markdown-tab-section .te-tab button.te-tab-active,
+.te-markdown-tab-section .te-tab button:hover.te-tab-active {
+ background-color: #fff;
+ color: #333;
+ border-bottom: 1px solid #fff;
+ z-index: 2;
+}
+
+.te-markdown-tab-section .te-tab button:hover {
+ background-color: #fff;
+ color: #333;
+}
+
+.tui-popup-modal-background {
+ background-color: rgba(202, 202, 202, 0.6);
+ position: fixed;
+ margin: 0px;
+ left: 0px;
+ top: 0px;
+ width: 100%;
+ height: 100%;
+ z-index: 9999;
+}
+
+.tui-popup-wrapper.fit-window,
+.tui-popup-modal-background.fit-window .tui-popup-wrapper {
+ width: 100%;
+ height: 100%;
+}
+
+.tui-popup-wrapper {
+ width: 500px;
+ margin-right: auto;
+ border: 1px solid #cacaca;
+ background: white;
+ z-index: 9999;
+}
+
+.tui-popup-modal-background .tui-popup-wrapper {
+ position: absolute;
+ margin: auto;
+ top: 0px;
+ right: 0px;
+ bottom: 0px;
+ left: 0px;
+}
+
+.tui-popup-header {
+ padding: 10px;
+ height: auto;
+ line-height: normal;
+ position: relative;
+ border-bottom: 1px solid #cacaca;
+}
+
+.tui-popup-header .tui-popup-header-buttons {
+ float: right;
+}
+
+.tui-popup-header .tui-popup-header-buttons button {
+ padding: 0px;
+ background-color: transparent;
+ background-size: cover;
+ float: left;
+}
+
+.tui-popup-header .tui-popup-close-button {
+ margin: 3px;
+ width: 13px;
+ height: 13px;
+ background-image: url();
+}
+
+.tui-popup-header .tui-popup-title {
+ font-size: 13px;
+ font-weight: bold;
+ color: #333;
+ vertical-align: bottom;
+}
+
+.tui-popup-body {
+ padding: 15px;
+ font-size: 12px;
+}
+
+.tui-editor-popup {
+ position: absolute;
+ top: 30px;
+ left: 50%;
+ margin-left: -250px;
+}
+
+.tui-editor-popup.tui-popup-modal-background {
+ position: fixed;
+ top: 0px;
+ left: 0px;
+ margin: 0px;
+}
+
+.tui-editor-popup .tui-popup-body label {
+ font-weight: bold;
+ color: #666;
+ display: block;
+ margin: 10px 0 5px;
+}
+
+.tui-editor-popup .tui-popup-body .te-button-section {
+ margin-top: 15px;
+}
+
+.tui-editor-popup .tui-popup-body input[type='text'],
+.tui-editor-popup .tui-popup-body input[type='file'] {
+ padding: 4px 10px;
+ border: 1px solid #bfbfbf;
+ box-sizing: border-box;
+ width: 100%;
+}
+
+.tui-editor-popup .tui-popup-body input[type='text'].disabled {
+ border-color: #e5e5e5;
+ background-color: #eee;
+ color: #e5e5e5;
+}
+
+.tui-editor-popup .tui-popup-body input.wrong {
+ border-color: #ff0000;
+}
+
+.te-popup-add-link .tui-popup-wrapper {
+ height: 219px;
+}
+
+.te-popup-add-image .tui-popup-wrapper {
+ height: 243px;
+}
+
+.te-popup-add-image .te-tab {
+ display: block;
+ background: none;
+ border-bottom: 1px solid #ebebeb;
+ margin-bottom: 8px;
+}
+
+.te-popup-add-image .te-url-type {
+ display: none;
+}
+
+.te-popup-add-image .te-file-type {
+ display: none;
+}
+
+.te-popup-add-image div.te-tab-active,
+.te-popup-add-image form.te-tab-active {
+ display: block;
+}
+
+.te-popup-add-image .te-tab button {
+ border: 1px solid #ccc;
+ background: #eee;
+ min-width: 100px;
+ margin-left: -1px;
+ border-bottom: 0px;
+ border-radius: 3px 3px 0px 0px;
+}
+
+.te-popup-add-image .te-tab button.te-tab-active {
+ background: #fff;
+}
+
+.te-popup-add-table .te-table-selection {
+ position: relative;
+}
+
+.te-popup-add-table .te-table-body {
+ background-image: url('');
+}
+
+.te-popup-add-table .te-table-header {
+ background-image: url('');
+}
+
+.te-popup-add-table .te-selection-area {
+ position: absolute;
+ top: 0;
+ left: 0;
+ background: #80d2ff;
+ opacity: 0.3;
+ z-index: 999;
+}
+
+.te-popup-add-table .te-description {
+ margin: 10px 0 0 0;
+ text-align: center;
+}
+
+.te-popup-table-utils {
+ width: 120px;
+}
+
+.te-popup-table-utils .tui-popup-body {
+ padding: 0px;
+}
+
+.te-popup-table-utils button {
+ width: 100%;
+ background-color: #fff;
+ border: none;
+ outline: 0;
+ padding: 0px 10px 0px 10px;
+ font-size: 12px;
+ line-height: 28px;
+ text-align: left;
+ color: #777;
+}
+
+.te-popup-table-utils button:hover {
+ background-color: #f4f4f4;
+}
+
+.te-popup-table-utils hr {
+ background-color: #cacaca;
+ border-style: none;
+ height: 1px;
+}
+
+.te-popup-table-utils .te-context-menu-disabled {
+ color: #ccc;
+}
+
+.te-popup-table-utils .te-context-menu-disabled:hover {
+ background-color: #fff;
+}
+
+.te-heading-add {
+ width: auto;
+}
+
+.te-heading-add .tui-popup-body {
+ padding: 0;
+}
+
+.te-heading-add h1,
+.te-heading-add h2,
+.te-heading-add h3,
+.te-heading-add h4,
+.te-heading-add h5,
+.te-heading-add h6,
+.te-heading-add ul,
+.te-heading-add p {
+ padding: 0;
+ margin: 0;
+}
+
+.te-heading-add ul {
+ list-style: none;
+}
+
+.te-heading-add ul li {
+ padding: 2px 10px;
+ cursor: pointer;
+}
+
+.te-heading-add ul li:hover {
+ background-color: #eee;
+}
+
+.te-heading-add h1 {
+ font-size: 24px;
+}
+
+.te-heading-add h2 {
+ font-size: 22px;
+}
+
+.te-heading-add h3 {
+ font-size: 20px;
+}
+
+.te-heading-add h4 {
+ font-size: 18px;
+}
+
+.te-heading-add h5 {
+ font-size: 16px;
+}
+
+.te-heading-add h6 {
+ font-size: 14px;
+}
+
+.te-dropdown-toolbar {
+ position: absolute;
+ width: auto;
+}
+
+.te-dropdown-toolbar .tui-popup-body {
+ padding: 0px;
+}
+
+.tui-popup-color {
+ padding: 0;
+}
+
+.tui-popup-color .tui-colorpicker-container,
+.tui-popup-color .tui-colorpicker-palette-container {
+ width: 144px;
+}
+
+.tui-popup-color .tui-colorpicker-container ul {
+ width: 144px;
+ margin-bottom: 8px;
+}
+
+.tui-popup-color .tui-colorpicker-container li {
+ padding: 0 1px 1px 0;
+}
+
+.tui-popup-color .tui-colorpicker-container li .tui-colorpicker-palette-button {
+ border: 0;
+ width: 17px;
+ height: 17px;
+}
+
+.tui-popup-color .tui-popup-body {
+ padding: 10px;
+}
+
+.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-toggle-slider {
+ display: none;
+}
+
+.tui-popup-color .te-apply-button,
+.tui-popup-color .tui-colorpicker-palette-hex {
+ float: right;
+}
+
+.tui-popup-color .te-apply-button {
+ height: 21px;
+ width: 35px;
+ background: #fff;
+ border: 1px solid #efefef;
+ position: absolute;
+ bottom: 135px;
+ right: 10px;
+ color: black;
+}
+
+.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-hex {
+ border: 1px solid #e1e1e1;
+ padding: 3px 14px;
+ margin-left: -1px;
+}
+
+.tui-popup-color .tui-colorpicker-container div.tui-colorpicker-clearfix {
+ display: inline-block;
+}
+
+.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-preview {
+ width: 19px;
+ height: 19px;
+}
+
+.tui-popup-color .tui-colorpicker-slider-container .tui-colorpicker-slider-right {
+ width: 22px;
+}
+
+.tui-popup-color .tui-colorpicker-slider-container .tui-colorpicker-huebar-handle {
+ display: none;
+}
+
+.tui-tooltip {
+ position: absolute;
+ background-color: #222;
+ z-index: 999;
+ opacity: 0.8;
+ color: #fff;
+ padding: 2px 5px;
+ font-size: 10px;
+}
+
+.tui-tooltip .arrow {
+ content: '';
+ display: inline-block;
+ width: 10px;
+ height: 10px;
+ background-color: #222;
+ -webkit-transform: rotate(45deg);
+ -moz-transform: rotate(45deg);
+ -ms-transform: rotate(45deg);
+ -o-transform: rotate(45deg);
+ transform: rotate(45deg);
+ position: absolute;
+ top: -3px;
+ left: 6px;
+ z-index: -1;
+}
+
+.tui-toolbar-icons {
+ background: url();
+ background-size: 218px 188px;
+ display: inline-block;
+}
+
+@media only screen and (-webkit-min-device-pixel-ratio: 2),
+ only screen and (min--moz-device-pixel-ratio: 2),
+ only screen and (-o-min-device-pixel-ratio: 2/1),
+ only screen and (min-device-pixel-ratio: 2),
+ only screen and (min-resolution: 192dpi),
+ only screen and (min-resolution: 2dppx) {
+ .tui-toolbar-icons {
+ background: url();
+ background-size: 218px 188px;
+ display: inline-block;
+ }
+}
+
+.tui-toolbar-icons.tui-heading {
+ background-position: -172px -48px;
+}
+
+.tui-toolbar-icons.tui-heading:disabled {
+ background-position: -193px -48px;
+}
+
+.tui-toolbar-icons.tui-bold {
+ background-position: -4px -4px;
+}
+
+.tui-toolbar-icons.tui-bold:disabled {
+ background-position: -25px -4px;
+}
+
+.tui-toolbar-icons.tui-italic {
+ background-position: -4px -48px;
+}
+
+.tui-toolbar-icons.tui-italic:disabled {
+ background-position: -25px -48px;
+}
+
+.tui-toolbar-icons.tui-color {
+ background-position: -172px -70px;
+}
+
+.tui-toolbar-icons.tui-color:disabled {
+ background-position: -193px -70px;
+}
+
+.tui-toolbar-icons.tui-strike {
+ background-position: -4px -26px;
+}
+
+.tui-toolbar-icons.tui-strike:disabled {
+ background-position: -25px -26px;
+}
+
+.tui-toolbar-icons.tui-hrline {
+ background-position: -46px -92px;
+}
+
+.tui-toolbar-icons.tui-hrline:disabled {
+ background-position: -67px -92px;
+}
+
+.tui-toolbar-icons.tui-quote {
+ background-position: -4px -114px;
+}
+
+.tui-toolbar-icons.tui-quote:disabled {
+ background-position: -25px -114px;
+}
+
+.tui-toolbar-icons.tui-ul {
+ background-position: -46px -4px;
+}
+
+.tui-toolbar-icons.tui-ul:disabled {
+ background-position: -67px -4px;
+}
+
+.tui-toolbar-icons.tui-ol {
+ background-position: -46px -26px;
+}
+
+.tui-toolbar-icons.tui-ol:disabled {
+ background-position: -67px -26px;
+}
+
+.tui-toolbar-icons.tui-task {
+ background-position: -130px -48px;
+}
+
+.tui-toolbar-icons.tui-task:disabled {
+ background-position: -151px -48px;
+}
+
+.tui-toolbar-icons.tui-indent {
+ background-position: -46px -48px;
+}
+
+.tui-toolbar-icons.tui-indent:disabled {
+ background-position: -67px -48px;
+}
+
+.tui-toolbar-icons.tui-outdent {
+ background-position: -46px -70px;
+}
+
+.tui-toolbar-icons.tui-outdent:disabled {
+ background-position: -67px -70px;
+}
+
+.tui-toolbar-icons.tui-table {
+ background-position: -88px -92px;
+}
+
+.tui-toolbar-icons.tui-table:disabled {
+ background-position: -109px -92px;
+}
+
+.tui-toolbar-icons.tui-image {
+ background-position: -130px -4px;
+}
+
+.tui-toolbar-icons.tui-image:disabled {
+ background-position: -151px -4px;
+}
+
+.tui-toolbar-icons.tui-link {
+ background-position: -130px -26px;
+}
+
+.tui-toolbar-icons.tui-link:disabled {
+ background-position: -151px -26px;
+}
+
+.tui-toolbar-icons.tui-code {
+ background-position: -130px -92px;
+}
+
+.tui-toolbar-icons.tui-code:disabled {
+ background-position: -151px -92px;
+}
+
+.tui-toolbar-icons.tui-codeblock {
+ background-position: -130px -70px;
+}
+
+.tui-toolbar-icons.tui-codeblock:disabled {
+ background-position: -151px -70px;
+}
+
+.tui-toolbar-icons.tui-more {
+ background-position: -172px -92px;
+}
+
+.tui-toolbar-icons.tui-more:disabled {
+ background-position: -193px -92px;
+}
+.tui-colorpicker-svg-slider {
+ border: 1px solid #ebebeb;
+}
+.tui-colorpicker-vml-slider {
+ border: 1px solid #ebebeb;
+}
+.tui-colorpicker-svg-huebar {
+ border: 1px solid #ebebeb;
+}
+
+.tui-editor-pseudo-clipboard {
+ position: fixed;
+ left: -1000px;
+ top: -1000px;
+ width: 100px;
+ height: 100px;
+}
+
+.te-ww-block-overlay.code-block-header {
+ text-align: right;
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+}
+
+.te-ww-block-overlay.code-block-header span {
+ font-size: 10px;
+ font-weight: 600;
+ padding: 0px 10px;
+ color: #333333;
+ cursor: default;
+}
+
+.te-ww-block-overlay.code-block-header button {
+ margin: 8px;
+ font-size: 10px;
+ color: #333333;
+ background-color: #f9f9f9;
+ border: 1px solid #dddddd;
+ padding: 4px;
+ height: auto;
+}
+
+.te-popup-code-block-languages {
+ position: fixed;
+ box-sizing: border-box;
+ width: 130px;
+}
+
+.te-popup-code-block-languages .tui-popup-body {
+ max-height: 169px;
+ overflow: auto;
+ padding: 0px;
+}
+
+.te-popup-code-block-languages button {
+ width: 100%;
+ background-color: #fff;
+ border: none;
+ outline: 0;
+ padding: 0px 10px 0px 10px;
+ font-size: 12px;
+ line-height: 24px;
+ text-align: left;
+ color: #777;
+}
+
+.te-popup-code-block-languages button.active {
+ background-color: #f4f4f4;
+}
+
+.tui-popup-code-block-editor .tui-popup-wrapper {
+ width: 70%;
+ height: 70%;
+ margin: auto;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-direction: column;
+ flex-direction: column;
+}
+
+.te-input-language {
+ position: relative;
+ margin-left: 15px;
+ cursor: pointer;
+}
+
+.te-input-language input {
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+ font-size: 10px;
+ padding: 3px 5px;
+ border: 1px solid #dddddd;
+ background-color: #f9f9f9;
+ box-sizing: border-box;
+ width: 130px;
+ outline: none;
+}
+
+.te-input-language input::-ms-clear {
+ display: none;
+}
+
+.te-input-language::after {
+ content: url();
+ position: absolute;
+ top: 1px;
+ right: 3px;
+}
+
+.te-input-language.active::after {
+ content: url();
+}
+
+.tui-popup-code-block-editor button {
+ margin: -1px 3px;
+}
+
+.tui-popup-code-block-editor .tui-popup-header-buttons {
+ height: 20px;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-preview::after {
+ content: 'Preview off';
+ color: #777;
+ margin-right: 22px;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-preview.active::after {
+ content: 'Preview on';
+ color: #4b96e6;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-scroll::after {
+ content: 'Scroll off';
+ color: #777;
+ margin-right: 16px;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-scroll.active::after {
+ content: 'Scroll on';
+ color: #4b96e6;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-fit {
+ width: 18px;
+ height: 18px;
+ margin-top: 4px;
+ margin-right: 14px;
+ background-image: url();
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-fit.active {
+ background-image: url();
+}
+
+.tui-popup-code-block-editor .tui-popup-close-button {
+ margin-top: 6px;
+}
+
+.tui-popup-code-block-editor .tui-popup-body {
+ z-index: -1;
+ padding: 0px;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ -ms-flex: 1;
+ flex: 1;
+}
+
+.tui-popup-code-block-editor .popup-editor-body {
+ position: relative;
+ -ms-flex: 1;
+ flex: 1;
+ border-bottom: 1px solid #cacaca;
+}
+
+.tui-popup-code-block-editor .te-button-section {
+ padding: 15px;
+}
+
+.tui-popup-code-block-editor .te-button-section button {
+ float: left;
+}
+
+.tui-popup-code-block-editor .tui-editor-contents pre {
+ margin: 0px;
+ background-color: transparent;
+}
+
+.tui-popup-code-block-editor .CodeMirror {
+ height: auto;
+}
+
+.tui-popup-code-block-editor .CodeMirror-line {
+ font-family: Consolas, Courier, 'Apple SD 산돌고딕 Neo', -apple-system, 'Lucida Grande',
+ 'Apple SD Gothic Neo', '맑은 고딕', 'Malgun Gothic', 'Segoe UI', '돋움', dotum, sans-serif;
+ font-size: 13px;
+ line-height: 160%;
+ letter-spacing: -0.3px;
+}
+
+.tui-popup-code-block-editor .popup-editor-editor-wrapper {
+ min-height: 100%;
+}
+
+.tui-split-scroll-wrapper {
+ position: relative;
+}
+
+.tui-split-scroll {
+ position: absolute;
+}
+
+.tui-split-scroll,
+.tui-split-scroll-wrapper {
+ width: 100%;
+ height: 100%;
+}
+
+.tui-split-scroll .tui-split-content-left,
+.tui-split-scroll .tui-split-content-right {
+ position: absolute;
+ top: 0px;
+ width: 50%;
+ box-sizing: border-box;
+}
+
+.tui-split-scroll .tui-split-content-left {
+ left: 0px;
+}
+
+.tui-split-scroll .tui-split-content-right {
+ left: 50%;
+}
+
+.tui-split-scroll .tui-splitter {
+ position: absolute;
+ left: 50%;
+ top: 0;
+ height: 100%;
+ width: 1px;
+ border-left: 1px solid #cacaca;
+}
+
+.tui-split-scroll .tui-split-scroll-content {
+ width: 100%;
+ height: 100%;
+ overflow: hidden;
+ position: relative;
+}
+
+.tui-split-scroll .tui-split-content-left,
+.tui-split-scroll .tui-split-content-right {
+ height: 100%;
+ overflow-x: hidden;
+ overflow-y: auto;
+}
+
+.tui-split-scroll button.tui-scrollsync {
+ top: 10px;
+ opacity: 0.2;
+}
+
+.tui-split-scroll button.tui-scrollsync::after {
+ content: 'scroll off';
+}
+
+.tui-split-scroll.scroll-sync button.tui-scrollsync {
+ opacity: 0.5;
+}
+
+.tui-split-scroll.scroll-sync .tui-split-content-left,
+.tui-split-scroll.scroll-sync .tui-split-content-right {
+ height: auto;
+ overflow: initial;
+}
+
+.tui-split-scroll.scroll-sync button.tui-scrollsync::after {
+ content: 'scroll on';
+}
+
+.tui-split-scroll.scroll-sync .tui-split-scroll-content {
+ overflow-y: auto;
+}
+
+.tui-split-scroll.single-content .tui-splitter {
+ display: none;
+}
+
+.tui-split-scroll.single-content .tui-split-content-left {
+ width: 100%;
+}
+
+.tui-split-scroll.single-content .tui-split-content-right {
+ display: none;
+}
+
+.tui-split-scroll.single-content button.tui-scrollsync {
+ display: none;
+}
+
+@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
+ /* IE10+ */
+ .tui-split-scroll-wrapper .tui-splitter {
+ left: calc(50% - 9px);
+ }
+}
+
+@supports (-ms-accelerator: true) {
+ /* IE Edge 12+ CSS styles go here */
+ .tui-split-scroll-wrapper .tui-splitter {
+ left: calc(50% - 9px);
+ }
+}
+
+@media screen and (max-width: 480px) {
+ .tui-popup-wrapper {
+ max-width: 300px;
+ }
+
+ .tui-editor-popup {
+ margin-left: -150px;
+ }
+
+ .te-dropdown-toolbar {
+ max-width: none;
+ }
+}
+
+@charset "utf-8";
+.CodeMirror {
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+}
+
+.tui-editor-contents *:not(table) {
+ line-height: 160%;
+ box-sizing: content-box;
+}
+
+.tui-editor-contents i,
+.tui-editor-contents cite,
+.tui-editor-contents em,
+.tui-editor-contents var,
+.tui-editor-contents address,
+.tui-editor-contents dfn {
+ font-style: italic;
+}
+
+.tui-editor-contents strong {
+ font-weight: bold;
+}
+
+.tui-editor-contents p {
+ margin: 10px 0;
+ color: #555;
+}
+
+.tui-editor-contents > h1:first-of-type,
+.tui-editor-contents > div > div:first-of-type h1 {
+ margin-top: 14px;
+}
+
+.tui-editor-contents h1,
+.tui-editor-contents h2,
+.tui-editor-contents h3,
+.tui-editor-contents h5 {
+ font-weight: bold;
+}
+
+.tui-editor-contents h1 {
+ font-size: 1.6rem;
+ line-height: 28px;
+ border-bottom: 3px double #999;
+ margin: 52px 0 15px 0;
+ padding-bottom: 7px;
+ color: #000;
+}
+
+.tui-editor-contents h2 {
+ font-size: 1.3rem;
+ line-height: 23px;
+ border-bottom: 1px solid #dbdbdb;
+ margin: 30px 0 13px 0;
+ padding-bottom: 7px;
+ color: #333;
+}
+
+.tui-editor-contents h3,
+.tui-editor-contents h4 {
+ font-size: 1.2rem;
+ line-height: 18px;
+ margin: 20px 0 2px;
+ color: #333;
+}
+
+.tui-editor-contents h5,
+.tui-editor-contents h6 {
+ font-size: 1rem;
+ line-height: 17px;
+ margin: 10px 0 -4px;
+ color: #333;
+}
+
+.tui-editor-contents blockquote {
+ margin: 15px 0;
+}
+
+.tui-editor-contents blockquote {
+ border-left: 4px solid #dddddd;
+ padding: 0 15px;
+ color: #777777;
+}
+
+.tui-editor-contents blockquote > :first-child {
+ margin-top: 0;
+}
+
+.tui-editor-contents blockquote > :last-child {
+ margin-bottom: 0;
+}
+
+.tui-editor-contents pre,
+.tui-editor-contents code {
+ font-family: Consolas, Courier, 'Lucida Grande', '나눔바른고딕', 'Nanum Barun Gothic', '맑은고딕',
+ 'Malgun Gothic', sans-serif;
+ border: 0;
+ border-radius: 0;
+}
+
+.tui-editor-contents pre {
+ margin: 2px 0 8px;
+ padding: 18px;
+ background-color: #f5f7f8;
+}
+
+.tui-editor-contents code {
+ color: #c1788b;
+ padding: 4px 4px 2px 0;
+ letter-spacing: -0.3px;
+}
+
+.tui-editor-contents pre code {
+ padding: 0;
+ color: inherit;
+ white-space: pre-wrap;
+ background-color: transparent;
+}
+
+.tui-editor-contents pre.addon {
+ border: 1px solid #e8ebed;
+ background-color: #fff;
+}
+
+.tui-editor-contents img {
+ margin: 4px 0 10px;
+ box-sizing: border-box;
+ vertical-align: top;
+ max-width: 100%;
+}
+
+.tui-editor-contents table {
+ margin: 2px 0 14px;
+ color: #555;
+ width: auto;
+ border-collapse: collapse;
+ box-sizing: border-box;
+}
+
+.tui-editor-contents table th,
+.tui-editor-contents table td {
+ height: 32px;
+ padding: 5px 14px 5px 12px;
+}
+
+.tui-editor-contents table td {
+ border: 1px solid #eaeaea;
+}
+
+.tui-editor-contents table th {
+ border: 1px solid #72777b;
+ border-top: 0;
+ background-color: #7b8184;
+ font-weight: 300;
+ color: #fff;
+ padding-top: 6px;
+}
+
+.tui-editor-contents ul,
+.tui-editor-contents menu,
+.tui-editor-contents ol,
+.tui-editor-contents dir {
+ display: block;
+ list-style-type: disc;
+ padding-left: 17px;
+ margin: 6px 0 10px;
+ color: #555;
+}
+
+.tui-editor-contents ol {
+ list-style-type: decimal;
+}
+
+.tui-editor-contents ul ul,
+.tui-editor-contents ul ol,
+.tui-editor-contents ol ol,
+.tui-editor-contents ol ul {
+ margin-top: 0 !important;
+ margin-bottom: 0 !important;
+}
+
+.tui-editor-contents ul li,
+.tui-editor-contents ol li {
+ position: relative;
+}
+
+.tui-editor-contents ul p,
+ol p {
+ margin: 0;
+}
+
+.tui-editor-contents ul li.task-list-item:before,
+.tui-editor-contents ol li.task-list-item:before,
+.tui-editor-contents pre ul li:before {
+ content: '';
+}
+
+.tui-editor-contents hr {
+ border-top: 1px solid #eee;
+ margin: 16px 0;
+}
+
+.tui-editor-contents a {
+ text-decoration: underline;
+ color: #5286bc;
+}
+
+.tui-editor-contents a:hover {
+ color: #007cff;
+}
+
+.tui-editor-contents a.image-link {
+ position: relative;
+}
+
+.tui-editor-contents a.image-link::before {
+ content: '';
+ position: absolute;
+ margin: 0;
+ width: 20px;
+ height: 20px;
+ top: 2px;
+ right: 2px;
+ background-repeat: no-repeat;
+ background-image: url('');
+ cursor: pointer;
+}
+
+.tui-editor-contents {
+ font-size: 13px;
+ margin: 0;
+ padding: 0;
+}
+
+.tui-editor-contents .task-list-item {
+ border: 0;
+ list-style: none;
+ padding-left: 22px;
+ margin-left: -22px;
+ min-height: 20px;
+}
+
+.tui-editor-contents .task-list-item:before {
+ background-repeat: no-repeat;
+ background-size: 16px 16px;
+ background-position: center;
+ content: '';
+ height: 18px;
+ width: 18px;
+ position: absolute;
+ left: 0;
+ top: 1px;
+ cursor: pointer;
+ background-image: url('');
+}
+
+.tui-editor-contents .task-list-item.checked:before {
+ background-image: url('');
+}
+
+.tui-editor-contents .task-list-item input[type='checkbox'],
+.tui-editor-contents .task-list-item .task-list-item-checkbox {
+ margin-left: -17px;
+ margin-right: 3.8px;
+ margin-top: 3px;
+}
+
+.tui-editor-contents-placeholder:before {
+ content: attr(data-placeholder);
+ color: grey;
+ line-height: 160%;
+ position: absolute;
+}
+
+.tui-editor-contents .te-preview-highlight {
+ position: relative;
+ z-index: 0;
+}
+
+.tui-editor-contents .te-preview-highlight::after {
+ content: '';
+ background-color: rgba(255, 245, 131, 0.5);
+ border-radius: 4px;
+ z-index: -1;
+ position: absolute;
+ top: -4px;
+ right: -4px;
+ left: -4px;
+ bottom: -4px;
+}
+
+.tui-editor-contents h1.te-preview-highlight::after,
+.tui-editor-contents h2.te-preview-highlight::after {
+ bottom: 0;
+}
+
+.tui-editor-contents td.te-preview-highlight::after,
+.tui-editor-contents th.te-preview-highlight::after {
+ display: none;
+}
+
+.tui-editor-contents th.te-preview-highlight,
+.tui-editor-contents td.te-preview-highlight {
+ background-color: rgba(255, 245, 131, 0.5);
+}
+
+.tui-editor-contents th.te-preview-highlight {
+ color: #222;
+}
+
+.te-md-container .CodeMirror {
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+ color: #000;
+}
+
+.tui-md-heading1 {
+ font-size: 24px;
+}
+
+.tui-md-heading2 {
+ font-size: 22px;
+}
+
+.tui-md-heading3 {
+ font-size: 20px;
+}
+
+.tui-md-heading4 {
+ font-size: 18px;
+}
+
+.tui-md-heading5 {
+ font-size: 16px;
+}
+
+.tui-md-heading6 {
+ font-size: 14px;
+}
+
+.tui-md-strong,
+.tui-md-heading {
+ font-weight: bold;
+}
+
+.tui-md-emph {
+ font-style: italic;
+}
+
+.tui-md-strike {
+ text-decoration: line-through;
+}
+
+.tui-md-thematic-break {
+ color: #ccc;
+}
+
+.tui-md-link.tui-md-link-desc {
+ color: #00c;
+}
+
+.tui-md-link.tui-md-link-url {
+ color: #a11;
+}
+
+.tui-md-block-quote {
+ color: #090;
+}
+
+.tui-md-code,
+.tui-md-code-block {
+ color: #a50;
+}
+
+.tui-md-list-item.first {
+ color: #000;
+}
+
+.tui-md-list-item.second {
+ color: #085;
+}
+
+.tui-md-list-item.third {
+ color: #708;
+}
+
+.tui-md-list-item.tui-md-delimiter {
+ color: #555;
+}
+
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-only.css b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-only.css
new file mode 100644
index 0000000000..4d62f7c2c9
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-only.css
@@ -0,0 +1,1433 @@
+/*!
+ * @toast-ui/editor
+ * @version 2.5.1 | Tue Nov 24 2020
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+@charset "utf-8";
+/* height */
+.auto-height,
+.auto-height .tui-editor-defaultUI {
+ height: auto;
+}
+
+.auto-height .tui-editor {
+ position: relative;
+}
+
+:not(.auto-height) > .tui-editor-defaultUI,
+:not(.auto-height) > .tui-editor-defaultUI > .te-editor-section {
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-direction: column;
+ flex-direction: column;
+}
+
+:not(.auto-height) > .tui-editor-defaultUI > .te-editor-section {
+ -ms-flex: 1;
+ flex: 1;
+}
+
+/* tui editor */
+.tui-editor:after,
+.tui-editor-defaultUI-toolbar:after {
+ content: '';
+ display: block;
+ height: 0;
+ clear: both;
+}
+
+.tui-editor {
+ position: absolute;
+ line-height: 1;
+ color: #222;
+ width: 100%;
+ height: inherit;
+}
+
+.te-editor-section {
+ min-height: 0px;
+ position: relative;
+ height: inherit;
+}
+
+.te-md-container {
+ display: none;
+ overflow: hidden;
+ height: 100%;
+}
+
+.te-md-container .te-editor {
+ line-height: 1.5;
+}
+
+.te-md-container .te-editor,
+.te-md-container .te-preview {
+ box-sizing: border-box;
+ padding: 0;
+ height: inherit;
+}
+
+.te-md-container .CodeMirror {
+ font-size: 13px;
+ height: inherit;
+}
+
+.te-md-container .te-preview {
+ overflow: auto;
+ padding: 0 25px;
+ height: 100%;
+}
+
+.te-md-container .te-preview > p:first-child {
+ margin-top: 0 !important;
+}
+
+.te-md-container .te-preview .tui-editor-contents {
+ padding-top: 8px;
+}
+
+.tui-editor .te-preview-style-tab > .te-editor,
+.tui-editor .te-preview-style-tab > .te-preview {
+ float: left;
+ width: 100%;
+ display: none;
+}
+
+.tui-editor .te-preview-style-tab > .te-tab-active {
+ display: block;
+}
+
+.tui-editor .te-preview-style-vertical > .te-tab-section {
+ display: none;
+}
+
+.tui-editor .te-preview-style-tab > .te-tab-section {
+ display: block;
+}
+
+.tui-editor .te-preview-style-vertical .te-editor {
+ float: left;
+ width: 50%;
+}
+
+.tui-editor .te-preview-style-vertical .te-preview {
+ float: left;
+ width: 50%;
+}
+
+.tui-editor .te-md-splitter {
+ display: none;
+ position: absolute;
+ left: 50%;
+ top: 0;
+ height: 100%;
+ width: 1px;
+ border-left: 1px solid #e5e5e5;
+}
+
+.tui-editor .te-preview-style-vertical .te-md-splitter {
+ display: block;
+}
+
+.te-ww-container {
+ display: none;
+ overflow: hidden;
+ z-index: 10;
+ height: inherit;
+ background-color: #fff;
+}
+
+.te-ww-container > .te-editor {
+ overflow: auto;
+ height: inherit;
+}
+
+.te-ww-container .tui-editor-contents:focus {
+ outline: none;
+}
+
+.te-ww-container .tui-editor-contents {
+ padding: 0 25px;
+}
+
+.te-ww-container .tui-editor-contents:first-child {
+ box-sizing: border-box;
+ margin: 0px;
+ padding: 16px 25px 0px 25px;
+ height: inherit;
+}
+
+.te-ww-container .tui-editor-contents:last-child {
+ margin-bottom: 16px;
+}
+
+.te-md-mode .te-md-container {
+ display: block;
+ z-index: 100;
+}
+
+.te-ww-mode .te-ww-container {
+ display: block;
+ z-index: 100;
+}
+
+.tui-editor.te-hide,
+.tui-editor-defaultUI.te-hide {
+ display: none;
+}
+
+.tui-editor-defaultUI .CodeMirror-lines {
+ padding-top: 18px;
+ padding-bottom: 18px;
+}
+
+.tui-editor-defaultUI pre.CodeMirror-line {
+ padding-left: 25px;
+ padding-right: 25px;
+}
+
+.tui-editor-defaultUI .CodeMirror pre.CodeMirror-placeholder {
+ margin: 0;
+ padding-left: 25px;
+ color: grey;
+}
+
+.tui-editor-defaultUI .CodeMirror-scroll {
+ cursor: text;
+}
+
+/* Essential element style */
+.tui-editor-contents td.te-cell-selected {
+ background-color: #d8dfec;
+}
+.tui-editor-contents td.te-cell-selected::selection {
+ background-color: #d8dfec;
+}
+.tui-editor-contents th.te-cell-selected {
+ background-color: #908f8f;
+}
+.tui-editor-contents th.te-cell-selected::selection {
+ background-color: #908f8f;
+}
+
+/* default UI Styles */
+.tui-editor-defaultUI {
+ position: relative;
+ border: 1px solid #e5e5e5;
+ height: 100%;
+ font-family: 'Open Sans', 'Helvetica Neue', 'Helvetica', 'Arial', '나눔바른고딕',
+ 'Nanum Barun Gothic', '맑은고딕', 'Malgun Gothic', sans-serif;
+}
+
+.tui-editor-defaultUI button {
+ color: #fff;
+ padding: 0px 14px 0px 15px;
+ height: 28px;
+ font-size: 12px;
+ border: none;
+ cursor: pointer;
+ outline: none;
+}
+.tui-editor-defaultUI button.te-ok-button {
+ background-color: #4b96e6;
+}
+.tui-editor-defaultUI button.te-close-button {
+ background-color: #777;
+}
+
+.tui-editor-defaultUI-toolbar {
+ padding: 0 25px;
+ height: 31px;
+ background-color: #fff;
+ border: 0;
+ overflow: hidden;
+}
+
+.tui-toolbar-divider {
+ float: left;
+ display: inline-block;
+ width: 1px;
+ height: 14px;
+ background-color: #ddd;
+ margin: 9px 6px;
+}
+
+.tui-toolbar-button-group {
+ height: 28px;
+ border-right: 1px solid #d9d9d9;
+ float: left;
+}
+
+.te-toolbar-section {
+ height: 32px;
+ box-sizing: border-box;
+ border-bottom: 1px solid #e5e5e5;
+}
+
+.tui-editor-defaultUI-toolbar button {
+ float: left;
+ box-sizing: border-box;
+ outline: none;
+ cursor: pointer;
+ background-color: #fff;
+ width: 22px;
+ height: 22px;
+ padding: 3px;
+ border-radius: 0;
+ margin: 5px 3px;
+ border: 1px solid #fff;
+}
+
+.tui-editor-defaultUI-toolbar button:hover,
+.tui-editor-defaultUI-toolbar button:active,
+.tui-editor-defaultUI-toolbar button.active {
+ border: 1px solid #aaa;
+ background-color: #fff;
+}
+
+.tui-editor-defaultUI-toolbar button:first-child {
+ margin-left: 0;
+}
+
+.tui-editor-defaultUI-toolbar button:last-child {
+ margin-right: 0;
+}
+
+.tui-editor-defaultUI-toolbar button.tui-scrollsync {
+ width: auto;
+ color: #777777;
+ border: 0;
+}
+
+.tui-editor-defaultUI button.tui-scrollsync:after {
+ content: 'Scroll off';
+}
+
+.tui-editor-defaultUI button.tui-scrollsync.active {
+ color: #4b96e6;
+ font-weight: bold;
+}
+
+.tui-editor-defaultUI button.tui-scrollsync.active:after {
+ content: 'Scroll on';
+}
+
+.tui-editor-defaultUI .te-mode-switch-section {
+ background-color: #f9f9f9;
+ border-top: 1px solid #e5e5e5;
+ height: 20px;
+ font-size: 12px;
+}
+
+.tui-editor-defaultUI .te-mode-switch {
+ float: right;
+ height: 100%;
+}
+
+.tui-editor-defaultUI .te-switch-button {
+ width: 92px;
+ height: inherit;
+ background: #e5e5e5;
+ outline: 0;
+ color: #a0aabf;
+ cursor: pointer;
+ border: 0;
+ border-left: 1px solid #ddd;
+ border-right: 1px solid #ddd;
+}
+
+.tui-editor-defaultUI .te-switch-button.active {
+ background-color: #fff;
+ color: #000;
+}
+
+.tui-editor-defaultUI .te-markdown-tab-section {
+ float: left;
+ height: 31px;
+ background: #fff;
+}
+
+.te-markdown-tab-section .te-tab {
+ margin: 0 -7px 0 24px;
+ background: #fff;
+}
+
+.tui-editor-defaultUI .te-tab button {
+ box-sizing: border-box;
+ line-height: 100%;
+ position: relative;
+ cursor: pointer;
+ z-index: 1;
+ font-size: 13px;
+ background-color: #f9f9f9;
+ border: solid 1px #e5e5e5;
+ border-top: 0;
+ padding: 0 9px;
+ color: #777;
+ border-radius: 0;
+ outline: 0;
+}
+
+.te-markdown-tab-section .te-tab button:last-child {
+ margin-left: -1px;
+}
+
+.te-markdown-tab-section .te-tab button.te-tab-active,
+.te-markdown-tab-section .te-tab button:hover.te-tab-active {
+ background-color: #fff;
+ color: #333;
+ border-bottom: 1px solid #fff;
+ z-index: 2;
+}
+
+.te-markdown-tab-section .te-tab button:hover {
+ background-color: #fff;
+ color: #333;
+}
+
+.tui-popup-modal-background {
+ background-color: rgba(202, 202, 202, 0.6);
+ position: fixed;
+ margin: 0px;
+ left: 0px;
+ top: 0px;
+ width: 100%;
+ height: 100%;
+ z-index: 9999;
+}
+
+.tui-popup-wrapper.fit-window,
+.tui-popup-modal-background.fit-window .tui-popup-wrapper {
+ width: 100%;
+ height: 100%;
+}
+
+.tui-popup-wrapper {
+ width: 500px;
+ margin-right: auto;
+ border: 1px solid #cacaca;
+ background: white;
+ z-index: 9999;
+}
+
+.tui-popup-modal-background .tui-popup-wrapper {
+ position: absolute;
+ margin: auto;
+ top: 0px;
+ right: 0px;
+ bottom: 0px;
+ left: 0px;
+}
+
+.tui-popup-header {
+ padding: 10px;
+ height: auto;
+ line-height: normal;
+ position: relative;
+ border-bottom: 1px solid #cacaca;
+}
+
+.tui-popup-header .tui-popup-header-buttons {
+ float: right;
+}
+
+.tui-popup-header .tui-popup-header-buttons button {
+ padding: 0px;
+ background-color: transparent;
+ background-size: cover;
+ float: left;
+}
+
+.tui-popup-header .tui-popup-close-button {
+ margin: 3px;
+ width: 13px;
+ height: 13px;
+ background-image: url();
+}
+
+.tui-popup-header .tui-popup-title {
+ font-size: 13px;
+ font-weight: bold;
+ color: #333;
+ vertical-align: bottom;
+}
+
+.tui-popup-body {
+ padding: 15px;
+ font-size: 12px;
+}
+
+.tui-editor-popup {
+ position: absolute;
+ top: 30px;
+ left: 50%;
+ margin-left: -250px;
+}
+
+.tui-editor-popup.tui-popup-modal-background {
+ position: fixed;
+ top: 0px;
+ left: 0px;
+ margin: 0px;
+}
+
+.tui-editor-popup .tui-popup-body label {
+ font-weight: bold;
+ color: #666;
+ display: block;
+ margin: 10px 0 5px;
+}
+
+.tui-editor-popup .tui-popup-body .te-button-section {
+ margin-top: 15px;
+}
+
+.tui-editor-popup .tui-popup-body input[type='text'],
+.tui-editor-popup .tui-popup-body input[type='file'] {
+ padding: 4px 10px;
+ border: 1px solid #bfbfbf;
+ box-sizing: border-box;
+ width: 100%;
+}
+
+.tui-editor-popup .tui-popup-body input[type='text'].disabled {
+ border-color: #e5e5e5;
+ background-color: #eee;
+ color: #e5e5e5;
+}
+
+.tui-editor-popup .tui-popup-body input.wrong {
+ border-color: #ff0000;
+}
+
+.te-popup-add-link .tui-popup-wrapper {
+ height: 219px;
+}
+
+.te-popup-add-image .tui-popup-wrapper {
+ height: 243px;
+}
+
+.te-popup-add-image .te-tab {
+ display: block;
+ background: none;
+ border-bottom: 1px solid #ebebeb;
+ margin-bottom: 8px;
+}
+
+.te-popup-add-image .te-url-type {
+ display: none;
+}
+
+.te-popup-add-image .te-file-type {
+ display: none;
+}
+
+.te-popup-add-image div.te-tab-active,
+.te-popup-add-image form.te-tab-active {
+ display: block;
+}
+
+.te-popup-add-image .te-tab button {
+ border: 1px solid #ccc;
+ background: #eee;
+ min-width: 100px;
+ margin-left: -1px;
+ border-bottom: 0px;
+ border-radius: 3px 3px 0px 0px;
+}
+
+.te-popup-add-image .te-tab button.te-tab-active {
+ background: #fff;
+}
+
+.te-popup-add-table .te-table-selection {
+ position: relative;
+}
+
+.te-popup-add-table .te-table-body {
+ background-image: url('');
+}
+
+.te-popup-add-table .te-table-header {
+ background-image: url('');
+}
+
+.te-popup-add-table .te-selection-area {
+ position: absolute;
+ top: 0;
+ left: 0;
+ background: #80d2ff;
+ opacity: 0.3;
+ z-index: 999;
+}
+
+.te-popup-add-table .te-description {
+ margin: 10px 0 0 0;
+ text-align: center;
+}
+
+.te-popup-table-utils {
+ width: auto;
+ min-width: 120px;
+}
+
+.te-popup-table-utils .tui-popup-body {
+ padding: 0px;
+}
+
+.te-popup-table-utils button {
+ display: block;
+ width: 100%;
+ background-color: #fff;
+ border: none;
+ outline: 0;
+ padding: 0px 10px 0px 10px;
+ font-size: 12px;
+ line-height: 28px;
+ text-align: left;
+ color: #777;
+}
+
+.te-popup-table-utils button:hover {
+ background-color: #f4f4f4;
+}
+
+.te-popup-table-utils hr {
+ margin: 0;
+ background-color: #cacaca;
+ border-style: none;
+ height: 1px;
+}
+
+.te-popup-table-utils .te-context-menu-disabled {
+ color: #ccc;
+}
+
+.te-popup-table-utils .te-context-menu-disabled:hover {
+ background-color: #fff;
+}
+
+.te-heading-add {
+ width: auto;
+}
+
+.te-heading-add .tui-popup-body {
+ padding: 0;
+}
+
+.te-heading-add h1,
+.te-heading-add h2,
+.te-heading-add h3,
+.te-heading-add h4,
+.te-heading-add h5,
+.te-heading-add h6,
+.te-heading-add ul,
+.te-heading-add p {
+ padding: 0;
+ margin: 0;
+}
+
+.te-heading-add ul {
+ list-style: none;
+}
+
+.te-heading-add ul li {
+ padding: 2px 10px;
+ cursor: pointer;
+}
+
+.te-heading-add ul li:hover {
+ background-color: #eee;
+}
+
+.te-heading-add h1 {
+ font-size: 24px;
+}
+
+.te-heading-add h2 {
+ font-size: 22px;
+}
+
+.te-heading-add h3 {
+ font-size: 20px;
+}
+
+.te-heading-add h4 {
+ font-size: 18px;
+}
+
+.te-heading-add h5 {
+ font-size: 16px;
+}
+
+.te-heading-add h6 {
+ font-size: 14px;
+}
+
+.te-dropdown-toolbar {
+ position: absolute;
+ width: auto;
+}
+
+.te-dropdown-toolbar .tui-popup-body {
+ padding: 0px;
+}
+
+.tui-popup-color {
+ padding: 0;
+}
+
+.tui-popup-color .tui-colorpicker-container,
+.tui-popup-color .tui-colorpicker-palette-container {
+ width: 144px;
+}
+
+.tui-popup-color .tui-colorpicker-container ul {
+ width: 144px;
+ margin-bottom: 8px;
+}
+
+.tui-popup-color .tui-colorpicker-container li {
+ padding: 0 1px 1px 0;
+}
+
+.tui-popup-color .tui-colorpicker-container li .tui-colorpicker-palette-button {
+ border: 0;
+ width: 17px;
+ height: 17px;
+}
+
+.tui-popup-color .tui-popup-body {
+ padding: 10px;
+}
+
+.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-toggle-slider {
+ display: none;
+}
+
+.tui-popup-color .te-apply-button,
+.tui-popup-color .tui-colorpicker-palette-hex {
+ float: right;
+}
+
+.tui-popup-color .te-apply-button {
+ height: 21px;
+ width: 35px;
+ background: #fff;
+ border: 1px solid #efefef;
+ position: absolute;
+ bottom: 135px;
+ right: 10px;
+ color: black;
+}
+
+.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-hex {
+ border: 1px solid #e1e1e1;
+ padding: 3px 14px;
+ margin-left: -1px;
+}
+
+.tui-popup-color .tui-colorpicker-container div.tui-colorpicker-clearfix {
+ display: inline-block;
+}
+
+.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-preview {
+ width: 19px;
+ height: 19px;
+}
+
+.tui-popup-color .tui-colorpicker-slider-container .tui-colorpicker-slider-right {
+ width: 22px;
+}
+
+.tui-popup-color .tui-colorpicker-slider-container .tui-colorpicker-huebar-handle {
+ display: none;
+}
+
+.tui-tooltip {
+ position: absolute;
+ background-color: #222;
+ z-index: 999;
+ opacity: 0.8;
+ color: #fff;
+ padding: 2px 5px;
+ font-size: 10px;
+}
+
+.tui-tooltip .arrow {
+ content: '';
+ display: inline-block;
+ width: 10px;
+ height: 10px;
+ background-color: #222;
+ -webkit-transform: rotate(45deg);
+ -moz-transform: rotate(45deg);
+ -ms-transform: rotate(45deg);
+ -o-transform: rotate(45deg);
+ transform: rotate(45deg);
+ position: absolute;
+ top: -3px;
+ left: 6px;
+ z-index: -1;
+}
+
+.tui-toolbar-icons {
+ background: url();
+ background-size: 218px 188px;
+ display: inline-block;
+}
+
+@media only screen and (-webkit-min-device-pixel-ratio: 2),
+ only screen and (min--moz-device-pixel-ratio: 2),
+ only screen and (-o-min-device-pixel-ratio: 2/1),
+ only screen and (min-device-pixel-ratio: 2),
+ only screen and (min-resolution: 192dpi),
+ only screen and (min-resolution: 2dppx) {
+ .tui-toolbar-icons {
+ background: url();
+ background-size: 218px 188px;
+ display: inline-block;
+ }
+}
+
+.tui-toolbar-icons.tui-heading {
+ background-position: -172px -48px;
+}
+
+.tui-toolbar-icons.tui-heading:disabled {
+ background-position: -193px -48px;
+}
+
+.tui-toolbar-icons.tui-bold {
+ background-position: -4px -4px;
+}
+
+.tui-toolbar-icons.tui-bold:disabled {
+ background-position: -25px -4px;
+}
+
+.tui-toolbar-icons.tui-italic {
+ background-position: -4px -48px;
+}
+
+.tui-toolbar-icons.tui-italic:disabled {
+ background-position: -25px -48px;
+}
+
+.tui-toolbar-icons.tui-color {
+ background-position: -172px -70px;
+}
+
+.tui-toolbar-icons.tui-color:disabled {
+ background-position: -193px -70px;
+}
+
+.tui-toolbar-icons.tui-strike {
+ background-position: -4px -26px;
+}
+
+.tui-toolbar-icons.tui-strike:disabled {
+ background-position: -25px -26px;
+}
+
+.tui-toolbar-icons.tui-hrline {
+ background-position: -46px -92px;
+}
+
+.tui-toolbar-icons.tui-hrline:disabled {
+ background-position: -67px -92px;
+}
+
+.tui-toolbar-icons.tui-quote {
+ background-position: -4px -114px;
+}
+
+.tui-toolbar-icons.tui-quote:disabled {
+ background-position: -25px -114px;
+}
+
+.tui-toolbar-icons.tui-ul {
+ background-position: -46px -4px;
+}
+
+.tui-toolbar-icons.tui-ul:disabled {
+ background-position: -67px -4px;
+}
+
+.tui-toolbar-icons.tui-ol {
+ background-position: -46px -26px;
+}
+
+.tui-toolbar-icons.tui-ol:disabled {
+ background-position: -67px -26px;
+}
+
+.tui-toolbar-icons.tui-task {
+ background-position: -130px -48px;
+}
+
+.tui-toolbar-icons.tui-task:disabled {
+ background-position: -151px -48px;
+}
+
+.tui-toolbar-icons.tui-indent {
+ background-position: -46px -48px;
+}
+
+.tui-toolbar-icons.tui-indent:disabled {
+ background-position: -67px -48px;
+}
+
+.tui-toolbar-icons.tui-outdent {
+ background-position: -46px -70px;
+}
+
+.tui-toolbar-icons.tui-outdent:disabled {
+ background-position: -67px -70px;
+}
+
+.tui-toolbar-icons.tui-table {
+ background-position: -88px -92px;
+}
+
+.tui-toolbar-icons.tui-table:disabled {
+ background-position: -109px -92px;
+}
+
+.tui-toolbar-icons.tui-image {
+ background-position: -130px -4px;
+}
+
+.tui-toolbar-icons.tui-image:disabled {
+ background-position: -151px -4px;
+}
+
+.tui-toolbar-icons.tui-link {
+ background-position: -130px -26px;
+}
+
+.tui-toolbar-icons.tui-link:disabled {
+ background-position: -151px -26px;
+}
+
+.tui-toolbar-icons.tui-code {
+ background-position: -130px -92px;
+}
+
+.tui-toolbar-icons.tui-code:disabled {
+ background-position: -151px -92px;
+}
+
+.tui-toolbar-icons.tui-codeblock {
+ background-position: -130px -70px;
+}
+
+.tui-toolbar-icons.tui-codeblock:disabled {
+ background-position: -151px -70px;
+}
+
+.tui-toolbar-icons.tui-more {
+ background-position: -172px -92px;
+}
+
+.tui-toolbar-icons.tui-more:disabled {
+ background-position: -193px -92px;
+}
+.tui-colorpicker-svg-slider {
+ border: 1px solid #ebebeb;
+}
+.tui-colorpicker-vml-slider {
+ border: 1px solid #ebebeb;
+}
+.tui-colorpicker-svg-huebar {
+ border: 1px solid #ebebeb;
+}
+
+.tui-editor-pseudo-clipboard {
+ position: fixed;
+ left: -1000px;
+ top: -1000px;
+ width: 100px;
+ height: 100px;
+}
+
+.te-ww-block-overlay.code-block-header {
+ text-align: right;
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+}
+
+.te-ww-block-overlay.code-block-header span {
+ font-size: 10px;
+ font-weight: 600;
+ padding: 0px 10px;
+ color: #333333;
+ cursor: default;
+}
+
+.te-ww-block-overlay.code-block-header button {
+ margin: 8px;
+ font-size: 10px;
+ color: #333333;
+ background-color: #f9f9f9;
+ border: 1px solid #dddddd;
+ padding: 4px;
+ height: auto;
+}
+
+.te-popup-code-block-languages {
+ position: fixed;
+ box-sizing: border-box;
+ width: 130px;
+}
+
+.te-popup-code-block-languages .tui-popup-body {
+ max-height: 169px;
+ overflow: auto;
+ padding: 0px;
+}
+
+.te-popup-code-block-languages button {
+ width: 100%;
+ background-color: #fff;
+ border: none;
+ outline: 0;
+ padding: 0px 10px 0px 10px;
+ font-size: 12px;
+ line-height: 24px;
+ text-align: left;
+ color: #777;
+}
+
+.te-popup-code-block-languages button.active {
+ background-color: #f4f4f4;
+}
+
+.tui-popup-code-block-editor .tui-popup-wrapper {
+ width: 70%;
+ height: 70%;
+ margin: auto;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-direction: column;
+ flex-direction: column;
+}
+
+.te-input-language {
+ position: relative;
+ margin-left: 15px;
+ cursor: pointer;
+}
+
+.te-input-language input {
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+ font-size: 10px;
+ padding: 3px 5px;
+ border: 1px solid #dddddd;
+ background-color: #f9f9f9;
+ box-sizing: border-box;
+ width: 130px;
+ outline: none;
+}
+
+.te-input-language input::-ms-clear {
+ display: none;
+}
+
+.te-input-language::after {
+ content: url();
+ position: absolute;
+ top: 1px;
+ right: 3px;
+}
+
+.te-input-language.active::after {
+ content: url();
+}
+
+.tui-popup-code-block-editor button {
+ margin: -1px 3px;
+}
+
+.tui-popup-code-block-editor .tui-popup-header-buttons {
+ height: 20px;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-preview::after {
+ content: 'Preview off';
+ color: #777;
+ margin-right: 22px;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-preview.active::after {
+ content: 'Preview on';
+ color: #4b96e6;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-scroll::after {
+ content: 'Scroll off';
+ color: #777;
+ margin-right: 16px;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-scroll.active::after {
+ content: 'Scroll on';
+ color: #4b96e6;
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-fit {
+ width: 18px;
+ height: 18px;
+ margin-top: 4px;
+ margin-right: 14px;
+ background-image: url();
+}
+
+.tui-popup-code-block-editor .popup-editor-toggle-fit.active {
+ background-image: url();
+}
+
+.tui-popup-code-block-editor .tui-popup-close-button {
+ margin-top: 6px;
+}
+
+.tui-popup-code-block-editor .tui-popup-body {
+ z-index: -1;
+ padding: 0px;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ -ms-flex: 1;
+ flex: 1;
+}
+
+.tui-popup-code-block-editor .popup-editor-body {
+ position: relative;
+ -ms-flex: 1;
+ flex: 1;
+ border-bottom: 1px solid #cacaca;
+}
+
+.tui-popup-code-block-editor .te-button-section {
+ padding: 15px;
+}
+
+.tui-popup-code-block-editor .te-button-section button {
+ float: left;
+}
+
+.tui-popup-code-block-editor .tui-editor-contents pre {
+ margin: 0px;
+ background-color: transparent;
+}
+
+.tui-popup-code-block-editor .CodeMirror {
+ height: auto;
+}
+
+.tui-popup-code-block-editor .CodeMirror-line {
+ font-family: Consolas, Courier, 'Lucida Grande', '나눔바른고딕', 'Nanum Barun Gothic', '맑은고딕',
+ 'Malgun Gothic', sans-serif;
+ font-size: 13px;
+ line-height: 160%;
+ letter-spacing: -0.3px;
+}
+
+.tui-popup-code-block-editor .popup-editor-editor-wrapper {
+ min-height: 100%;
+}
+
+.tui-split-scroll-wrapper {
+ position: relative;
+}
+
+.tui-split-scroll {
+ position: absolute;
+}
+
+.tui-split-scroll,
+.tui-split-scroll-wrapper {
+ width: 100%;
+ height: 100%;
+}
+
+.tui-split-scroll .tui-split-content-left,
+.tui-split-scroll .tui-split-content-right {
+ position: absolute;
+ top: 0px;
+ width: 50%;
+ box-sizing: border-box;
+}
+
+.tui-split-scroll .tui-split-content-left {
+ left: 0px;
+}
+
+.tui-split-scroll .tui-split-content-right {
+ left: 50%;
+}
+
+.tui-split-scroll .tui-splitter {
+ position: absolute;
+ left: 50%;
+ top: 0;
+ height: 100%;
+ width: 1px;
+ border-left: 1px solid #cacaca;
+}
+
+.tui-split-scroll .tui-split-scroll-content {
+ width: 100%;
+ height: 100%;
+ overflow: hidden;
+ position: relative;
+}
+
+.tui-split-scroll .tui-split-content-left,
+.tui-split-scroll .tui-split-content-right {
+ height: 100%;
+ overflow-x: hidden;
+ overflow-y: auto;
+}
+
+.tui-split-scroll button.tui-scrollsync {
+ top: 10px;
+ opacity: 0.2;
+}
+
+.tui-split-scroll button.tui-scrollsync::after {
+ content: 'scroll off';
+}
+
+.tui-split-scroll.scroll-sync button.tui-scrollsync {
+ opacity: 0.5;
+}
+
+.tui-split-scroll.scroll-sync .tui-split-content-left,
+.tui-split-scroll.scroll-sync .tui-split-content-right {
+ height: auto;
+ overflow: initial;
+}
+
+.tui-split-scroll.scroll-sync button.tui-scrollsync::after {
+ content: 'scroll on';
+}
+
+.tui-split-scroll.scroll-sync .tui-split-scroll-content {
+ overflow-y: auto;
+}
+
+.tui-split-scroll.single-content .tui-splitter {
+ display: none;
+}
+
+.tui-split-scroll.single-content .tui-split-content-left {
+ width: 100%;
+}
+
+.tui-split-scroll.single-content .tui-split-content-right {
+ display: none;
+}
+
+.tui-split-scroll.single-content button.tui-scrollsync {
+ display: none;
+}
+
+@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
+ /* IE10+ */
+ .tui-split-scroll-wrapper .tui-splitter {
+ left: calc(50% - 9px);
+ }
+}
+
+@supports (-ms-accelerator: true) {
+ /* IE Edge 12+ CSS styles go here */
+ .tui-split-scroll-wrapper .tui-splitter {
+ left: calc(50% - 9px);
+ }
+}
+
+@media screen and (max-width: 480px) {
+ .tui-popup-wrapper {
+ max-width: 300px;
+ }
+
+ .tui-editor-popup {
+ margin-left: -150px;
+ }
+
+ .te-dropdown-toolbar {
+ max-width: none;
+ }
+}
+
+.tui-editor-contents .te-preview-highlight {
+ position: relative;
+ z-index: 0;
+}
+
+.tui-editor-contents .te-preview-highlight::after {
+ content: '';
+ background-color: rgba(255, 245, 131, 0.5);
+ border-radius: 4px;
+ z-index: -1;
+ position: absolute;
+ top: -4px;
+ right: -4px;
+ left: -4px;
+ bottom: -4px;
+}
+
+.tui-editor-contents h1.te-preview-highlight::after,
+.tui-editor-contents h2.te-preview-highlight::after {
+ bottom: 0;
+}
+
+.tui-editor-contents td.te-preview-highlight::after,
+.tui-editor-contents th.te-preview-highlight::after {
+ display: none;
+}
+
+.tui-editor-contents th.te-preview-highlight,
+.tui-editor-contents td.te-preview-highlight {
+ background-color: rgba(255, 245, 131, 0.5);
+}
+
+.tui-editor-contents th.te-preview-highlight {
+ color: #222;
+}
+
+.te-md-container .CodeMirror {
+ font-family: 'Open Sans', 'Helvetica Neue', 'Helvetica', 'Arial', '나눔바른고딕',
+ 'Nanum Barun Gothic', '맑은고딕', 'Malgun Gothic', sans-serif;
+ color: #222;
+}
+
+.tui-md-heading1 {
+ font-size: 24px;
+}
+
+.tui-md-heading2 {
+ font-size: 22px;
+}
+
+.tui-md-heading3 {
+ font-size: 20px;
+}
+
+.tui-md-heading4 {
+ font-size: 18px;
+}
+
+.tui-md-heading5 {
+ font-size: 16px;
+}
+
+.tui-md-heading6 {
+ font-size: 14px;
+}
+
+.tui-md-heading.tui-md-delimiter.setext {
+ line-height: 15px;
+}
+
+.tui-md-strong,
+.tui-md-heading,
+.tui-md-list-item.tui-md-list-item-bullet,
+.tui-md-list-item.tui-md-meta {
+ font-weight: bold;
+}
+
+.tui-md-emph {
+ font-style: italic;
+}
+
+.tui-md-strike {
+ text-decoration: line-through;
+}
+
+.tui-md-strike.tui-md-delimiter {
+ text-decoration: none;
+}
+
+.tui-md-delimiter,
+.tui-md-thematic-break,
+.tui-md-link,
+.tui-md-table,
+.tui-md-block-quote {
+ color: #ccc;
+}
+
+.tui-md-code-block.tui-md-meta,
+.tui-md-code.tui-md-delimiter {
+ color: #aaa;
+}
+
+.tui-md-meta,
+.tui-md-html,
+.tui-md-link.tui-md-link-url.tui-md-marked-text {
+ color: #999;
+}
+
+.tui-md-block-quote.tui-md-marked-text,
+.tui-md-list-item.tui-md-meta {
+ color: #555;
+}
+
+.tui-md-table.tui-md-marked-text {
+ color: #222;
+}
+
+.tui-md-link.tui-md-link-desc.tui-md-marked-text,
+.tui-md-list-item-odd.tui-md-list-item-bullet {
+ color: #4b96e6;
+}
+
+.tui-md-list-item-even.tui-md-list-item-bullet {
+ color: #cb4848;
+}
+
+.tui-md-code.tui-md-marked-text {
+ color: #c1798b;
+}
+
+.tui-md-code {
+ background-color: rgba(243, 229, 233, 0.5);
+ padding: 2px 0;
+ letter-spacing: -0.3px;
+}
+
+.tui-md-code.tui-md-delimiter.start {
+ padding-left: 2px;
+ border-top-left-radius: 2px;
+ border-bottom-left-radius: 2px;
+}
+
+.tui-md-code.tui-md-delimiter.end {
+ padding-right: 2px;
+ border-top-right-radius: 2px;
+ border-bottom-right-radius: 2px;
+}
+
+.tui-md-code-block.CodeMirror-linebackground {
+ left: 20px;
+ right: 20px;
+ background-color: #f5f7f8;
+}
+
+.tui-md-code-block.CodeMirror-linebackground.start {
+ top: 2px;
+}
+
+.tui-md-code-block.CodeMirror-linebackground.end {
+ bottom: 2px;
+}
+
+.tui-md-code,
+.tui-md-code-block {
+ font-family: Consolas, Courier, 'Lucida Grande', '나눔바른고딕', 'Nanum Barun Gothic', '맑은고딕',
+ 'Malgun Gothic', sans-serif;
+}
+
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer-old.css b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer-old.css
new file mode 100644
index 0000000000..ed8facad3c
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer-old.css
@@ -0,0 +1,280 @@
+/*!
+ * @toast-ui/editor
+ * @version 2.5.1 | Tue Nov 24 2020
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+@charset "utf-8";
+.CodeMirror {
+ font-family: 'Open Sans', 'Helvetica Neue', Helvetica, Arial, sans-serif;
+}
+
+.tui-editor-contents *:not(table) {
+ line-height: 160%;
+ box-sizing: content-box;
+}
+
+.tui-editor-contents i,
+.tui-editor-contents cite,
+.tui-editor-contents em,
+.tui-editor-contents var,
+.tui-editor-contents address,
+.tui-editor-contents dfn {
+ font-style: italic;
+}
+
+.tui-editor-contents strong {
+ font-weight: bold;
+}
+
+.tui-editor-contents p {
+ margin: 10px 0;
+ color: #555;
+}
+
+.tui-editor-contents > h1:first-of-type,
+.tui-editor-contents > div > div:first-of-type h1 {
+ margin-top: 14px;
+}
+
+.tui-editor-contents h1,
+.tui-editor-contents h2,
+.tui-editor-contents h3,
+.tui-editor-contents h5 {
+ font-weight: bold;
+}
+
+.tui-editor-contents h1 {
+ font-size: 1.6rem;
+ line-height: 28px;
+ border-bottom: 3px double #999;
+ margin: 52px 0 15px 0;
+ padding-bottom: 7px;
+ color: #000;
+}
+
+.tui-editor-contents h2 {
+ font-size: 1.3rem;
+ line-height: 23px;
+ border-bottom: 1px solid #dbdbdb;
+ margin: 30px 0 13px 0;
+ padding-bottom: 7px;
+ color: #333;
+}
+
+.tui-editor-contents h3,
+.tui-editor-contents h4 {
+ font-size: 1.2rem;
+ line-height: 18px;
+ margin: 20px 0 2px;
+ color: #333;
+}
+
+.tui-editor-contents h5,
+.tui-editor-contents h6 {
+ font-size: 1rem;
+ line-height: 17px;
+ margin: 10px 0 -4px;
+ color: #333;
+}
+
+.tui-editor-contents blockquote {
+ margin: 15px 0;
+}
+
+.tui-editor-contents blockquote {
+ border-left: 4px solid #dddddd;
+ padding: 0 15px;
+ color: #777777;
+}
+
+.tui-editor-contents blockquote > :first-child {
+ margin-top: 0;
+}
+
+.tui-editor-contents blockquote > :last-child {
+ margin-bottom: 0;
+}
+
+.tui-editor-contents pre,
+.tui-editor-contents code {
+ font-family: Consolas, Courier, 'Lucida Grande', '나눔바른고딕', 'Nanum Barun Gothic', '맑은고딕',
+ 'Malgun Gothic', sans-serif;
+ border: 0;
+ border-radius: 0;
+}
+
+.tui-editor-contents pre {
+ margin: 2px 0 8px;
+ padding: 18px;
+ background-color: #f5f7f8;
+}
+
+.tui-editor-contents code {
+ color: #c1788b;
+ padding: 4px 4px 2px 0;
+ letter-spacing: -0.3px;
+}
+
+.tui-editor-contents pre code {
+ padding: 0;
+ color: inherit;
+ white-space: pre-wrap;
+ background-color: transparent;
+}
+
+.tui-editor-contents pre.addon {
+ border: 1px solid #e8ebed;
+ background-color: #fff;
+}
+
+.tui-editor-contents img {
+ margin: 4px 0 10px;
+ box-sizing: border-box;
+ vertical-align: top;
+ max-width: 100%;
+}
+
+.tui-editor-contents table {
+ margin: 2px 0 14px;
+ color: #555;
+ width: auto;
+ border-collapse: collapse;
+ box-sizing: border-box;
+}
+
+.tui-editor-contents table th,
+.tui-editor-contents table td {
+ height: 32px;
+ padding: 5px 14px 5px 12px;
+}
+
+.tui-editor-contents table td {
+ border: 1px solid #eaeaea;
+}
+
+.tui-editor-contents table th {
+ border: 1px solid #72777b;
+ border-top: 0;
+ background-color: #7b8184;
+ font-weight: 300;
+ color: #fff;
+ padding-top: 6px;
+}
+
+.tui-editor-contents ul,
+.tui-editor-contents menu,
+.tui-editor-contents ol,
+.tui-editor-contents dir {
+ display: block;
+ list-style-type: disc;
+ padding-left: 17px;
+ margin: 6px 0 10px;
+ color: #555;
+}
+
+.tui-editor-contents ol {
+ list-style-type: decimal;
+}
+
+.tui-editor-contents ul ul,
+.tui-editor-contents ul ol,
+.tui-editor-contents ol ol,
+.tui-editor-contents ol ul {
+ margin-top: 0 !important;
+ margin-bottom: 0 !important;
+}
+
+.tui-editor-contents ul li,
+.tui-editor-contents ol li {
+ position: relative;
+}
+
+.tui-editor-contents ul p,
+ol p {
+ margin: 0;
+}
+
+.tui-editor-contents ul li.task-list-item:before,
+.tui-editor-contents ol li.task-list-item:before,
+.tui-editor-contents pre ul li:before {
+ content: '';
+}
+
+.tui-editor-contents hr {
+ border-top: 1px solid #eee;
+ margin: 16px 0;
+}
+
+.tui-editor-contents a {
+ text-decoration: underline;
+ color: #5286bc;
+}
+
+.tui-editor-contents a:hover {
+ color: #007cff;
+}
+
+.tui-editor-contents a.image-link {
+ position: relative;
+}
+
+.tui-editor-contents a.image-link::before {
+ content: '';
+ position: absolute;
+ margin: 0;
+ width: 20px;
+ height: 20px;
+ top: 2px;
+ right: 2px;
+ background-repeat: no-repeat;
+ background-image: url('');
+ cursor: pointer;
+}
+
+.tui-editor-contents {
+ font-size: 13px;
+ margin: 0;
+ padding: 0;
+}
+
+.tui-editor-contents .task-list-item {
+ border: 0;
+ list-style: none;
+ padding-left: 22px;
+ margin-left: -22px;
+ min-height: 20px;
+}
+
+.tui-editor-contents .task-list-item:before {
+ background-repeat: no-repeat;
+ background-size: 16px 16px;
+ background-position: center;
+ content: '';
+ height: 18px;
+ width: 18px;
+ position: absolute;
+ left: 0;
+ top: 1px;
+ cursor: pointer;
+ background-image: url('');
+}
+
+.tui-editor-contents .task-list-item.checked:before {
+ background-image: url('');
+}
+
+.tui-editor-contents .task-list-item input[type='checkbox'],
+.tui-editor-contents .task-list-item .task-list-item-checkbox {
+ margin-left: -17px;
+ margin-right: 3.8px;
+ margin-top: 3px;
+}
+
+.tui-editor-contents-placeholder:before {
+ content: attr(data-placeholder);
+ color: grey;
+ line-height: 160%;
+ position: absolute;
+}
+
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer.css b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer.css
new file mode 100644
index 0000000000..2766d2969d
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer.css
@@ -0,0 +1,377 @@
+/*!
+ * @toast-ui/editor
+ * @version 2.5.1 | Tue Nov 24 2020
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+@charset "utf-8";
+.tui-editor-contents {
+ margin: 0;
+ padding: 0;
+ font-size: 13px;
+ font-family: 'Open Sans', 'Helvetica Neue', 'Helvetica', 'Arial', '나눔바른고딕',
+ 'Nanum Barun Gothic', '맑은고딕', 'Malgun Gothic', sans-serif;
+}
+
+.tui-editor-contents *:not(table) {
+ line-height: 160%;
+ box-sizing: content-box;
+}
+
+.tui-editor-contents i,
+.tui-editor-contents cite,
+.tui-editor-contents em,
+.tui-editor-contents var,
+.tui-editor-contents address,
+.tui-editor-contents dfn {
+ font-style: italic;
+}
+
+.tui-editor-contents strong {
+ font-weight: bold;
+}
+
+.tui-editor-contents p {
+ margin: 10px 0;
+ color: #222;
+}
+
+.tui-editor-contents > h1:first-of-type,
+.tui-editor-contents > div > div:first-of-type h1 {
+ margin-top: 14px;
+}
+
+.tui-editor-contents h1,
+.tui-editor-contents h2,
+.tui-editor-contents h3,
+.tui-editor-contents h4,
+.tui-editor-contents h5,
+.tui-editor-contents h6 {
+ font-weight: bold;
+ color: #222;
+}
+
+.tui-editor-contents h1 {
+ font-size: 24px;
+ line-height: 28px;
+ border-bottom: 3px double #999;
+ margin: 52px 0 15px 0;
+ padding-bottom: 7px;
+}
+
+.tui-editor-contents h2 {
+ font-size: 22px;
+ line-height: 23px;
+ border-bottom: 1px solid #dbdbdb;
+ margin: 20px 0 13px 0;
+ padding-bottom: 7px;
+}
+
+.tui-editor-contents h3 {
+ font-size: 20px;
+ margin: 18px 0 2px;
+}
+
+.tui-editor-contents h4 {
+ font-size: 18px;
+ margin: 10px 0 2px;
+}
+
+.tui-editor-contents h3,
+.tui-editor-contents h4 {
+ line-height: 18px;
+}
+
+.tui-editor-contents h5 {
+ font-size: 16px;
+}
+
+.tui-editor-contents h6 {
+ font-size: 14px;
+}
+
+.tui-editor-contents h5,
+.tui-editor-contents h6 {
+ line-height: 17px;
+ margin: 9px 0 -4px;
+}
+
+.tui-editor-contents del {
+ color: #999;
+}
+
+.tui-editor-contents blockquote {
+ margin: 14px 0;
+ border-left: 4px solid #e5e5e5;
+ padding: 0 16px;
+ color: #999;
+}
+
+.tui-editor-contents blockquote p,
+.tui-editor-contents blockquote ul,
+.tui-editor-contents blockquote ol {
+ color: #999;
+}
+
+.tui-editor-contents blockquote > :first-child {
+ margin-top: 0;
+}
+
+.tui-editor-contents blockquote > :last-child {
+ margin-bottom: 0;
+}
+
+.tui-editor-contents pre,
+.tui-editor-contents code {
+ font-family: Consolas, Courier, 'Apple SD 산돌고딕 Neo', -apple-system, 'Lucida Grande',
+ 'Apple SD Gothic Neo', '맑은 고딕', 'Malgun Gothic', 'Segoe UI', '돋움', dotum, sans-serif;
+ border: 0;
+ border-radius: 0;
+}
+
+.tui-editor-contents pre {
+ margin: 2px 0 8px;
+ padding: 18px;
+ background-color: #f5f7f8;
+}
+
+.tui-editor-contents code {
+ color: #c1798b;
+ background-color: #f9f2f4;
+ padding: 2px 3px;
+ letter-spacing: -0.3px;
+ border-radius: 2px;
+}
+
+.tui-editor-contents pre code {
+ padding: 0;
+ color: inherit;
+ white-space: pre-wrap;
+ background-color: transparent;
+}
+
+.tui-editor-contents pre.addon {
+ border: 1px solid #e8ebed;
+ background-color: #fff;
+}
+
+.tui-editor-contents img {
+ margin: 4px 0 10px;
+ box-sizing: border-box;
+ vertical-align: top;
+ max-width: 100%;
+}
+
+.tui-editor-contents table {
+ border: 1px solid rgba(0, 0, 0, 0.1);
+ margin: 12px 0 14px;
+ color: #222;
+ width: auto;
+ border-collapse: collapse;
+ box-sizing: border-box;
+}
+
+.tui-editor-contents table th,
+.tui-editor-contents table td {
+ border: 1px solid rgba(0, 0, 0, 0.1);
+ padding: 5px 14px 5px 12px;
+ height: 32px;
+}
+
+.tui-editor-contents table th {
+ background-color: #555;
+ font-weight: 300;
+ color: #fff;
+ padding-top: 6px;
+}
+
+.tui-editor-contents ul,
+.tui-editor-contents menu,
+.tui-editor-contents ol,
+.tui-editor-contents dir {
+ display: block;
+ list-style-type: none;
+ padding-left: 24px;
+ margin: 6px 0 10px;
+ color: #222;
+}
+
+.tui-editor-contents ol {
+ list-style-type: none;
+ counter-reset: li;
+}
+
+.tui-editor-contents ol > li {
+ counter-increment: li;
+}
+
+.tui-editor-contents ul > li::before,
+.tui-editor-contents ol > li::before {
+ display: inline-block;
+ position: absolute;
+}
+
+.tui-editor-contents ul > li::before {
+ content: '';
+ margin-top: 6px;
+ margin-left: -17px;
+ width: 5px;
+ height: 5px;
+ border-radius: 50%;
+ background-color: #ccc;
+}
+
+.tui-editor-contents ol > li::before {
+ content: '.' counter(li);
+ margin-left: -28px;
+ width: 24px;
+ text-align: right;
+ direction: rtl;
+ color: #aaa;
+}
+
+.tui-editor-contents ul ul,
+.tui-editor-contents ul ol,
+.tui-editor-contents ol ol,
+.tui-editor-contents ol ul {
+ margin-top: 0 !important;
+ margin-bottom: 0 !important;
+}
+
+.tui-editor-contents ul li,
+.tui-editor-contents ol li {
+ position: relative;
+}
+
+.tui-editor-contents ul p,
+.tui-editor-contents ol p {
+ margin: 0;
+}
+
+.tui-editor-contents ul li.task-list-item::before,
+.tui-editor-contents ol li.task-list-item::before,
+.tui-editor-contents pre ul li::before {
+ content: '';
+}
+
+.tui-editor-contents th ol,
+.tui-editor-contents th ul {
+ color: #fff;
+}
+
+.tui-editor-contents hr {
+ border-top: 1px solid #eee;
+ margin: 16px 0;
+}
+
+.tui-editor-contents a {
+ text-decoration: underline;
+ color: #4b96e6;
+}
+
+.tui-editor-contents a:hover {
+ color: #1f70de;
+}
+
+.tui-editor-contents a.image-link {
+ position: relative;
+}
+
+.tui-editor-contents a.image-link::before {
+ content: '';
+ position: absolute;
+ margin: 0;
+ width: 20px;
+ height: 20px;
+ top: 2px;
+ right: 2px;
+ background-repeat: no-repeat;
+ background-image: url('');
+ cursor: pointer;
+}
+
+.tui-editor-contents .task-list-item {
+ border: 0;
+ list-style: none;
+ padding-left: 24px;
+ margin-left: -24px;
+}
+
+.tui-editor-contents .task-list-item::before {
+ background-repeat: no-repeat;
+ background-size: 18px 18px;
+ background-position: center;
+ content: '';
+ margin-left: 0;
+ margin-top: 0;
+ border-radius: 0;
+ height: 18px;
+ width: 18px;
+ position: absolute;
+ left: 0;
+ top: 1px;
+ cursor: pointer;
+ background-image: url('');
+}
+
+.tui-editor-contents .task-list-item.checked::before {
+ background-image: url('');
+}
+
+.tui-editor-contents .task-list-item input[type='checkbox'],
+.tui-editor-contents .task-list-item .task-list-item-checkbox {
+ margin-left: -17px;
+ margin-right: 3.8px;
+ margin-top: 3px;
+}
+
+.tui-editor-contents-placeholder::before {
+ content: attr(data-placeholder);
+ color: grey;
+ line-height: 160%;
+ position: absolute;
+}
+
+.te-preview .tui-editor-contents h1 {
+ min-height: 28px;
+}
+
+.te-preview .tui-editor-contents h2 {
+ min-height: 23px;
+}
+
+.te-preview .tui-editor-contents blockquote {
+ min-height: 20px;
+}
+
+.te-preview .tui-editor-contents li {
+ min-height: 22px;
+}
+
+@media all and (-ms-high-contrast: none), (-ms-high-contrast: active) {
+ /* IE10+11 */
+ .te-ww-container .tui-editor-contents li {
+ vertical-align: middle;
+ }
+
+ .te-ww-container .tui-editor-contents ul > li::before,
+ .te-ww-container .tui-editor-contents ol > li::before,
+ .te-ww-container .tui-editor-contents .task-list-item:before {
+ position: static;
+ vertical-align: middle;
+ }
+
+ .te-ww-container .tui-editor-contents ul > li::before {
+ margin-top: -3px;
+ margin-right: 12px;
+ }
+
+ .te-ww-container .tui-editor-contents ol > li::before {
+ margin-right: 6px;
+ }
+
+ .te-ww-container .tui-editor-contents .task-list-item {
+ padding-left: 2px;
+ }
+}
+
diff --git a/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer.js b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer.js
new file mode 100644
index 0000000000..7a89d894fa
--- /dev/null
+++ b/modules/blogging/app/Volo.BloggingTestApp/wwwroot/libs/tui-editor/toastui-editor-viewer.js
@@ -0,0 +1,6878 @@
+/*!
+ * @toast-ui/editor
+ * @version 2.5.1 | Tue Nov 24 2020
+ * @author NHN FE Development Lab
+ * @license MIT
+ */
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory();
+ else if(typeof define === 'function' && define.amd)
+ define([], factory);
+ else if(typeof exports === 'object')
+ exports["Editor"] = factory();
+ else
+ root["toastui"] = root["toastui"] || {}, root["toastui"]["Editor"] = factory();
+})(window, function() {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
+/******/ }
+/******/ };
+/******/
+/******/ // define __esModule on exports
+/******/ __webpack_require__.r = function(exports) {
+/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
+/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
+/******/ }
+/******/ Object.defineProperty(exports, '__esModule', { value: true });
+/******/ };
+/******/
+/******/ // create a fake namespace object
+/******/ // mode & 1: value is a module id, require it
+/******/ // mode & 2: merge all properties of value into the ns
+/******/ // mode & 4: return value when already ns object
+/******/ // mode & 8|1: behave like require
+/******/ __webpack_require__.t = function(value, mode) {
+/******/ if(mode & 1) value = __webpack_require__(value);
+/******/ if(mode & 8) return value;
+/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
+/******/ var ns = Object.create(null);
+/******/ __webpack_require__.r(ns);
+/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
+/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
+/******/ return ns;
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 60);
+/******/ })
+/************************************************************************/
+/******/ ([
+/* 0 */
+/***/ (function(module, __webpack_exports__, __webpack_require__) {
+
+"use strict";
+/* harmony import */ var tui_code_snippet_collection_toArray__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(3);
+/* harmony import */ var tui_code_snippet_collection_toArray__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_collection_toArray__WEBPACK_IMPORTED_MODULE_0__);
+/* harmony import */ var tui_code_snippet_type_isUndefined__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(8);
+/* harmony import */ var tui_code_snippet_type_isUndefined__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_type_isUndefined__WEBPACK_IMPORTED_MODULE_1__);
+/* harmony import */ var tui_code_snippet_type_isString__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(9);
+/* harmony import */ var tui_code_snippet_type_isString__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_type_isString__WEBPACK_IMPORTED_MODULE_2__);
+/* harmony import */ var tui_code_snippet_domUtil_css__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(4);
+/* harmony import */ var tui_code_snippet_domUtil_css__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_domUtil_css__WEBPACK_IMPORTED_MODULE_3__);
+/* harmony import */ var tui_code_snippet_domUtil_addClass__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(5);
+/* harmony import */ var tui_code_snippet_domUtil_addClass__WEBPACK_IMPORTED_MODULE_4___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_domUtil_addClass__WEBPACK_IMPORTED_MODULE_4__);
+/* harmony import */ var tui_code_snippet_domUtil_removeClass__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(6);
+/* harmony import */ var tui_code_snippet_domUtil_removeClass__WEBPACK_IMPORTED_MODULE_5___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_domUtil_removeClass__WEBPACK_IMPORTED_MODULE_5__);
+/* harmony import */ var tui_code_snippet_domUtil_hasClass__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(16);
+/* harmony import */ var tui_code_snippet_domUtil_hasClass__WEBPACK_IMPORTED_MODULE_6___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_domUtil_hasClass__WEBPACK_IMPORTED_MODULE_6__);
+/* harmony import */ var tui_code_snippet_domUtil_matches__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(13);
+/* harmony import */ var tui_code_snippet_domUtil_matches__WEBPACK_IMPORTED_MODULE_7___default = /*#__PURE__*/__webpack_require__.n(tui_code_snippet_domUtil_matches__WEBPACK_IMPORTED_MODULE_7__);
+/**
+ * @fileoverview DOM Utils
+ * @author NHN FE Development Lab
+ */
+
+
+
+
+
+
+
+
+var FIND_ZWB = /\u200B/g;
+var _window = window,
+ getComputedStyle = _window.getComputedStyle;
+/**
+ * Check if node is text node
+ * @param {Node} node node to check
+ * @returns {boolean} result
+ * @ignore
+ */
+
+var isTextNode = function isTextNode(node) {
+ return node && node.nodeType === Node.TEXT_NODE;
+};
+/**
+ * Check if node is element node
+ * @param {Node} node node to check
+ * @returns {boolean} result
+ * @ignore
+ */
+
+
+var isElemNode = function isElemNode(node) {
+ return node && node.nodeType === Node.ELEMENT_NODE;
+};
+/**
+ * Check that the node is block node
+ * @param {Node} node node
+ * @returns {boolean}
+ * @ignore
+ */
+
+
+var isBlockNode = function isBlockNode(node) {
+ return /^(ADDRESS|ARTICLE|ASIDE|BLOCKQUOTE|DETAILS|DIALOG|DD|DIV|DL|DT|FIELDSET|FIGCAPTION|FIGURE|FOOTER|FORM|H[\d]|HEADER|HGROUP|HR|LI|MAIN|NAV|OL|P|PRE|SECTION|UL)$/gi.test(this.getNodeName(node));
+};
+/**
+ * Get node name of node
+ * @param {Node} node node
+ * @returns {string} node name
+ * @ignore
+ */
+
+
+var getNodeName = function getNodeName(node) {
+ if (isElemNode(node)) {
+ return node.tagName;
+ }
+
+ return 'TEXT';
+};
+/**
+ * Get node offset length of node(for Range API)
+ * @param {Node} node node
+ * @returns {number} length
+ * @ignore
+ */
+
+
+var getTextLength = function getTextLength(node) {
+ var len;
+
+ if (isElemNode(node)) {
+ len = node.textContent.replace(FIND_ZWB, '').length;
+ } else if (isTextNode(node)) {
+ len = node.nodeValue.replace(FIND_ZWB, '').length;
+ }
+
+ return len;
+};
+/**
+ * Get node offset length of node(for Range API)
+ * @param {Node} node node
+ * @returns {number} length
+ * @ignore
+ */
+
+
+var getOffsetLength = function getOffsetLength(node) {
+ var len;
+
+ if (isElemNode(node)) {
+ len = node.childNodes.length;
+ } else if (isTextNode(node)) {
+ len = node.nodeValue.replace(FIND_ZWB, '').length;
+ }
+
+ return len;
+};
+/**
+ * get node offset between parent's childnodes
+ * @param {Node} node node
+ * @returns {number} offset(index)
+ * @ignore
+ */
+
+
+var getNodeOffsetOfParent = function getNodeOffsetOfParent(node) {
+ var childNodesOfParent = node.parentNode.childNodes;
+ var i, t, found;
+
+ for (i = 0, t = childNodesOfParent.length; i < t; i += 1) {
+ if (childNodesOfParent[i] === node) {
+ found = i;
+ break;
+ }
+ }
+
+ return found;
+};
+/**
+ * get child node by offset
+ * @param {Node} node node
+ * @param {number} index offset index
+ * @returns {Node} foudned node
+ * @ignore
+ */
+
+
+var getChildNodeByOffset = function getChildNodeByOffset(node, index) {
+ var currentNode;
+
+ if (isTextNode(node)) {
+ currentNode = node;
+ } else if (node.childNodes.length && index >= 0) {
+ currentNode = node.childNodes[index];
+ }
+
+ return currentNode;
+};
+/**
+ * find next node from passed node
+ * @param {strong} direction previous or next
+ * @param {Node} node node
+ * @param {string} untilNodeName parent node name to limit
+ * @returns {Node} founded node
+ * @ignore
+ */
+
+
+var getNodeWithDirectionUntil = function getNodeWithDirectionUntil(direction, node, untilNodeName) {
+ var directionKey = direction + "Sibling";
+ var nodeName, foundedNode;
+
+ while (node && !node[directionKey]) {
+ nodeName = getNodeName(node.parentNode);
+
+ if (nodeName === untilNodeName || nodeName === 'BODY') {
+ break;
+ }
+
+ node = node.parentNode;
+ }
+
+ if (node[directionKey]) {
+ foundedNode = node[directionKey];
+ }
+
+ return foundedNode;
+};
+/**
+ * get prev node of childnode pointed with index
+ * @param {Node} node node
+ * @param {number} index offset index
+ * @param {string} untilNodeName parent node name to limit
+ * @returns {Node} founded node
+ * @ignore
+ */
+
+
+var getPrevOffsetNodeUntil = function getPrevOffsetNodeUntil(node, index, untilNodeName) {
+ var prevNode;
+
+ if (index > 0) {
+ prevNode = getChildNodeByOffset(node, index - 1);
+ } else {
+ prevNode = getNodeWithDirectionUntil('previous', node, untilNodeName);
+ }
+
+ return prevNode;
+};
+
+var getParentUntilBy = function getParentUntilBy(node, matchCondition, stopCondition) {
+ while (node.parentNode && !matchCondition(node.parentNode)) {
+ node = node.parentNode;
+
+ if (stopCondition && stopCondition(node)) {
+ break;
+ }
+ }
+
+ if (matchCondition(node.parentNode)) {
+ return node;
+ }
+
+ return null;
+};
+/**
+ * get parent node until paseed node name
+ * @param {Node} node node
+ * @param {string|HTMLNode} untilNode node name or node to limit
+ * @returns {Node} founded node
+ * @ignore
+ */
+
+
+var getParentUntil = function getParentUntil(node, untilNode) {
+ var foundedNode;
+
+ if (tui_code_snippet_type_isString__WEBPACK_IMPORTED_MODULE_2___default()(untilNode)) {
+ foundedNode = getParentUntilBy(node, function (targetNode) {
+ return untilNode === getNodeName(targetNode);
+ });
+ } else {
+ foundedNode = getParentUntilBy(node, function (targetNode) {
+ return untilNode === targetNode;
+ });
+ }
+
+ return foundedNode;
+};
+/**
+ * get node on the given direction under given parent
+ * @param {strong} direction previous or next
+ * @param {Node} node node
+ * @param {string|Node} underNode parent node name to limit
+ * @returns {Node} founded node
+ * @ignore
+ */
+
+
+var getNodeWithDirectionUnderParent = function getNodeWithDirectionUnderParent(direction, node, underNode) {
+ var directionKey = direction + "Sibling";
+ var foundedNode;
+ node = getParentUntil(node, underNode);
+
+ if (node && node[directionKey]) {
+ foundedNode = node[directionKey];
+ }
+
+ return foundedNode;
+};
+/**
+ * get top previous top level node under given node
+ * @param {Node} node node
+ * @param {Node} underNode underNode
+ * @returns {Node} founded node
+ * @ignore
+ */
+
+
+var getTopPrevNodeUnder = function getTopPrevNodeUnder(node, underNode) {
+ return getNodeWithDirectionUnderParent('previous', node, underNode);
+};
+/**
+ * get next top level block node
+ * @param {Node} node node
+ * @param {Node} underNode underNode
+ * @returns {Node} founded node
+ * @ignore
+ */
+
+
+var getTopNextNodeUnder = function getTopNextNodeUnder(node, underNode) {
+ return getNodeWithDirectionUnderParent('next', node, underNode);
+};
+/**
+ * Get parent element the body element
+ * @param {Node} node Node for start searching
+ * @returns {Node}
+ * @ignore
+ */
+
+
+var getTopBlockNode = function getTopBlockNode(node) {
+ return getParentUntil(node, 'BODY');
+};
+/**
+ * Get previous text node
+ * @param {Node} node Node for start searching
+ * @returns {Node}
+ * @ignore
+ */
+
+
+var getPrevTextNode = function getPrevTextNode(node) {
+ node = node.previousSibling || node.parentNode;
+
+ while (!isTextNode(node) && getNodeName(node) !== 'BODY') {
+ if (node.previousSibling) {
+ node = node.previousSibling;
+
+ while (node.lastChild) {
+ node = node.lastChild;
+ }
+ } else {
+ node = node.parentNode;
+ }
+ }
+
+ if (getNodeName(node) === 'BODY') {
+ node = null;
+ }
+
+ return node;
+};
+/**
+ * test whether root contains the given node
+ * @param {HTMLNode|string} root - root node
+ * @param {HTMLNode} found - node to test
+ * @returns {Boolean} true if root contains node
+ * @ignore
+ */
+
+
+var containsNode = function containsNode(root, node) {
+ var walker = document.createTreeWalker(root, 4, null, false);
+ var found = root === node;
+
+ while (!found && walker.nextNode()) {
+ found = walker.currentNode === node;
+ }
+
+ return found;
+};
+/**
+ * find node by offset
+ * @param {HTMLElement} root Root element
+ * @param {Array.} offsetList offset list
+ * @param {function} textNodeFilter Text node filter
+ * @returns {Array}
+ * @ignore
+ */
+
+
+var findOffsetNode = function findOffsetNode(root, offsetList, textNodeFilter) {
+ var result = [];
+ var text = '';
+ var walkerOffset = 0;
+ var newWalkerOffset;
+
+ if (!offsetList.length) {
+ return result;
+ }
+
+ var offset = offsetList.shift();
+ var walker = document.createTreeWalker(root, 4, null, false);
+
+ while (walker.nextNode()) {
+ text = walker.currentNode.nodeValue || '';
+
+ if (textNodeFilter) {
+ text = textNodeFilter(text);
+ }
+
+ newWalkerOffset = walkerOffset + text.length;
+
+ while (newWalkerOffset >= offset) {
+ result.push({
+ container: walker.currentNode,
+ offsetInContainer: offset - walkerOffset,
+ offset: offset
+ });
+
+ if (!offsetList.length) {
+ return result;
+ }
+
+ offset = offsetList.shift();
+ }
+
+ walkerOffset = newWalkerOffset;
+ } // there should be offset left
+
+
+ do {
+ result.push({
+ container: walker.currentNode,
+ offsetInContainer: text.length,
+ offset: offset
+ });
+ offset = offsetList.shift();
+ } while (!tui_code_snippet_type_isUndefined__WEBPACK_IMPORTED_MODULE_1___default()(offset));
+
+ return result;
+};
+
+var getNodeInfo = function getNodeInfo(node) {
+ var path = {};
+ path.tagName = node.nodeName;
+
+ if (node.id) {
+ path.id = node.id;
+ }
+
+ var className = node.className.trim();
+
+ if (className) {
+ path.className = className;
+ }
+
+ return path;
+};
+
+var getPath = function getPath(node, root) {
+ var paths = [];
+
+ while (node && node !== root) {
+ if (isElemNode(node)) {
+ paths.unshift(getNodeInfo(node));
+ }
+
+ node = node.parentNode;
+ }
+
+ return paths;
+};
+/**
+ * Find next, previous TD or TH element by given TE element
+ * @param {HTMLElement} node TD element
+ * @param {string} direction 'next' or 'previous'
+ * @returns {HTMLElement|null}
+ * @ignore
+ */
+
+
+var getTableCellByDirection = function getTableCellByDirection(node, direction) {
+ var targetElement = null;
+
+ if (!tui_code_snippet_type_isUndefined__WEBPACK_IMPORTED_MODULE_1___default()(direction) && (direction === 'next' || direction === 'previous')) {
+ if (direction === 'next') {
+ targetElement = node.nextElementSibling;
+ } else {
+ targetElement = node.previousElementSibling;
+ }
+ }
+
+ return targetElement;
+};
+/**
+ * Find sibling TR's TD element by given TD and direction
+ * @param {HTMLElement} node TD element
+ * @param {string} direction Boolean value for find first TD in next line
+ * @param {boolean} [needEdgeCell=false] Boolean value for find first TD in next line
+ * @returns {HTMLElement|null}
+ * @ignore
+ */
+
+
+var getSiblingRowCellByDirection = function getSiblingRowCellByDirection(node, direction, needEdgeCell) {
+ var tableCellElement = null;
+ var index, targetRowElement, currentContainer, siblingContainer, isSiblingContainerExists;
+
+ if (!tui_code_snippet_type_isUndefined__WEBPACK_IMPORTED_MODULE_1___default()(direction) && (direction === 'next' || direction === 'previous')) {
+ if (node) {
+ if (direction === 'next') {
+ targetRowElement = node.parentNode && node.parentNode.nextSibling;
+ currentContainer = parents(node, 'thead');
+ siblingContainer = currentContainer[0] && currentContainer[0].nextSibling;
+ isSiblingContainerExists = siblingContainer && getNodeName(siblingContainer) === 'TBODY';
+ index = 0;
+ } else {
+ targetRowElement = node.parentNode && node.parentNode.previousSibling;
+ currentContainer = parents(node, 'tbody');
+ siblingContainer = currentContainer[0] && currentContainer[0].previousSibling;
+ isSiblingContainerExists = siblingContainer && getNodeName(siblingContainer) === 'THEAD';
+ index = node.parentNode.childNodes.length - 1;
+ }
+
+ if (tui_code_snippet_type_isUndefined__WEBPACK_IMPORTED_MODULE_1___default()(needEdgeCell) || !needEdgeCell) {
+ index = getNodeOffsetOfParent(node);
+ }
+
+ if (targetRowElement) {
+ tableCellElement = children(targetRowElement, 'td,th')[index];
+ } else if (currentContainer[0] && isSiblingContainerExists) {
+ tableCellElement = findAll(siblingContainer, 'td,th')[index];
+ }
+ }
+ }
+
+ return tableCellElement;
+};
+/**
+ * Check that the inline node is supported by markdown
+ * @param {Node} node TD element
+ * @returns {boolean}
+ * @ignore
+ */
+
+
+var isMDSupportInlineNode = function isMDSupportInlineNode(node) {
+ return /^(A|B|BR|CODE|DEL|EM|I|IMG|S|SPAN|STRONG)$/gi.test(node.nodeName);
+};
+/**
+ * Check that node is styled node.
+ * Styled node is a node that has text and decorates text.
+ * @param {Node} node TD element
+ * @returns {boolean}
+ * @ignore
+ */
+
+
+var isStyledNode = function isStyledNode(node) {
+ return /^(A|ABBR|ACRONYM|B|BDI|BDO|BIG|CITE|CODE|DEL|DFN|EM|I|INS|KBD|MARK|Q|S|SAMP|SMALL|SPAN|STRONG|SUB|SUP|U|VAR)$/gi.test(node.nodeName);
+};
+/**
+ * remove node from 'start' node to 'end-1' node inside parent
+ * if 'end' node is null, remove all child nodes after 'start' node.
+ * @param {Node} parentNode - parent node
+ * @param {Node} start - start node to remove
+ * @param {Node} end - end node to remove
+ * @ignore
+ */
+
+
+var removeChildFromStartToEndNode = function removeChildFromStartToEndNode(parentNode, start, end) {
+ var child = start;
+
+ if (!child || parentNode !== child.parentNode) {
+ return;
+ }
+
+ while (child !== end) {
+ var nextNode = child.nextSibling;
+ parentNode.removeChild(child);
+ child = nextNode;
+ }
+};
+/**
+ * remove nodes along the direction from the node to reach targetParent node
+ * @param {Node} targetParent - stop removing when reach target parent node
+ * @param {Node} node - start node
+ * @param {boolean} isForward - direction
+ * @ignore
+ */
+
+
+var removeNodesByDirection = function removeNodesByDirection(targetParent, node, isForward) {
+ var parentNode = node;
+
+ while (parentNode !== targetParent) {
+ var nextParent = parentNode.parentNode;
+ var _parentNode = parentNode,
+ nextSibling = _parentNode.nextSibling,
+ previousSibling = _parentNode.previousSibling;
+
+ if (!isForward && nextSibling) {
+ removeChildFromStartToEndNode(nextParent, nextSibling, null);
+ } else if (isForward && previousSibling) {
+ removeChildFromStartToEndNode(nextParent, nextParent.childNodes[0], parentNode);
+ }
+
+ parentNode = nextParent;
+ }
+};
+
+var getLeafNode = function getLeafNode(node) {
+ var result = node;
+
+ while (result.childNodes && result.childNodes.length) {
+ var _result = result,
+ nextLeaf = _result.firstChild; // When inline tag have empty text node with other childnodes, ignore empty text node.
+
+ if (isTextNode(nextLeaf) && !getTextLength(nextLeaf)) {
+ result = nextLeaf.nextSibling || nextLeaf;
+ } else {
+ result = nextLeaf;
+ }
+ }
+
+ return result;
+};
+/**
+ * check if a coordinates is inside a button box
+ * @param {object} style - computed style of task box
+ * @param {number} offsetX - event x offset
+ * @param {number} offsetY - event y offset
+ * @returns {boolean}
+ * @ignore
+ */
+
+
+var isInsideButtonBox = function isInsideButtonBox(style, offsetX, offsetY) {
+ var rect = {
+ left: parseInt(style.left, 10),
+ top: parseInt(style.top, 10),
+ width: parseInt(style.width, 10),
+ height: parseInt(style.height, 10)
+ };
+ return offsetX >= rect.left && offsetX <= rect.left + rect.width && offsetY >= rect.top && offsetY <= rect.top + rect.height;
+};
+/**
+ * Check whether node is OL or UL
+ * @param {node} node - node
+ * @returns {boolean} - whether node is OL or UL
+ * @ignore
+ */
+
+
+var isListNode = function isListNode(node) {
+ if (!node) {
+ return false;
+ }
+
+ return node.nodeName === 'UL' || node.nodeName === 'OL';
+};
+/**
+ * Check whether node is first list item
+ * @param {node} node - node
+ * @returns {boolean} whether node is first list item
+ * @ignore
+ */
+
+
+var isFirstListItem = function isFirstListItem(node) {
+ var nodeName = node.nodeName,
+ parentNode = node.parentNode;
+ return nodeName === 'LI' && node === parentNode.firstChild;
+};
+/**
+ * Check whether node is first level list item
+ * @param {node} node - node
+ * @returns {boolean} whether node is first level list item
+ * @ignore
+ */
+
+
+var isFirstLevelListItem = function isFirstLevelListItem(node) {
+ var nodeName = node.nodeName,
+ listNode = node.parentNode;
+ var listParentNode = listNode.parentNode;
+ return nodeName === 'LI' && !isListNode(listParentNode);
+};
+/**
+ * Merge node to target node and detach node
+ * @param {node} node - node
+ * @param {node} targetNode - target node
+ * @ignore
+ */
+
+
+var mergeNode = function mergeNode(node, targetNode) {
+ if (node.hasChildNodes()) {
+ tui_code_snippet_collection_toArray__WEBPACK_IMPORTED_MODULE_0___default()(node.childNodes).forEach(function () {
+ targetNode.appendChild(node.firstChild);
+ });
+ targetNode.normalize();
+ }
+
+ if (node.parentNode) {
+ node.parentNode.removeChild(node);
+ }
+};
+/**
+ * Create hr that is not contenteditable
+ * @returns {node} hr is wraped div
+ * @ignore
+ */
+
+
+var createHorizontalRule = function createHorizontalRule() {
+ var div = document.createElement('div');
+ var hr = document.createElement('hr');
+ div.setAttribute('contenteditable', false);
+ hr.setAttribute('contenteditable', false);
+ div.appendChild(hr);
+ return div;
+};
+/**
+ * Create Empty Line
+ * @returns {node}