diff --git a/Orchard.proj b/Orchard.proj index 5f1ae878d..f84846d28 100644 --- a/Orchard.proj +++ b/Orchard.proj @@ -8,6 +8,7 @@ $(MSBuildProjectDirectory)\src $(MSBuildProjectDirectory)\build $(MSBuildProjectDirectory)\buildtasks + $(LibFolder)\msbuild $(MSBuildProjectDirectory)\artifacts $(MSBuildProjectDirectory)\lib\sqlce $(ArtifactsFolder)\Source @@ -33,7 +34,7 @@ $(BUILD_NUMBER) - + @@ -156,7 +157,7 @@ - + @@ -338,6 +339,7 @@ $(MSBuildProjectDirectory)\src\**\bin\**\*; $(MSBuildProjectDirectory)\src\**\obj\**\*; $(MSBuildProjectDirectory)\**\App_Data\**\*; + $(MSBuildProjectDirectory)\**\node_modules\**\*; $(MSBuildProjectDirectory)\**\_ReSharper*\**\*; $(MSBuildProjectDirectory)\**\*.sln.cache; $(MSBuildProjectDirectory)\**\*.suo; @@ -369,8 +371,8 @@ - - + + diff --git a/lib/msbuild/ICSharpCode.SharpZipLib.dll b/lib/msbuild/ICSharpCode.SharpZipLib.dll deleted file mode 100644 index bcdb00ea3..000000000 Binary files a/lib/msbuild/ICSharpCode.SharpZipLib.dll and /dev/null differ diff --git a/lib/msbuild/Ionic.Zip.Reduced.dll b/lib/msbuild/Ionic.Zip.Reduced.dll new file mode 100644 index 000000000..9622cc53e Binary files /dev/null and b/lib/msbuild/Ionic.Zip.Reduced.dll differ diff --git a/lib/msbuild/MSBuild.Community.Tasks.Targets b/lib/msbuild/MSBuild.Community.Tasks.Targets index a04f036a0..b7ad52d62 100644 --- a/lib/msbuild/MSBuild.Community.Tasks.Targets +++ b/lib/msbuild/MSBuild.Community.Tasks.Targets @@ -1,116 +1,158 @@ - + - - - . - $(MSBuildCommunityTasksPath)\MSBuild.Community.Tasks.dll - + + $(MSBuildThisFileDirectory) + $([MSBUILD]::Unescape($(MSBuildCommunityTasksPath)\MSBuild.Community.Tasks.dll)) + - + - - - - - - - - - - - + + + + + + + + + + + + + + + - - - + + + - - - - - + + + + + - - + + - + - - - - - - - - - - + + + + + + + + + + + - - - - - - - + + + + + + + - - + + - + - + - - - - - - - - - + + + + + + + + + - - - - - - - - + + + + + + + - + + + + + + + + + - - - - - + - - + + + + + + - - - - + + - + + + + - - + + - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/lib/msbuild/MSBuild.Community.Tasks.dll b/lib/msbuild/MSBuild.Community.Tasks.dll index ee8046a1d..1288d54c1 100644 Binary files a/lib/msbuild/MSBuild.Community.Tasks.dll and b/lib/msbuild/MSBuild.Community.Tasks.dll differ diff --git a/lib/msdeploy/createlogin.sql b/lib/msdeploy/createlogin.sql new file mode 100644 index 000000000..f9679c4e8 --- /dev/null +++ b/lib/msdeploy/createlogin.sql @@ -0,0 +1,9 @@ +/**********************************************************************/ +/* Install.SQL */ +/* Creates a login and makes the user a member of db roles */ +/* */ +/* Modifications for SQL AZURE - ON MASTER */ +/**********************************************************************/ + + +CREATE LOGIN PlaceHolderForUser WITH PASSWORD = 'PlaceHolderForPassword' \ No newline at end of file diff --git a/lib/msdeploy/createuser.sql b/lib/msdeploy/createuser.sql new file mode 100644 index 000000000..1db827770 --- /dev/null +++ b/lib/msdeploy/createuser.sql @@ -0,0 +1,15 @@ +/**********************************************************************/ +/* CreateUser.SQL */ +/* Creates a user and makes the user a member of db roles */ +/* This script runs against the User database and requires connection string */ +/* Supports SQL Server and SQL AZURE */ +/**********************************************************************/ + +-- Create database user and map to login +-- and add user to the datareader, datawriter, ddladmin and securityadmin roles +-- + +CREATE USER PlaceHolderForUser FOR LOGIN PlaceHolderForUser; +GO +EXEC sp_addrolemember 'db_owner', PlaceHolderForUser; +GO diff --git a/lib/msdeploy/install.sql b/lib/msdeploy/install.sql deleted file mode 100644 index c998bcbfc..000000000 --- a/lib/msdeploy/install.sql +++ /dev/null @@ -1,45 +0,0 @@ -/**********************************************************************/ -/* Install.SQL */ -/* Creates a login and makes the user a member of db roles */ -/* */ -/**********************************************************************/ - --- Declare variables for database name, username and password -DECLARE @dbName sysname, - @dbUser sysname, - @dbPwd nvarchar(max); - --- Set variables for database name, username and password -SET @dbName = 'PlaceHolderForDb'; -SET @dbUser = 'PlaceHolderForUser'; -SET @dbPwd = 'PlaceHolderForPassword'; - -DECLARE @cmd nvarchar(max) - --- Create login -IF( SUSER_SID(@dbUser) is null ) -BEGIN - print '-- Creating login ' - SET @cmd = N'CREATE LOGIN ' + quotename(@dbUser) + N' WITH PASSWORD ='''+ replace(@dbPwd, '''', '''''') + N'''' - EXEC(@cmd) -END - --- Create database user and map to login --- and add user to the datareader, datawriter, ddladmin and securityadmin roles --- -SET @cmd = N'USE ' + quotename(@DBName) + N'; -IF( NOT EXISTS (SELECT * FROM sys.database_principals WHERE name = ''' + replace(@dbUser, '''', '''''') + N''')) -BEGIN - print ''-- Creating user''; - CREATE USER ' + quotename(@dbUser) + N' FOR LOGIN ' + quotename(@dbUser) + N'; - print ''-- Adding user''; - EXEC sp_addrolemember ''db_ddladmin'', ''' + replace(@dbUser, '''', '''''') + N'''; - print ''-- Adding user''; - EXEC sp_addrolemember ''db_securityadmin'', ''' + replace(@dbUser, '''', '''''') + N'''; - print ''-- Adding user''; - EXEC sp_addrolemember ''db_datareader'', ''' + replace(@dbUser, '''', '''''') + N'''; - print ''-- Adding user''; - EXEC sp_addrolemember ''db_datawriter'', ''' + replace(@dbUser, '''', '''''') + N'''; -END' -EXEC(@cmd) -GO \ No newline at end of file diff --git a/lib/msdeploy/manifest.xml b/lib/msdeploy/manifest.xml index 8c0bcb819..670d2c3e3 100644 --- a/lib/msdeploy/manifest.xml +++ b/lib/msdeploy/manifest.xml @@ -1,7 +1,13 @@ - + - - + + + + + + \ No newline at end of file diff --git a/lib/msdeploy/parameters.xml b/lib/msdeploy/parameters.xml index 3a2e494c9..db78a9991 100644 --- a/lib/msdeploy/parameters.xml +++ b/lib/msdeploy/parameters.xml @@ -47,19 +47,17 @@ - + - - + + - + + + diff --git a/src/Gulpfile.js b/src/Gulpfile.js index fce38f9fe..b602f5ca4 100644 --- a/src/Gulpfile.js +++ b/src/Gulpfile.js @@ -1,4 +1,5 @@ -var glob = require("glob"), +var fs = require("fs"), + glob = require("glob"), path = require("path-posix"), merge = require("merge-stream"), gulpif = require("gulp-if"), @@ -42,14 +43,24 @@ gulp.task("watch", function () { var pathWin32 = require("path"); getAssetGroups().forEach(function (assetGroup) { var watchPaths = assetGroup.inputPaths.concat(assetGroup.watchPaths); - gulp.watch(watchPaths, function (event) { - var isConcat = path.basename(assetGroup.outputFileName, path.extname(assetGroup.outputFileName)) !== "@"; - if (isConcat) - console.log("Asset file '" + event.path + "' was " + event.type + ", rebuilding asset group with output '" + assetGroup.outputPath + "'."); - else - console.log("Asset file '" + event.path + "' was " + event.type + ", rebuilding asset group."); - var doRebuild = true; - var task = createAssetGroupTask(assetGroup, doRebuild); + var inputWatcher; + function createWatcher() { + inputWatcher = gulp.watch(watchPaths, function (event) { + var isConcat = path.basename(assetGroup.outputFileName, path.extname(assetGroup.outputFileName)) !== "@"; + if (isConcat) + console.log("Asset file '" + event.path + "' was " + event.type + ", rebuilding asset group with output '" + assetGroup.outputPath + "'."); + else + console.log("Asset file '" + event.path + "' was " + event.type + ", rebuilding asset group."); + var doRebuild = true; + var task = createAssetGroupTask(assetGroup, doRebuild); + }); + } + createWatcher(); + gulp.watch(assetGroup.manifestPath, function (event) { + console.log("Asset manifest file '" + event.path + "' was " + event.type + ", restarting watcher."); + inputWatcher.remove(); + inputWatcher.end(); + createWatcher(); }); }); }); @@ -72,6 +83,7 @@ function getAssetGroups() { } function resolveAssetGroupPaths(assetGroup, assetManifestPath) { + assetGroup.manifestPath = assetManifestPath; assetGroup.basePath = path.dirname(assetManifestPath); assetGroup.inputPaths = assetGroup.inputs.map(function (inputPath) { return path.resolve(path.join(assetGroup.basePath, inputPath)); @@ -89,11 +101,18 @@ function resolveAssetGroupPaths(assetGroup, assetManifestPath) { function createAssetGroupTask(assetGroup, doRebuild) { var outputExt = path.extname(assetGroup.output).toLowerCase(); + var doConcat = path.basename(assetGroup.outputFileName, outputExt) !== "@"; + if (doConcat && !doRebuild) { + // Force a rebuild of this asset group is the asset manifest file itself is newer than the output. + var assetManifestStats = fs.statSync(assetGroup.manifestPath); + var outputStats = fs.existsSync(assetGroup.outputPath) ? fs.statSync(assetGroup.outputPath) : null; + doRebuild = !outputStats || assetManifestStats.mtime > outputStats.mtime; + } switch (outputExt) { case ".css": - return buildCssPipeline(assetGroup, doRebuild); + return buildCssPipeline(assetGroup, doConcat, doRebuild); case ".js": - return buildJsPipeline(assetGroup, doRebuild); + return buildJsPipeline(assetGroup, doConcat, doRebuild); } } @@ -101,13 +120,12 @@ function createAssetGroupTask(assetGroup, doRebuild) { ** PROCESSING PIPELINES */ -function buildCssPipeline(assetGroup, doRebuild) { +function buildCssPipeline(assetGroup, doConcat, doRebuild) { assetGroup.inputPaths.forEach(function (inputPath) { var ext = path.extname(inputPath).toLowerCase(); if (ext !== ".less" && ext !== ".css") throw "Input file '" + inputPath + "' is not of a valid type for output file '" + assetGroup.outputPath + "'."; }); - var doConcat = path.basename(assetGroup.outputFileName, ".css") !== "@"; var generateSourceMaps = assetGroup.hasOwnProperty("generateSourceMaps") ? assetGroup.generateSourceMaps : true; return gulp.src(assetGroup.inputPaths) .pipe(gulpif(!doRebuild, @@ -138,13 +156,12 @@ function buildCssPipeline(assetGroup, doRebuild) { .pipe(gulp.dest(assetGroup.outputDir)); } -function buildJsPipeline(assetGroup, doRebuild) { +function buildJsPipeline(assetGroup, doConcat, doRebuild) { assetGroup.inputPaths.forEach(function (inputPath) { var ext = path.extname(inputPath).toLowerCase(); if (ext !== ".ts" && ext !== ".js") throw "Input file '" + inputPath + "' is not of a valid type for output file '" + assetGroup.outputPath + "'."; }); - var doConcat = path.basename(assetGroup.outputFileName, ".js") !== "@"; var generateSourceMaps = assetGroup.hasOwnProperty("generateSourceMaps") ? assetGroup.generateSourceMaps : true; return gulp.src(assetGroup.inputPaths) .pipe(gulpif(!doRebuild, diff --git a/src/Orchard.Azure/.vs/config/applicationhost.config b/src/Orchard.Azure/.vs/config/applicationhost.config new file mode 100644 index 000000000..c5660c9a1 --- /dev/null +++ b/src/Orchard.Azure/.vs/config/applicationhost.config @@ -0,0 +1,1038 @@ + + + + + + + + +
+
+
+
+
+
+
+
+ + + +
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ +
+
+ +
+
+
+
+
+
+ +
+
+
+
+
+ +
+
+
+ +
+
+ +
+
+ +
+
+
+ + +
+
+
+
+
+
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Orchard.Azure/Orchard.Azure.CloudService/Orchard.Azure.CloudService.ccproj b/src/Orchard.Azure/Orchard.Azure.CloudService/Orchard.Azure.CloudService.ccproj index 66bb61da8..0008ce829 100644 --- a/src/Orchard.Azure/Orchard.Azure.CloudService/Orchard.Azure.CloudService.ccproj +++ b/src/Orchard.Azure/Orchard.Azure.CloudService/Orchard.Azure.CloudService.ccproj @@ -35,8 +35,9 @@ + - + diff --git a/src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.cscfg b/src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.Cloud.cscfg similarity index 99% rename from src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.cscfg rename to src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.Cloud.cscfg index f61f4bfc9..be5261216 100644 --- a/src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.cscfg +++ b/src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.Cloud.cscfg @@ -1,21 +1,21 @@ - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.Local.cscfg b/src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.Local.cscfg new file mode 100644 index 000000000..be5261216 --- /dev/null +++ b/src/Orchard.Azure/Orchard.Azure.CloudService/ServiceConfiguration.Local.cscfg @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Orchard.Azure/Orchard.Azure.CloudService/ecf/Orchard.Azure.WebContent/diagnostics.wadcfgx b/src/Orchard.Azure/Orchard.Azure.CloudService/ecf/Orchard.Azure.WebContent/diagnostics.wadcfgx new file mode 100644 index 000000000..99fad34fd --- /dev/null +++ b/src/Orchard.Azure/Orchard.Azure.CloudService/ecf/Orchard.Azure.WebContent/diagnostics.wadcfgx @@ -0,0 +1,31 @@ + + + + + + + + + + + + + + + + + + + + + + + + + devstoreaccount1 + + + + + true + \ No newline at end of file diff --git a/src/Orchard.Azure/Orchard.Azure.Web/Config/Host.config b/src/Orchard.Azure/Orchard.Azure.Web/Config/Host.config index 1593a1e2c..827c5a9ab 100644 --- a/src/Orchard.Azure/Orchard.Azure.Web/Config/Host.config +++ b/src/Orchard.Azure/Orchard.Azure.Web/Config/Host.config @@ -15,7 +15,7 @@ - + diff --git a/src/Orchard.Azure/Orchard.Azure.Web/Config/HostComponents.config b/src/Orchard.Azure/Orchard.Azure.Web/Config/HostComponents.config index 2da96521c..91ea6a690 100644 --- a/src/Orchard.Azure/Orchard.Azure.Web/Config/HostComponents.config +++ b/src/Orchard.Azure/Orchard.Azure.Web/Config/HostComponents.config @@ -1,98 +1,110 @@  - + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - - - + + + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - + + + + + + - - - - - - - + + + + + + + + + + + + + + + + + + diff --git a/src/Orchard.Azure/Orchard.Azure.Web/Config/Sites.config b/src/Orchard.Azure/Orchard.Azure.Web/Config/Sites.config index a9b061b5c..52ff25006 100644 --- a/src/Orchard.Azure/Orchard.Azure.Web/Config/Sites.config +++ b/src/Orchard.Azure/Orchard.Azure.Web/Config/Sites.config @@ -1,31 +1,50 @@  - -
- + +
+ + + - - + + - - - + Uncomment to use ReadUncommitted as the default isolation level. Please not that + Sql Server Ce doesn't support ReadUncommitted. - \ No newline at end of file + Isolation level for all database transaction. + See http://msdn.microsoft.com/en-us/library/system.transactions.isolationlevel.aspx + --> + + + + + + + diff --git a/src/Orchard.Azure/Orchard.Azure.Web/Orchard.Azure.Web.csproj b/src/Orchard.Azure/Orchard.Azure.Web/Orchard.Azure.Web.csproj index cff6ebb0b..aceb7edb8 100644 --- a/src/Orchard.Azure/Orchard.Azure.Web/Orchard.Azure.Web.csproj +++ b/src/Orchard.Azure/Orchard.Azure.Web/Orchard.Azure.Web.csproj @@ -20,7 +20,7 @@ - true + true @@ -91,11 +91,11 @@ False ..\..\..\lib\windowsazure\Microsoft.WindowsAzure.Configuration.dll - + ..\..\..\lib\windowsazure\Microsoft.WindowsAzure.Diagnostics.dll True - + ..\..\..\lib\windowsazure\Microsoft.WindowsAzure.ServiceRuntime.dll False diff --git a/src/Orchard.Core.Tests/Properties/AssemblyInfo.cs b/src/Orchard.Core.Tests/Properties/AssemblyInfo.cs index b36a0684f..eb3081530 100644 --- a/src/Orchard.Core.Tests/Properties/AssemblyInfo.cs +++ b/src/Orchard.Core.Tests/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Profile/Properties/AssemblyInfo.cs b/src/Orchard.Profile/Properties/AssemblyInfo.cs index c23f2ddb7..c26e4ba29 100644 --- a/src/Orchard.Profile/Properties/AssemblyInfo.cs +++ b/src/Orchard.Profile/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Specs/Autoroute.feature b/src/Orchard.Specs/Autoroute.feature index 605d324ec..35311ab19 100644 --- a/src/Orchard.Specs/Autoroute.feature +++ b/src/Orchard.Specs/Autoroute.feature @@ -19,7 +19,7 @@ Scenario: I can create and publish a new Home Page And I fill in | name | value | | Title.Title | Foo | - | Autoroute.PromoteToHomePage | True | + | AutoroutePart.PromoteToHomePage | True | And I hit "Publish Now" And I go to "/" Then I should see "]*>.*?Foo.*?" diff --git a/src/Orchard.Specs/Autoroute.feature.cs b/src/Orchard.Specs/Autoroute.feature.cs index 9acc565f4..c21be5298 100644 --- a/src/Orchard.Specs/Autoroute.feature.cs +++ b/src/Orchard.Specs/Autoroute.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -114,7 +114,7 @@ this.ScenarioSetup(scenarioInfo); "Title.Title", "Foo"}); table2.AddRow(new string[] { - "Autoroute.PromoteToHomePage", + "AutoroutePart.PromoteToHomePage", "True"}); #line 19 testRunner.And("I fill in", ((string)(null)), table2, "And "); diff --git a/src/Orchard.Specs/Blogs.feature b/src/Orchard.Specs/Blogs.feature index c3e87d12f..237384d14 100644 --- a/src/Orchard.Specs/Blogs.feature +++ b/src/Orchard.Specs/Blogs.feature @@ -67,7 +67,7 @@ Scenario: I can create a new blog with multiple blog posts each with the same ti And I fill in | name | value | | Title.Title | My Post | - | Autoroute.CurrentUrl | my-blog/my-post | + | AutoroutePart.CurrentUrl | my-blog/my-post | | Body.Text | Are you still there? | And I hit "Publish Now" And I go to "my-blog/my-post-3" @@ -141,7 +141,7 @@ Scenario: I set my blog to be the content for the home page and the posts for th And I fill in | name | value | | Title.Title | My Blog | - | Autoroute.PromoteToHomePage | true | + | AutoroutePart.PromoteToHomePage | true | And I hit "Save" And I go to "admin/blogs" And I follow "My Blog" @@ -275,20 +275,20 @@ Scenario: I can create browse blog posts on several pages And I should not see "]*>.*?My Post 3.*?" Scenario: I can create a new blog with a percent sign in the title and it gets stripped out of the slug - Given I have installed Orchard - When I go to "admin/blogs/create" - And I fill in - | name | value | - | Title.Title | My Blog | - And I hit "Save" - And I go to "admin/blogs" - And I follow "My Blog" - And I follow "New Post" where class name has "primaryAction" - And I fill in - | name | value | - | Title.Title | My Post with a % Sign | - | Body.Text | Hi there. | - And I hit "Publish Now" - And I go to "my-blog/my-post-with-a-sign" - Then I should see "]*>.*?My Post with a % Sign.*?" - And I should see "Hi there." \ No newline at end of file + Given I have installed Orchard + When I go to "admin/blogs/create" + And I fill in + | name | value | + | Title.Title | My Blog | + And I hit "Save" + And I go to "admin/blogs" + And I follow "My Blog" + And I follow "New Post" where class name has "primaryAction" + And I fill in + | name | value | + | Title.Title | My Post with a % Sign | + | Body.Text | Hi there. | + And I hit "Publish Now" + And I go to "my-blog/my-post-with-a-sign" + Then I should see "]*>.*?My Post with a % Sign.*?" + And I should see "Hi there." \ No newline at end of file diff --git a/src/Orchard.Specs/Blogs.feature.cs b/src/Orchard.Specs/Blogs.feature.cs index f420e7133..74fabba3f 100644 --- a/src/Orchard.Specs/Blogs.feature.cs +++ b/src/Orchard.Specs/Blogs.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -234,7 +234,7 @@ this.ScenarioSetup(scenarioInfo); "Title.Title", "My Post"}); table6.AddRow(new string[] { - "Autoroute.CurrentUrl", + "AutoroutePart.CurrentUrl", "my-blog/my-post"}); table6.AddRow(new string[] { "Body.Text", @@ -449,7 +449,7 @@ this.ScenarioSetup(scenarioInfo); "Title.Title", "My Blog"}); table12.AddRow(new string[] { - "Autoroute.PromoteToHomePage", + "AutoroutePart.PromoteToHomePage", "true"}); #line 141 testRunner.And("I fill in", ((string)(null)), table12, "And "); @@ -779,9 +779,9 @@ this.ScenarioSetup(scenarioInfo); #line 277 this.ScenarioSetup(scenarioInfo); #line 278 - testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); + testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 279 - testRunner.When("I go to \"admin/blogs/create\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"admin/blogs/create\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table27 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -790,15 +790,15 @@ this.ScenarioSetup(scenarioInfo); "Title.Title", "My Blog"}); #line 280 - testRunner.And("I fill in", ((string)(null)), table27, "And "); + testRunner.And("I fill in", ((string)(null)), table27, "And "); #line 283 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 284 - testRunner.And("I go to \"admin/blogs\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"admin/blogs\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 285 - testRunner.And("I follow \"My Blog\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"My Blog\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 286 - testRunner.And("I follow \"New Post\" where class name has \"primaryAction\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"New Post\" where class name has \"primaryAction\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table28 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -810,15 +810,15 @@ this.ScenarioSetup(scenarioInfo); "Body.Text", "Hi there."}); #line 287 - testRunner.And("I fill in", ((string)(null)), table28, "And "); + testRunner.And("I fill in", ((string)(null)), table28, "And "); #line 291 - testRunner.And("I hit \"Publish Now\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Publish Now\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 292 - testRunner.And("I go to \"my-blog/my-post-with-a-sign\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"my-blog/my-post-with-a-sign\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 293 - testRunner.Then("I should see \"]*>.*?My Post with a % Sign.*?\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"]*>.*?My Post with a % Sign.*?\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 294 - testRunner.And("I should see \"Hi there.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"Hi there.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/Boolean.feature b/src/Orchard.Specs/Boolean.feature index 25c5a21e8..c36d964ba 100644 --- a/src/Orchard.Specs/Boolean.feature +++ b/src/Orchard.Specs/Boolean.feature @@ -1,13 +1,13 @@ Feature: Boolean Field In order to add boolean content to my types - As an administrator + As an administrator I want to create, edit and publish boolean fields Scenario: Creating and using Boolean fields - - # Creating an Event content type + + # Creating an Event content type Given I have installed Orchard - And I have installed "Orchard.Fields" + And I have installed "Orchard.Fields" When I go to "Admin/ContentTypes" Then I should see "]*>.*?Create new type" When I go to "Admin/ContentTypes/Create" @@ -18,65 +18,65 @@ Scenario: Creating and using Boolean fields And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a Boolean field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a Boolean field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | Active | | Name | Active | | FieldTypeName | BooleanField | - And I hit "Save" - And I am redirected - Then I should see "The \"Active\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"Active\" field has been added." - # Creating an Event content item - When I go to "Admin/Contents/Create/Event" - Then I should see "Active" - When I fill in - | name | value | - | Event.Active.Value | true | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." - When I go to "Admin/Contents/List" - Then I should see "Active:" - And I should see "Yes" + # Creating an Event content item + When I go to "Admin/Contents/Create/Event" + Then I should see "Active" + When I fill in + | name | value | + | Event.Active.Value | true | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." + When I go to "Admin/Contents/List" + Then I should see "Active:" + And I should see "Yes" - # The hint should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].BooleanFieldSettings.Hint | Check if the event is active | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Check if the event is active" - - # The default value should be selected - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].BooleanFieldSettings.DefaultValue | True | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "checked=\"checked\"" + # The hint should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].BooleanFieldSettings.Hint | Check if the event is active | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Check if the event is active" + + # The default value should be selected + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].BooleanFieldSettings.DefaultValue | True | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "checked=\"checked\"" - # The value should be required - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].BooleanFieldSettings.Optional | false | - And I fill in - | name | value | - | Fields[0].BooleanFieldSettings.NotSetLabel | May be | - And I fill in - | name | value | - | Fields[0].BooleanFieldSettings.SelectionMode | Radiobutton | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.Active.Value | | - And I hit "Save" - Then I should see "The field Active is mandatory." \ No newline at end of file + # The value should be required + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].BooleanFieldSettings.Optional | false | + And I fill in + | name | value | + | Fields[0].BooleanFieldSettings.NotSetLabel | May be | + And I fill in + | name | value | + | Fields[0].BooleanFieldSettings.SelectionMode | Radiobutton | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.Active.Value | | + And I hit "Save" + Then I should see "The field Active is mandatory." \ No newline at end of file diff --git a/src/Orchard.Specs/Boolean.feature.cs b/src/Orchard.Specs/Boolean.feature.cs index 6d95a877a..f074e5cb2 100644 --- a/src/Orchard.Specs/Boolean.feature.cs +++ b/src/Orchard.Specs/Boolean.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -32,8 +32,8 @@ namespace Orchard.Specs public virtual void FeatureSetup() { testRunner = TechTalk.SpecFlow.TestRunnerManager.GetTestRunner(); - TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Boolean Field", " In order to add boolean content to my types\r\nAs an administrator\r\n I want to c" + - "reate, edit and publish boolean fields", ProgrammingLanguage.CSharp, ((string[])(null))); + TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Boolean Field", " In order to add boolean content to my types\r\n As an administrator\r\n I want to" + + " create, edit and publish boolean fields", ProgrammingLanguage.CSharp, ((string[])(null))); testRunner.OnFeatureStart(featureInfo); } @@ -75,7 +75,7 @@ this.ScenarioSetup(scenarioInfo); #line 9 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 10 - testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 12 @@ -101,9 +101,9 @@ this.ScenarioSetup(scenarioInfo); #line 20 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 23 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 24 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -118,17 +118,17 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "BooleanField"}); #line 25 - testRunner.And("I fill in", ((string)(null)), table2, "And "); + testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 30 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 31 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 32 - testRunner.Then("I should see \"The \\\"Active\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"Active\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 35 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 36 - testRunner.Then("I should see \"Active\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Active\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table3 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -137,21 +137,21 @@ this.ScenarioSetup(scenarioInfo); "Event.Active.Value", "true"}); #line 37 - testRunner.When("I fill in", ((string)(null)), table3, "When "); + testRunner.When("I fill in", ((string)(null)), table3, "When "); #line 40 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 41 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 42 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 43 - testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 44 - testRunner.Then("I should see \"Active:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Active:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 45 - testRunner.And("I should see \"Yes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"Yes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 48 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table4 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -160,15 +160,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].BooleanFieldSettings.Hint", "Check if the event is active"}); #line 49 - testRunner.And("I fill in", ((string)(null)), table4, "And "); + testRunner.And("I fill in", ((string)(null)), table4, "And "); #line 52 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 53 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 54 - testRunner.Then("I should see \"Check if the event is active\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Check if the event is active\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 57 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table5 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -177,15 +177,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].BooleanFieldSettings.DefaultValue", "True"}); #line 58 - testRunner.And("I fill in", ((string)(null)), table5, "And "); + testRunner.And("I fill in", ((string)(null)), table5, "And "); #line 61 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 62 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 63 - testRunner.Then("I should see \"checked=\\\"checked\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"checked=\\\"checked\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 66 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -194,7 +194,7 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].BooleanFieldSettings.Optional", "false"}); #line 67 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -203,7 +203,7 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].BooleanFieldSettings.NotSetLabel", "May be"}); #line 70 - testRunner.And("I fill in", ((string)(null)), table7, "And "); + testRunner.And("I fill in", ((string)(null)), table7, "And "); #line hidden TechTalk.SpecFlow.Table table8 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -212,11 +212,11 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].BooleanFieldSettings.SelectionMode", "Radiobutton"}); #line 73 - testRunner.And("I fill in", ((string)(null)), table8, "And "); + testRunner.And("I fill in", ((string)(null)), table8, "And "); #line 76 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 77 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table9 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -225,11 +225,11 @@ this.ScenarioSetup(scenarioInfo); "Event.Active.Value", ""}); #line 78 - testRunner.And("I fill in", ((string)(null)), table9, "And "); + testRunner.And("I fill in", ((string)(null)), table9, "And "); #line 81 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 82 - testRunner.Then("I should see \"The field Active is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The field Active is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/Comments.feature b/src/Orchard.Specs/Comments.feature index 22ca0eb1b..16072c89f 100644 --- a/src/Orchard.Specs/Comments.feature +++ b/src/Orchard.Specs/Comments.feature @@ -39,12 +39,12 @@ Scenario: HTML markup in any given comment is encoded And I hit "Submit Comment" And I am redirected # because the ToUrlString extension method breaks in this specific (test) environment, the returnUrl is broken... - And I go to "my-blog/my-post" + And I go to "my-blog/my-post" # And I go to "my-blog/my-post" Then I should see "This is<br id="bad-anon-br" />a <a href" And I should not see "
" - # Moderated comments are not displayed + # Moderated comments are not displayed When I go to "users/account/logon" And I fill in | name | value | @@ -52,15 +52,15 @@ Scenario: HTML markup in any given comment is encoded | password | 6655321 | And I hit "Sign In" And I am redirected - And I go to "admin/settings/comments" - And I fill in + And I go to "admin/settings/comments" + And I fill in | name | value | | CommentSettings.ModerateComments | true | - And I hit "Save" - And I am redirected - Then I should see "Settings updated" - When I go to "users/account/logoff" - And I go to "my-blog/my-post" + And I hit "Save" + And I am redirected + Then I should see "Settings updated" + When I go to "users/account/logoff" + And I go to "my-blog/my-post" And I fill in | name | value | | Comments.Author | Bill | diff --git a/src/Orchard.Specs/Comments.feature.cs b/src/Orchard.Specs/Comments.feature.cs index 4c48a16bb..d7c9de2d7 100644 --- a/src/Orchard.Specs/Comments.feature.cs +++ b/src/Orchard.Specs/Comments.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.0 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -151,7 +151,7 @@ this.ScenarioSetup(scenarioInfo); #line 40 testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 42 - testRunner.And("I go to \"my-blog/my-post\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"my-blog/my-post\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 44 testRunner.Then("I should see \"This is<br id="bad-anon-br" />a <a href\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 45 @@ -175,7 +175,7 @@ this.ScenarioSetup(scenarioInfo); #line 54 testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 55 - testRunner.And("I go to \"admin/settings/comments\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"admin/settings/comments\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -184,17 +184,17 @@ this.ScenarioSetup(scenarioInfo); "CommentSettings.ModerateComments", "true"}); #line 56 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line 59 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 60 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 61 - testRunner.Then("I should see \"Settings updated\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Settings updated\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 62 - testRunner.When("I go to \"users/account/logoff\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"users/account/logoff\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 63 - testRunner.And("I go to \"my-blog/my-post\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"my-blog/my-post\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", diff --git a/src/Orchard.Specs/DateTime.feature b/src/Orchard.Specs/DateTime.feature index de1f61881..ac4c8d2d8 100644 --- a/src/Orchard.Specs/DateTime.feature +++ b/src/Orchard.Specs/DateTime.feature @@ -1,13 +1,13 @@ Feature: DateTime Field In order to add Date content to my types - As an administrator + As an administrator I want to create, edit and publish DateTime fields Scenario: Creating and using Date fields - - # Creating an Event content type + + # Creating an Event content type Given I have installed Orchard - And I have installed "Orchard.Fields" + And I have installed "Orchard.Fields" When I go to "Admin/ContentTypes" Then I should see "]*>.*?Create new type" When I go to "Admin/ContentTypes/Create" @@ -18,133 +18,133 @@ Scenario: Creating and using Date fields And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a Date field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a Date field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | Date of the event | | Name | EventDate | | FieldTypeName | DateTimeField | - And I hit "Save" - And I am redirected - Then I should see "The \"Date of the event\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"Date of the event\" field has been added." - # Invalid Date - When I go to "Admin/Contents/Create/Event" - Then I should see "Date of the event" - When I fill in - | name | value | - | Event.EventDate.Editor.Date | 31/01/2012 | - | Event.EventDate.Editor.Time | 12:00 AM | - And I hit "Save" - Then I should see "Date of the event could not be parsed as a valid date and time" + # Invalid Date + When I go to "Admin/Contents/Create/Event" + Then I should see "Date of the event" + When I fill in + | name | value | + | Event.EventDate.Editor.Date | 31/01/2012 | + | Event.EventDate.Editor.Time | 12:00 AM | + And I hit "Save" + Then I should see "Date of the event could not be parsed as a valid date and time" - # Creating an Event content item - When I go to "Admin/Contents/Create/Event" - Then I should see "Date of the event" - When I fill in - | name | value | - | Event.EventDate.Editor.Date | 01/31/2012 | - And I fill in - | name | value | - | Event.EventDate.Editor.Time | 12:00 AM | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." - When I go to "Admin/Contents/List" - Then I should see "Date of the event" - And I should see "1/31/2012 12:00" + # Creating an Event content item + When I go to "Admin/Contents/Create/Event" + Then I should see "Date of the event" + When I fill in + | name | value | + | Event.EventDate.Editor.Date | 01/31/2012 | + And I fill in + | name | value | + | Event.EventDate.Editor.Time | 12:00 AM | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." + When I go to "Admin/Contents/List" + Then I should see "Date of the event" + And I should see "1/31/2012 12:00" - # The hint should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].DateTimeFieldSettings.Hint | Enter the date of the event | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Enter the date of the event" - - # Display = DateOnly - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].DateTimeFieldSettings.Display | DateOnly | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Event.EventDate.Editor.Date" - And I should not see "Event.EventDate.Editor.Time" - - # Display = TimeOnly - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].DateTimeFieldSettings.Display | TimeOnly | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Event.EventDate.Editor.Time" - And I should not see "Event.EventDate.Editor.Date" + # The hint should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].DateTimeFieldSettings.Hint | Enter the date of the event | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Enter the date of the event" + + # Display = DateOnly + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].DateTimeFieldSettings.Display | DateOnly | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Event.EventDate.Editor.Date" + And I should not see "Event.EventDate.Editor.Time" + + # Display = TimeOnly + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].DateTimeFieldSettings.Display | TimeOnly | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Event.EventDate.Editor.Time" + And I should not see "Event.EventDate.Editor.Date" - # Required & Date and Time - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].DateTimeFieldSettings.Display | DateAndTime | - | Fields[0].DateTimeFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Event.EventDate.Editor.Date" - When I fill in - | name | value | - | Event.EventDate.Editor.Date | 01/31/2012 | - | Event.EventDate.Editor.Time | 12:00 AM | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." - When I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.EventDate.Editor.Date | 01/31/2012 | - And I hit "Save" - Then I should see "Date of the event is required." - When I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.EventDate.Editor.Time | 12:00 AM | - And I hit "Save" - Then I should see "Date of the event is required." + # Required & Date and Time + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].DateTimeFieldSettings.Display | DateAndTime | + | Fields[0].DateTimeFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Event.EventDate.Editor.Date" + When I fill in + | name | value | + | Event.EventDate.Editor.Date | 01/31/2012 | + | Event.EventDate.Editor.Time | 12:00 AM | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." + When I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.EventDate.Editor.Date | 01/31/2012 | + And I hit "Save" + Then I should see "Date of the event is required." + When I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.EventDate.Editor.Time | 12:00 AM | + And I hit "Save" + Then I should see "Date of the event is required." - # Required & Date only - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].DateTimeFieldSettings.Display | DateOnly | - | Fields[0].DateTimeFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Event.EventDate.Editor.Date" - When I hit "Save" - Then I should see "Date of the event is required." + # Required & Date only + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].DateTimeFieldSettings.Display | DateOnly | + | Fields[0].DateTimeFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Event.EventDate.Editor.Date" + When I hit "Save" + Then I should see "Date of the event is required." - # Required & Time only - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].DateTimeFieldSettings.Display | TimeOnly | - | Fields[0].DateTimeFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Event.EventDate.Editor.Date" - When I hit "Save" - Then I should see "Date of the event is required." + # Required & Time only + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].DateTimeFieldSettings.Display | TimeOnly | + | Fields[0].DateTimeFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Event.EventDate.Editor.Date" + When I hit "Save" + Then I should see "Date of the event is required." Scenario: Creating and using date time fields in another culture - # Creating an Event content type + # Creating an Event content type Given I have installed Orchard - And I have installed "Orchard.Fields" - And I have the file "Content\orchard.core.po" in "Core\App_Data\Localization\fr-FR\orchard.core.po" + And I have installed "Orchard.Fields" + And I have the file "Content\orchard.core.po" in "Core\App_Data\Localization\fr-FR\orchard.core.po" When I go to "Admin/ContentTypes" Then I should see "]*>.*?Create new type" When I go to "Admin/ContentTypes/Create" @@ -155,39 +155,39 @@ Scenario: Creating and using date time fields in another culture And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a Date field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a Date field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | Date of the event | | Name | EventDate | | FieldTypeName | DateTimeField | - And I hit "Save" - And I am redirected - Then I should see "The \"Date of the event\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"Date of the event\" field has been added." - # Date & Time are inputted based on current culture - When I have "fr-FR" as the default culture - And I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].DateTimeFieldSettings.Display | DateAndTime | - | Fields[0].DateTimeFieldSettings.Required | true | - And I hit "Save" - When I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.EventDate.Editor.Date | 01/31/2012 | - | Event.EventDate.Editor.Time | 12:00 AM | - And I hit "Save" - Then I should see "Date of the event could not be parsed as a valid date and time" - When I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.EventDate.Editor.Date | 31/01/2012 | - | Event.EventDate.Editor.Time | 18:00 | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." + # Date & Time are inputted based on current culture + When I have "fr-FR" as the default culture + And I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].DateTimeFieldSettings.Display | DateAndTime | + | Fields[0].DateTimeFieldSettings.Required | true | + And I hit "Save" + When I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.EventDate.Editor.Date | 01/31/2012 | + | Event.EventDate.Editor.Time | 12:00 AM | + And I hit "Save" + Then I should see "Date of the event could not be parsed as a valid date and time" + When I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.EventDate.Editor.Date | 31/01/2012 | + | Event.EventDate.Editor.Time | 18:00 | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." diff --git a/src/Orchard.Specs/DateTime.feature.cs b/src/Orchard.Specs/DateTime.feature.cs index 635470931..8439bf756 100644 --- a/src/Orchard.Specs/DateTime.feature.cs +++ b/src/Orchard.Specs/DateTime.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -32,8 +32,8 @@ namespace Orchard.Specs public virtual void FeatureSetup() { testRunner = TechTalk.SpecFlow.TestRunnerManager.GetTestRunner(); - TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "DateTime Field", " In order to add Date content to my types\r\nAs an administrator\r\n I want to crea" + - "te, edit and publish DateTime fields", ProgrammingLanguage.CSharp, ((string[])(null))); + TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "DateTime Field", " In order to add Date content to my types\r\n As an administrator\r\n I want to cr" + + "eate, edit and publish DateTime fields", ProgrammingLanguage.CSharp, ((string[])(null))); testRunner.OnFeatureStart(featureInfo); } @@ -75,7 +75,7 @@ this.ScenarioSetup(scenarioInfo); #line 9 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 10 - testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 12 @@ -101,9 +101,9 @@ this.ScenarioSetup(scenarioInfo); #line 20 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 23 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 24 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -118,17 +118,17 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "DateTimeField"}); #line 25 - testRunner.And("I fill in", ((string)(null)), table2, "And "); + testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 30 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 31 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 32 - testRunner.Then("I should see \"The \\\"Date of the event\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"Date of the event\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 35 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 36 - testRunner.Then("I should see \"Date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table3 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -140,15 +140,15 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Time", "12:00 AM"}); #line 37 - testRunner.When("I fill in", ((string)(null)), table3, "When "); + testRunner.When("I fill in", ((string)(null)), table3, "When "); #line 41 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 42 - testRunner.Then("I should see \"Date of the event could not be parsed as a valid date and time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event could not be parsed as a valid date and time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 45 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 46 - testRunner.Then("I should see \"Date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table4 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -157,7 +157,7 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Date", "01/31/2012"}); #line 47 - testRunner.When("I fill in", ((string)(null)), table4, "When "); + testRunner.When("I fill in", ((string)(null)), table4, "When "); #line hidden TechTalk.SpecFlow.Table table5 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -166,21 +166,21 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Time", "12:00 AM"}); #line 50 - testRunner.And("I fill in", ((string)(null)), table5, "And "); + testRunner.And("I fill in", ((string)(null)), table5, "And "); #line 53 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 54 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 55 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 56 - testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 57 - testRunner.Then("I should see \"Date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 58 - testRunner.And("I should see \"1/31/2012 12:00\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"1/31/2012 12:00\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 61 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -189,15 +189,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].DateTimeFieldSettings.Hint", "Enter the date of the event"}); #line 62 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line 65 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 66 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 67 - testRunner.Then("I should see \"Enter the date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Enter the date of the event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 70 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -206,17 +206,17 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].DateTimeFieldSettings.Display", "DateOnly"}); #line 71 - testRunner.And("I fill in", ((string)(null)), table7, "And "); + testRunner.And("I fill in", ((string)(null)), table7, "And "); #line 74 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 75 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 76 - testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 77 - testRunner.And("I should not see \"Event.EventDate.Editor.Time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should not see \"Event.EventDate.Editor.Time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 80 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table8 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -225,17 +225,17 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].DateTimeFieldSettings.Display", "TimeOnly"}); #line 81 - testRunner.And("I fill in", ((string)(null)), table8, "And "); + testRunner.And("I fill in", ((string)(null)), table8, "And "); #line 84 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 85 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 86 - testRunner.Then("I should see \"Event.EventDate.Editor.Time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Event.EventDate.Editor.Time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 87 - testRunner.And("I should not see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should not see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 90 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table9 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -247,13 +247,13 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].DateTimeFieldSettings.Required", "true"}); #line 91 - testRunner.And("I fill in", ((string)(null)), table9, "And "); + testRunner.And("I fill in", ((string)(null)), table9, "And "); #line 95 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 96 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 97 - testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table10 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -265,15 +265,15 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Time", "12:00 AM"}); #line 98 - testRunner.When("I fill in", ((string)(null)), table10, "When "); + testRunner.When("I fill in", ((string)(null)), table10, "When "); #line 102 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 103 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 104 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 105 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table11 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -282,13 +282,13 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Date", "01/31/2012"}); #line 106 - testRunner.And("I fill in", ((string)(null)), table11, "And "); + testRunner.And("I fill in", ((string)(null)), table11, "And "); #line 109 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 110 - testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 111 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table12 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -297,13 +297,13 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Time", "12:00 AM"}); #line 112 - testRunner.And("I fill in", ((string)(null)), table12, "And "); + testRunner.And("I fill in", ((string)(null)), table12, "And "); #line 115 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 116 - testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 119 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table13 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -315,19 +315,19 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].DateTimeFieldSettings.Required", "true"}); #line 120 - testRunner.And("I fill in", ((string)(null)), table13, "And "); + testRunner.And("I fill in", ((string)(null)), table13, "And "); #line 124 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 125 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 126 - testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 127 - testRunner.When("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 128 - testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 131 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table14 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -339,17 +339,17 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].DateTimeFieldSettings.Required", "true"}); #line 132 - testRunner.And("I fill in", ((string)(null)), table14, "And "); + testRunner.And("I fill in", ((string)(null)), table14, "And "); #line 136 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 137 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 138 - testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Event.EventDate.Editor.Date\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 139 - testRunner.When("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 140 - testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event is required.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden this.ScenarioCleanup(); } @@ -364,9 +364,9 @@ this.ScenarioSetup(scenarioInfo); #line 145 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 146 - testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 147 - testRunner.And("I have the file \"Content\\orchard.core.po\" in \"Core\\App_Data\\Localization\\fr-FR\\or" + + testRunner.And("I have the file \"Content\\orchard.core.po\" in \"Core\\App_Data\\Localization\\fr-FR\\or" + "chard.core.po\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 148 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); @@ -393,9 +393,9 @@ this.ScenarioSetup(scenarioInfo); #line 157 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 160 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 161 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table16 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -410,17 +410,17 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "DateTimeField"}); #line 162 - testRunner.And("I fill in", ((string)(null)), table16, "And "); + testRunner.And("I fill in", ((string)(null)), table16, "And "); #line 167 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 168 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 169 - testRunner.Then("I should see \"The \\\"Date of the event\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"Date of the event\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 172 - testRunner.When("I have \"fr-FR\" as the default culture", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I have \"fr-FR\" as the default culture", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 173 - testRunner.And("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table17 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -432,11 +432,11 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].DateTimeFieldSettings.Required", "true"}); #line 174 - testRunner.And("I fill in", ((string)(null)), table17, "And "); + testRunner.And("I fill in", ((string)(null)), table17, "And "); #line 178 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 179 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table18 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -448,13 +448,13 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Time", "12:00 AM"}); #line 180 - testRunner.And("I fill in", ((string)(null)), table18, "And "); + testRunner.And("I fill in", ((string)(null)), table18, "And "); #line 184 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 185 - testRunner.Then("I should see \"Date of the event could not be parsed as a valid date and time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Date of the event could not be parsed as a valid date and time\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 186 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table19 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -466,13 +466,13 @@ this.ScenarioSetup(scenarioInfo); "Event.EventDate.Editor.Time", "18:00"}); #line 187 - testRunner.And("I fill in", ((string)(null)), table19, "And "); + testRunner.And("I fill in", ((string)(null)), table19, "And "); #line 191 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 192 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 193 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/Enumeration.feature b/src/Orchard.Specs/Enumeration.feature index 9201b36c4..6864b9a83 100644 --- a/src/Orchard.Specs/Enumeration.feature +++ b/src/Orchard.Specs/Enumeration.feature @@ -1,13 +1,13 @@ Feature: Enumeration Field In order to add a list of elements to my types - As an administrator + As an administrator I want to create, edit and publish Enumeration fields Scenario: Creating and using Enumeration fields - - # Creating an Event content type + + # Creating an Event content type Given I have installed Orchard - And I have installed "Orchard.Fields" + And I have installed "Orchard.Fields" When I go to "Admin/ContentTypes" Then I should see "]*>.*?Create new type" When I go to "Admin/ContentTypes/Create" @@ -18,92 +18,92 @@ Scenario: Creating and using Enumeration fields And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a Enumeration field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a Enumeration field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | Location | | Name | Location | | FieldTypeName | EnumerationField | - And I hit "Save" - And I am redirected - Then I should see "The \"Location\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"Location\" field has been added." - # Specifying Options - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].EnumerationFieldSettings.Options | Seattle | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "" + # Specifying Options + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].EnumerationFieldSettings.Options | Seattle | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "" - # Creating an Event content item - When I go to "Admin/Contents/Create/Event" - Then I should see "Location" - When I fill in - | name | value | - | Event.Location.Value | Seattle | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." - When I go to "Admin/Contents/List" - Then I should see "Location:" - And I should see "Seattle" + # Creating an Event content item + When I go to "Admin/Contents/Create/Event" + Then I should see "Location" + When I fill in + | name | value | + | Event.Location.Value | Seattle | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." + When I go to "Admin/Contents/List" + Then I should see "Location:" + And I should see "Seattle" - # The hint should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].EnumerationFieldSettings.Hint | Please select a location | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Please select a location" + # The hint should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].EnumerationFieldSettings.Hint | Please select a location | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Please select a location" - # The List Mode Dropdown - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].EnumerationFieldSettings.ListMode | Dropdown | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "select id=\"Event_Location_Value\" name=\"Event.Location.Value\"" - - # The List Mode Radiobutton - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].EnumerationFieldSettings.ListMode | Radiobutton | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "input id=\"Event_Location_Value\" name=\"Event.Location.Value\" type=\"radio\"" - - # The List Mode Listbox - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].EnumerationFieldSettings.ListMode | Listbox | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "select id=\"Event_Location_SelectedValues\" multiple=\"multiple\" name=\"Event.Location.SelectedValues\"" - - # The List Mode Checkbox - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].EnumerationFieldSettings.ListMode | Checkbox | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "input type=\"checkbox\" name=\"Event.Location.SelectedValues\"" - - # The value should be required - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].EnumerationFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - And I hit "Save" - Then I should see "The field Location is mandatory." + # The List Mode Dropdown + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].EnumerationFieldSettings.ListMode | Dropdown | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "select id=\"Event_Location_Value\" name=\"Event.Location.Value\"" + + # The List Mode Radiobutton + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].EnumerationFieldSettings.ListMode | Radiobutton | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "input id=\"Event_Location_Value\" name=\"Event.Location.Value\" type=\"radio\"" + + # The List Mode Listbox + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].EnumerationFieldSettings.ListMode | Listbox | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "select id=\"Event_Location_SelectedValues\" multiple=\"multiple\" name=\"Event.Location.SelectedValues\"" + + # The List Mode Checkbox + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].EnumerationFieldSettings.ListMode | Checkbox | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "input type=\"checkbox\" name=\"Event.Location.SelectedValues\"" + + # The value should be required + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].EnumerationFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + And I hit "Save" + Then I should see "The field Location is mandatory." diff --git a/src/Orchard.Specs/Enumeration.feature.cs b/src/Orchard.Specs/Enumeration.feature.cs index 2241d667c..88a011d4f 100644 --- a/src/Orchard.Specs/Enumeration.feature.cs +++ b/src/Orchard.Specs/Enumeration.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -32,8 +32,8 @@ namespace Orchard.Specs public virtual void FeatureSetup() { testRunner = TechTalk.SpecFlow.TestRunnerManager.GetTestRunner(); - TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Enumeration Field", " In order to add a list of elements to my types\r\nAs an administrator\r\n I want t" + - "o create, edit and publish Enumeration fields", ProgrammingLanguage.CSharp, ((string[])(null))); + TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Enumeration Field", " In order to add a list of elements to my types\r\n As an administrator\r\n I want" + + " to create, edit and publish Enumeration fields", ProgrammingLanguage.CSharp, ((string[])(null))); testRunner.OnFeatureStart(featureInfo); } @@ -75,7 +75,7 @@ this.ScenarioSetup(scenarioInfo); #line 9 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 10 - testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 12 @@ -101,9 +101,9 @@ this.ScenarioSetup(scenarioInfo); #line 20 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 23 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 24 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -118,15 +118,15 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "EnumerationField"}); #line 25 - testRunner.And("I fill in", ((string)(null)), table2, "And "); + testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 30 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 31 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 32 - testRunner.Then("I should see \"The \\\"Location\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"Location\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 35 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table3 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -135,17 +135,17 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].EnumerationFieldSettings.Options", "Seattle"}); #line 36 - testRunner.And("I fill in", ((string)(null)), table3, "And "); + testRunner.And("I fill in", ((string)(null)), table3, "And "); #line 39 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 40 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 41 - testRunner.Then("I should see \"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 44 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 45 - testRunner.Then("I should see \"Location\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Location\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table4 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -154,21 +154,21 @@ this.ScenarioSetup(scenarioInfo); "Event.Location.Value", "Seattle"}); #line 46 - testRunner.When("I fill in", ((string)(null)), table4, "When "); + testRunner.When("I fill in", ((string)(null)), table4, "When "); #line 49 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 50 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 51 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 52 - testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 53 - testRunner.Then("I should see \"Location:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Location:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 54 - testRunner.And("I should see \"Seattle\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"Seattle\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 57 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table5 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -177,15 +177,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].EnumerationFieldSettings.Hint", "Please select a location"}); #line 58 - testRunner.And("I fill in", ((string)(null)), table5, "And "); + testRunner.And("I fill in", ((string)(null)), table5, "And "); #line 61 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 62 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 63 - testRunner.Then("I should see \"Please select a location\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Please select a location\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 66 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -194,15 +194,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].EnumerationFieldSettings.ListMode", "Dropdown"}); #line 67 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line 70 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 71 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 72 - testRunner.Then("I should see \"select id=\\\"Event_Location_Value\\\" name=\\\"Event.Location.Value\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"select id=\\\"Event_Location_Value\\\" name=\\\"Event.Location.Value\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 75 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -211,16 +211,16 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].EnumerationFieldSettings.ListMode", "Radiobutton"}); #line 76 - testRunner.And("I fill in", ((string)(null)), table7, "And "); + testRunner.And("I fill in", ((string)(null)), table7, "And "); #line 79 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 80 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 81 - testRunner.Then("I should see \"input id=\\\"Event_Location_Value\\\" name=\\\"Event.Location.Value\\\" typ" + + testRunner.Then("I should see \"input id=\\\"Event_Location_Value\\\" name=\\\"Event.Location.Value\\\" typ" + "e=\\\"radio\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 84 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table8 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -229,16 +229,16 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].EnumerationFieldSettings.ListMode", "Listbox"}); #line 85 - testRunner.And("I fill in", ((string)(null)), table8, "And "); + testRunner.And("I fill in", ((string)(null)), table8, "And "); #line 88 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 89 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 90 - testRunner.Then("I should see \"select id=\\\"Event_Location_SelectedValues\\\" multiple=\\\"multiple\\\" n" + + testRunner.Then("I should see \"select id=\\\"Event_Location_SelectedValues\\\" multiple=\\\"multiple\\\" n" + "ame=\\\"Event.Location.SelectedValues\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 93 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table9 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -247,15 +247,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].EnumerationFieldSettings.ListMode", "Checkbox"}); #line 94 - testRunner.And("I fill in", ((string)(null)), table9, "And "); + testRunner.And("I fill in", ((string)(null)), table9, "And "); #line 97 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 98 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 99 - testRunner.Then("I should see \"input type=\\\"checkbox\\\" name=\\\"Event.Location.SelectedValues\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"input type=\\\"checkbox\\\" name=\\\"Event.Location.SelectedValues\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 102 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table10 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -264,15 +264,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].EnumerationFieldSettings.Required", "true"}); #line 103 - testRunner.And("I fill in", ((string)(null)), table10, "And "); + testRunner.And("I fill in", ((string)(null)), table10, "And "); #line 106 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 107 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 108 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 109 - testRunner.Then("I should see \"The field Location is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The field Location is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/Hosting/Orchard.Web/Config/Host.config b/src/Orchard.Specs/Hosting/Orchard.Web/Config/Host.config index 27f896868..89797fb2d 100644 --- a/src/Orchard.Specs/Hosting/Orchard.Web/Config/Host.config +++ b/src/Orchard.Specs/Hosting/Orchard.Web/Config/Host.config @@ -7,7 +7,7 @@ - + @@ -190,6 +191,7 @@ + @@ -198,24 +200,34 @@ - - + + - + - + - + - + + + + + + + + + + + diff --git a/src/Orchard.Specs/Hosting/Simple.Web/Web.config b/src/Orchard.Specs/Hosting/Simple.Web/Web.config index 99931ccb3..1042a64d6 100644 --- a/src/Orchard.Specs/Hosting/Simple.Web/Web.config +++ b/src/Orchard.Specs/Hosting/Simple.Web/Web.config @@ -195,7 +195,7 @@ - + diff --git a/src/Orchard.Specs/Input.feature b/src/Orchard.Specs/Input.feature index e7e37fd2e..35a8e761c 100644 --- a/src/Orchard.Specs/Input.feature +++ b/src/Orchard.Specs/Input.feature @@ -1,13 +1,13 @@ Feature: Input Field In order to add an input to my types - As an administrator + As an administrator I want to create, edit and publish input fields Scenario: Creating and using Input fields - - # Creating an Event content type + + # Creating an Event content type Given I have installed Orchard - And I have installed "Orchard.Fields" + And I have installed "Orchard.Fields" When I go to "Admin/ContentTypes" Then I should see "]*>.*?Create new type" When I go to "Admin/ContentTypes/Create" @@ -18,113 +18,113 @@ Scenario: Creating and using Input fields And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a Input field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a Input field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | Contact | | Name | Contact | | FieldTypeName | InputField | - And I hit "Save" - And I am redirected - Then I should see "The \"Contact\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"Contact\" field has been added." - # The hint should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.Hint | Enter the contact email address | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Enter the contact email address" - - # The pattern should be effective - #When I go to "Admin/ContentTypes/Edit/Event" - # And I fill in - # | name | value | - # | Fields[0].InputFieldSettings.Pattern | [^@]*@[^@]* | - # And I hit "Save" - # And I go to "Admin/Contents/Create/Event" - #Then I should see "pattern=\"[^@]*@[^@]*\"" - - # The input type should be effective - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.Type | Email | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "type=\"email\"" - - # The title should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.Title | Enter an email address | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "title=\"Enter an email address\"" - - # The auto focus should be effective - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.AutoFocus | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "autofocus=\"autofocus\"" - - # The auto complete should be effective - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.AutoComplete | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "autocomplete=\"on\"" - - # The watermark should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.Placeholder | email@domain.com | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "placeholder=\"email@domain.com\"" - - # The maxlength should be effective - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.MaxLength | 100 | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "maxlength=\"100\"" - - # The value should be required - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].InputFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.Contact.Value | | - And I hit "Save" - Then I should see "The field Contact is mandatory." - - # Creating an Event content item - When I go to "Admin/Contents/Create/Event" - Then I should see "Contact" - When I fill in - | name | value | - | Event.Contact.Value | contact@orchardproject.net | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." - When I go to "Admin/Contents/List" - Then I should see "Contact:" - And I should see "contact@orchardproject.net" + # The hint should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.Hint | Enter the contact email address | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Enter the contact email address" + + # The pattern should be effective + #When I go to "Admin/ContentTypes/Edit/Event" + # And I fill in + # | name | value | + # | Fields[0].InputFieldSettings.Pattern | [^@]*@[^@]* | + # And I hit "Save" + # And I go to "Admin/Contents/Create/Event" + #Then I should see "pattern=\"[^@]*@[^@]*\"" + + # The input type should be effective + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.Type | Email | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "type=\"email\"" + + # The title should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.Title | Enter an email address | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "title=\"Enter an email address\"" + + # The auto focus should be effective + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.AutoFocus | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "autofocus=\"autofocus\"" + + # The auto complete should be effective + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.AutoComplete | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "autocomplete=\"on\"" + + # The watermark should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.Placeholder | email@domain.com | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "placeholder=\"email@domain.com\"" + + # The maxlength should be effective + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.MaxLength | 100 | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "maxlength=\"100\"" + + # The value should be required + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].InputFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.Contact.Value | | + And I hit "Save" + Then I should see "The field Contact is mandatory." + + # Creating an Event content item + When I go to "Admin/Contents/Create/Event" + Then I should see "Contact" + When I fill in + | name | value | + | Event.Contact.Value | contact@orchardproject.net | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." + When I go to "Admin/Contents/List" + Then I should see "Contact:" + And I should see "contact@orchardproject.net" diff --git a/src/Orchard.Specs/Input.feature.cs b/src/Orchard.Specs/Input.feature.cs index ade6902bf..2e9c54158 100644 --- a/src/Orchard.Specs/Input.feature.cs +++ b/src/Orchard.Specs/Input.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -32,8 +32,8 @@ namespace Orchard.Specs public virtual void FeatureSetup() { testRunner = TechTalk.SpecFlow.TestRunnerManager.GetTestRunner(); - TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Input Field", " In order to add an input to my types\r\nAs an administrator\r\n I want to create, " + - "edit and publish input fields", ProgrammingLanguage.CSharp, ((string[])(null))); + TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Input Field", " In order to add an input to my types\r\n As an administrator\r\n I want to create" + + ", edit and publish input fields", ProgrammingLanguage.CSharp, ((string[])(null))); testRunner.OnFeatureStart(featureInfo); } @@ -75,7 +75,7 @@ this.ScenarioSetup(scenarioInfo); #line 9 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 10 - testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 12 @@ -101,9 +101,9 @@ this.ScenarioSetup(scenarioInfo); #line 20 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 23 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 24 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -118,15 +118,15 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "InputField"}); #line 25 - testRunner.And("I fill in", ((string)(null)), table2, "And "); + testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 30 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 31 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 32 - testRunner.Then("I should see \"The \\\"Contact\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"Contact\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 35 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table3 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -135,15 +135,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.Hint", "Enter the contact email address"}); #line 36 - testRunner.And("I fill in", ((string)(null)), table3, "And "); + testRunner.And("I fill in", ((string)(null)), table3, "And "); #line 39 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 40 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 41 - testRunner.Then("I should see \"Enter the contact email address\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Enter the contact email address\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 53 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table4 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -152,15 +152,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.Type", "Email"}); #line 54 - testRunner.And("I fill in", ((string)(null)), table4, "And "); + testRunner.And("I fill in", ((string)(null)), table4, "And "); #line 57 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 58 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 59 - testRunner.Then("I should see \"type=\\\"email\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"type=\\\"email\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 62 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table5 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -169,15 +169,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.Title", "Enter an email address"}); #line 63 - testRunner.And("I fill in", ((string)(null)), table5, "And "); + testRunner.And("I fill in", ((string)(null)), table5, "And "); #line 66 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 67 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 68 - testRunner.Then("I should see \"title=\\\"Enter an email address\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"title=\\\"Enter an email address\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 71 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -186,15 +186,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.AutoFocus", "true"}); #line 72 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line 75 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 76 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 77 - testRunner.Then("I should see \"autofocus=\\\"autofocus\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"autofocus=\\\"autofocus\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 80 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -203,15 +203,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.AutoComplete", "true"}); #line 81 - testRunner.And("I fill in", ((string)(null)), table7, "And "); + testRunner.And("I fill in", ((string)(null)), table7, "And "); #line 84 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 85 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 86 - testRunner.Then("I should see \"autocomplete=\\\"on\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"autocomplete=\\\"on\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 89 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table8 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -220,15 +220,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.Placeholder", "email@domain.com"}); #line 90 - testRunner.And("I fill in", ((string)(null)), table8, "And "); + testRunner.And("I fill in", ((string)(null)), table8, "And "); #line 93 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 94 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 95 - testRunner.Then("I should see \"placeholder=\\\"email@domain.com\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"placeholder=\\\"email@domain.com\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 98 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table9 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -237,15 +237,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.MaxLength", "100"}); #line 99 - testRunner.And("I fill in", ((string)(null)), table9, "And "); + testRunner.And("I fill in", ((string)(null)), table9, "And "); #line 102 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 103 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 104 - testRunner.Then("I should see \"maxlength=\\\"100\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"maxlength=\\\"100\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 107 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table10 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -254,11 +254,11 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].InputFieldSettings.Required", "true"}); #line 108 - testRunner.And("I fill in", ((string)(null)), table10, "And "); + testRunner.And("I fill in", ((string)(null)), table10, "And "); #line 111 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 112 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table11 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -267,15 +267,15 @@ this.ScenarioSetup(scenarioInfo); "Event.Contact.Value", ""}); #line 113 - testRunner.And("I fill in", ((string)(null)), table11, "And "); + testRunner.And("I fill in", ((string)(null)), table11, "And "); #line 116 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 117 - testRunner.Then("I should see \"The field Contact is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The field Contact is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 120 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 121 - testRunner.Then("I should see \"Contact\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Contact\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table12 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -284,19 +284,19 @@ this.ScenarioSetup(scenarioInfo); "Event.Contact.Value", "contact@orchardproject.net"}); #line 122 - testRunner.When("I fill in", ((string)(null)), table12, "When "); + testRunner.When("I fill in", ((string)(null)), table12, "When "); #line 125 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 126 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 127 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 128 - testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 129 - testRunner.Then("I should see \"Contact:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Contact:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 130 - testRunner.And("I should see \"contact@orchardproject.net\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"contact@orchardproject.net\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/Link.feature b/src/Orchard.Specs/Link.feature index cf7d8d04f..9ae1de538 100644 --- a/src/Orchard.Specs/Link.feature +++ b/src/Orchard.Specs/Link.feature @@ -1,13 +1,13 @@ Feature: Link Field In order to add Link content to my types - As an administrator + As an administrator I want to create, edit and publish Link fields Scenario: Creating and using Link fields - - # Creating an Event content type + + # Creating an Event content type Given I have installed Orchard - And I have installed "Orchard.Fields" + And I have installed "Orchard.Fields" When I go to "Admin/ContentTypes" Then I should see "]*>.*?Create new type" When I go to "Admin/ContentTypes/Create" @@ -18,53 +18,53 @@ Scenario: Creating and using Link fields And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a Link field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a Link field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | Site Url | | Name | SiteUrl | | FieldTypeName | LinkField | - And I hit "Save" - And I am redirected - Then I should see "The \"Site Url\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"Site Url\" field has been added." - # Creating an Event content item - When I go to "Admin/Contents/Create/Event" - Then I should see "Site Url" - When I fill in - | name | value | - | Event.SiteUrl.Value | http://www.orchardproject.net | - And I fill in - | name | value | - | Event.SiteUrl.Text | Orchard | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." - When I go to "Admin/Contents/List" - Then I should see "Site Url:" - And I should see "Orchard" + # Creating an Event content item + When I go to "Admin/Contents/Create/Event" + Then I should see "Site Url" + When I fill in + | name | value | + | Event.SiteUrl.Value | http://www.orchardproject.net | + And I fill in + | name | value | + | Event.SiteUrl.Text | Orchard | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." + When I go to "Admin/Contents/List" + Then I should see "Site Url:" + And I should see "Orchard" - # The hint should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].LinkFieldSettings.Hint | Enter the url of the web site | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Enter the url of the web site" - - # The value should be required - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].LinkFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.SiteUrl.Value | | - And I hit "Save" - Then I should see "Url is required for Site Url." \ No newline at end of file + # The hint should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].LinkFieldSettings.Hint | Enter the url of the web site | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Enter the url of the web site" + + # The value should be required + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].LinkFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.SiteUrl.Value | | + And I hit "Save" + Then I should see "Url is required for Site Url." \ No newline at end of file diff --git a/src/Orchard.Specs/Link.feature.cs b/src/Orchard.Specs/Link.feature.cs index e226372db..84b100ed4 100644 --- a/src/Orchard.Specs/Link.feature.cs +++ b/src/Orchard.Specs/Link.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -32,8 +32,8 @@ namespace Orchard.Specs public virtual void FeatureSetup() { testRunner = TechTalk.SpecFlow.TestRunnerManager.GetTestRunner(); - TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Link Field", " In order to add Link content to my types\r\nAs an administrator\r\n I want to crea" + - "te, edit and publish Link fields", ProgrammingLanguage.CSharp, ((string[])(null))); + TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Link Field", " In order to add Link content to my types\r\n As an administrator\r\n I want to cr" + + "eate, edit and publish Link fields", ProgrammingLanguage.CSharp, ((string[])(null))); testRunner.OnFeatureStart(featureInfo); } @@ -75,7 +75,7 @@ this.ScenarioSetup(scenarioInfo); #line 9 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 10 - testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 12 @@ -101,9 +101,9 @@ this.ScenarioSetup(scenarioInfo); #line 20 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 23 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 24 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -118,17 +118,17 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "LinkField"}); #line 25 - testRunner.And("I fill in", ((string)(null)), table2, "And "); + testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 30 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 31 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 32 - testRunner.Then("I should see \"The \\\"Site Url\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"Site Url\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 35 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 36 - testRunner.Then("I should see \"Site Url\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Site Url\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table3 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -137,7 +137,7 @@ this.ScenarioSetup(scenarioInfo); "Event.SiteUrl.Value", "http://www.orchardproject.net"}); #line 37 - testRunner.When("I fill in", ((string)(null)), table3, "When "); + testRunner.When("I fill in", ((string)(null)), table3, "When "); #line hidden TechTalk.SpecFlow.Table table4 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -146,21 +146,21 @@ this.ScenarioSetup(scenarioInfo); "Event.SiteUrl.Text", "Orchard"}); #line 40 - testRunner.And("I fill in", ((string)(null)), table4, "And "); + testRunner.And("I fill in", ((string)(null)), table4, "And "); #line 43 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 44 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 45 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 46 - testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 47 - testRunner.Then("I should see \"Site Url:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Site Url:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 48 - testRunner.And("I should see \"Orchard\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"Orchard\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 51 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table5 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -169,15 +169,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].LinkFieldSettings.Hint", "Enter the url of the web site"}); #line 52 - testRunner.And("I fill in", ((string)(null)), table5, "And "); + testRunner.And("I fill in", ((string)(null)), table5, "And "); #line 55 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 56 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 57 - testRunner.Then("I should see \"Enter the url of the web site\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Enter the url of the web site\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 60 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -186,11 +186,11 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].LinkFieldSettings.Required", "true"}); #line 61 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line 64 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 65 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -199,11 +199,11 @@ this.ScenarioSetup(scenarioInfo); "Event.SiteUrl.Value", ""}); #line 66 - testRunner.And("I fill in", ((string)(null)), table7, "And "); + testRunner.And("I fill in", ((string)(null)), table7, "And "); #line 69 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 70 - testRunner.Then("I should see \"Url is required for Site Url.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Url is required for Site Url.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/Lists.feature b/src/Orchard.Specs/Lists.feature index 5074e822c..e68f3ccac 100644 --- a/src/Orchard.Specs/Lists.feature +++ b/src/Orchard.Specs/Lists.feature @@ -5,19 +5,19 @@ Scenario: I can create a new list Given I have installed Orchard - And I have installed "Orchard.Lists" + And I have installed "Orchard.Lists" When I go to "Admin/ContentTypes" - And I go to "Admin/ContentTypes/Create" + And I go to "Admin/ContentTypes/Create" And I fill in | name | value | | DisplayName | Event | | Name | Event | And I hit "Create" - And I am redirected + And I am redirected And I fill in | name | value | | PartSelections[5].IsSelected | True | - And I hit "Save" + And I hit "Save" And I go to "Admin/ContentTypes/" Then I should see "Event" @@ -27,8 +27,8 @@ Scenario: I can create a new list | Title.Title | MyList | | Container.SelectedItemContentTypes | Event | And I hit "Save" - And I am redirected - Then I should see "Your List has been created" + And I am redirected + Then I should see "Your List has been created" When I go to "Admin/Lists" Then I should see "MyList" When I follow "Contained Items (0)" diff --git a/src/Orchard.Specs/Lists.feature.cs b/src/Orchard.Specs/Lists.feature.cs index a5bd6ef8b..ec41dafc7 100644 --- a/src/Orchard.Specs/Lists.feature.cs +++ b/src/Orchard.Specs/Lists.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -75,11 +75,11 @@ this.ScenarioSetup(scenarioInfo); #line 7 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 8 - testRunner.And("I have installed \"Orchard.Lists\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Lists\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 9 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 10 - testRunner.And("I go to \"Admin/ContentTypes/Create\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/ContentTypes/Create\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table1 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -95,7 +95,7 @@ this.ScenarioSetup(scenarioInfo); #line 15 testRunner.And("I hit \"Create\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 16 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -106,7 +106,7 @@ this.ScenarioSetup(scenarioInfo); #line 17 testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 20 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 21 testRunner.And("I go to \"Admin/ContentTypes/\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 22 @@ -128,9 +128,9 @@ this.ScenarioSetup(scenarioInfo); #line 29 testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 30 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 31 - testRunner.Then("I should see \"Your List has been created\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your List has been created\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 32 testRunner.When("I go to \"Admin/Lists\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 33 diff --git a/src/Orchard.Specs/Media.feature b/src/Orchard.Specs/Media.feature index dd018ab39..5e1193fbf 100644 --- a/src/Orchard.Specs/Media.feature +++ b/src/Orchard.Specs/Media.feature @@ -5,14 +5,14 @@ Scenario: Media admin is available Given I have installed Orchard - And I have installed "Orchard.Media" + And I have installed "Orchard.Media" - # Accessing the media page + # Accessing the media page When I go to "admin/media" Then I should see "Media" And the status should be 200 "OK" - # Creating a folder + # Creating a folder When I go to "admin/media/create" And I fill in | name | value | @@ -23,7 +23,7 @@ Scenario: Media admin is available And I should see "Hello World" And the status should be 200 "OK" - # Editing a media with limited rights + # Editing a media with limited rights When I go to "admin/media/edit?name=..\..\bin&mediaPath=..\..\bin" And I am redirected Then I should see "Media" diff --git a/src/Orchard.Specs/Media.feature.cs b/src/Orchard.Specs/Media.feature.cs index a9da6dd42..2c1d61c13 100644 --- a/src/Orchard.Specs/Media.feature.cs +++ b/src/Orchard.Specs/Media.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -75,7 +75,7 @@ this.ScenarioSetup(scenarioInfo); #line 7 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 8 - testRunner.And("I have installed \"Orchard.Media\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Media\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 testRunner.When("I go to \"admin/media\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 12 diff --git a/src/Orchard.Specs/MediaPicker.feature b/src/Orchard.Specs/MediaPicker.feature index 3c1008de6..eba7506a1 100644 --- a/src/Orchard.Specs/MediaPicker.feature +++ b/src/Orchard.Specs/MediaPicker.feature @@ -1,14 +1,14 @@ Feature: Media Picker Field In order to add a media content to my types - As an administrator + As an administrator I want to create, edit and publish media fields Scenario: Creating and using media fields - - # Creating an Event content type + + # Creating an Event content type Given I have installed Orchard - And I have installed "Orchard.Media" - And I have installed "Orchard.MediaPicker" + And I have installed "Orchard.Media" + And I have installed "Orchard.MediaPicker" When I go to "Admin/ContentTypes" Then I should see "]*>.*?Create new type" @@ -20,61 +20,61 @@ Scenario: Creating and using media fields And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a media field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a media field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | File | | Name | File | | FieldTypeName | MediaPickerField | - And I hit "Save" - And I am redirected - Then I should see "The \"File\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"File\" field has been added." - # Creating an Event content item - When I go to "Admin/Contents/Create/Event" - Then I should see "File" - When I fill in - | name | value | - | Event.File.Url | | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." + # Creating an Event content item + When I go to "Admin/Contents/Create/Event" + Then I should see "File" + When I fill in + | name | value | + | Event.File.Url | | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." - # The hint should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].MediaPickerFieldSettings.Hint | Please select a file | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Please select a file" + # The hint should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].MediaPickerFieldSettings.Hint | Please select a file | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Please select a file" - # The value should be required - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].MediaPickerFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.File.Url | | - And I hit "Save" - Then I should see "The field File is mandatory." + # The value should be required + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].MediaPickerFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.File.Url | | + And I hit "Save" + Then I should see "The field File is mandatory." - # The value should be bound - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | ext-Fields[0].MediaPickerFieldSettings | true | - | Fields[0].MediaPickerFieldSettings.AllowedExtensions | jpg | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.File.Url | ~/Media/Default/images/Image.png | - And I hit "Save" - Then I should see "The field File must have one of these extensions: jpg" \ No newline at end of file + # The value should be bound + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | ext-Fields[0].MediaPickerFieldSettings | true | + | Fields[0].MediaPickerFieldSettings.AllowedExtensions | jpg | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.File.Url | ~/Media/Default/images/Image.png | + And I hit "Save" + Then I should see "The field File must have one of these extensions: jpg" \ No newline at end of file diff --git a/src/Orchard.Specs/MediaPicker.feature.cs b/src/Orchard.Specs/MediaPicker.feature.cs index 9ff2adb90..3189f9308 100644 --- a/src/Orchard.Specs/MediaPicker.feature.cs +++ b/src/Orchard.Specs/MediaPicker.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -32,8 +32,8 @@ namespace Orchard.Specs public virtual void FeatureSetup() { testRunner = TechTalk.SpecFlow.TestRunnerManager.GetTestRunner(); - TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Media Picker Field", " In order to add a media content to my types\r\nAs an administrator\r\n I want to c" + - "reate, edit and publish media fields", ProgrammingLanguage.CSharp, ((string[])(null))); + TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Media Picker Field", " In order to add a media content to my types\r\n As an administrator\r\n I want to" + + " create, edit and publish media fields", ProgrammingLanguage.CSharp, ((string[])(null))); testRunner.OnFeatureStart(featureInfo); } @@ -75,9 +75,9 @@ this.ScenarioSetup(scenarioInfo); #line 9 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 10 - testRunner.And("I have installed \"Orchard.Media\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Media\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 - testRunner.And("I have installed \"Orchard.MediaPicker\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.MediaPicker\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 13 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 14 @@ -103,9 +103,9 @@ this.ScenarioSetup(scenarioInfo); #line 22 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 25 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 26 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -120,17 +120,17 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "MediaPickerField"}); #line 27 - testRunner.And("I fill in", ((string)(null)), table2, "And "); + testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 32 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 33 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 34 - testRunner.Then("I should see \"The \\\"File\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"File\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 37 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 38 - testRunner.Then("I should see \"File\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"File\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table3 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -139,15 +139,15 @@ this.ScenarioSetup(scenarioInfo); "Event.File.Url", ""}); #line 39 - testRunner.When("I fill in", ((string)(null)), table3, "When "); + testRunner.When("I fill in", ((string)(null)), table3, "When "); #line 42 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 43 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 44 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 47 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table4 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -156,15 +156,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].MediaPickerFieldSettings.Hint", "Please select a file"}); #line 48 - testRunner.And("I fill in", ((string)(null)), table4, "And "); + testRunner.And("I fill in", ((string)(null)), table4, "And "); #line 51 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 52 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 53 - testRunner.Then("I should see \"Please select a file\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Please select a file\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 56 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table5 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -173,11 +173,11 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].MediaPickerFieldSettings.Required", "true"}); #line 57 - testRunner.And("I fill in", ((string)(null)), table5, "And "); + testRunner.And("I fill in", ((string)(null)), table5, "And "); #line 60 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 61 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -186,13 +186,13 @@ this.ScenarioSetup(scenarioInfo); "Event.File.Url", ""}); #line 62 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line 65 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 66 - testRunner.Then("I should see \"The field File is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The field File is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 69 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -204,11 +204,11 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].MediaPickerFieldSettings.AllowedExtensions", "jpg"}); #line 70 - testRunner.And("I fill in", ((string)(null)), table7, "And "); + testRunner.And("I fill in", ((string)(null)), table7, "And "); #line 74 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 75 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table8 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -217,11 +217,11 @@ this.ScenarioSetup(scenarioInfo); "Event.File.Url", "~/Media/Default/images/Image.png"}); #line 76 - testRunner.And("I fill in", ((string)(null)), table8, "And "); + testRunner.And("I fill in", ((string)(null)), table8, "And "); #line 79 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 80 - testRunner.Then("I should see \"The field File must have one of these extensions: jpg\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The field File must have one of these extensions: jpg\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/MultiTenancy.feature b/src/Orchard.Specs/MultiTenancy.feature index ae586c309..909569d7f 100644 --- a/src/Orchard.Specs/MultiTenancy.feature +++ b/src/Orchard.Specs/MultiTenancy.feature @@ -168,7 +168,7 @@ Scenario: A running tenant can be disabled And I go to "/Admin/MultiTenancy" on host localhost And I hit "Suspend" And I am redirected - Then I should see "
tenant list Then I should see "Name: Alpha" - And I should see "Request Url Host: example.org" \ No newline at end of file + And I should see "Request URL host: example.org" \ No newline at end of file diff --git a/src/Orchard.Specs/MultiTenancy.feature.cs b/src/Orchard.Specs/MultiTenancy.feature.cs index d1d54d065..1c6466f64 100644 --- a/src/Orchard.Specs/MultiTenancy.feature.cs +++ b/src/Orchard.Specs/MultiTenancy.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -525,7 +525,7 @@ this.ScenarioSetup(scenarioInfo); #line 170 testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 171 - testRunner.Then("I should see \"]*>.*?Create new type" When I go to "Admin/ContentTypes/Create" @@ -18,82 +18,82 @@ Scenario: Creating and using numeric fields And I hit "Create" And I go to "Admin/ContentTypes/" Then I should see "Event" - - # Adding a numeric field - When I go to "Admin/ContentTypes/Edit/Event" - And I follow "Add Field" - And I fill in + + # Adding a numeric field + When I go to "Admin/ContentTypes/Edit/Event" + And I follow "Add Field" + And I fill in | name | value | | DisplayName | Guests | | Name | Guests | | FieldTypeName | NumericField | - And I hit "Save" - And I am redirected - Then I should see "The \"Guests\" field has been added." + And I hit "Save" + And I am redirected + Then I should see "The \"Guests\" field has been added." - # Creating an Event content item - When I go to "Admin/Contents/Create/Event" - Then I should see "Guests" - When I fill in - | name | value | - | Event.Guests.Value | 3 | - And I hit "Save" - And I am redirected - Then I should see "Your Event has been created." - When I go to "Admin/Contents/List" - Then I should see "Guests:" - And I should see "3" + # Creating an Event content item + When I go to "Admin/Contents/Create/Event" + Then I should see "Guests" + When I fill in + | name | value | + | Event.Guests.Value | 3 | + And I hit "Save" + And I am redirected + Then I should see "Your Event has been created." + When I go to "Admin/Contents/List" + Then I should see "Guests:" + And I should see "3" - # The hint should be displayed - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].NumericFieldSettings.Hint | Please enter a number | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "Please enter a number" + # The hint should be displayed + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].NumericFieldSettings.Hint | Please enter a number | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "Please enter a number" - # The value should be required - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].NumericFieldSettings.Required | true | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.Guests.Value | | - And I hit "Save" - Then I should see "The field Guests is mandatory." + # The value should be required + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].NumericFieldSettings.Required | true | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.Guests.Value | | + And I hit "Save" + Then I should see "The field Guests is mandatory." - # The value should be bound - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].NumericFieldSettings.Minimum | -10 | - | Fields[0].NumericFieldSettings.Maximum | 100 | - And I hit "Save" - And I go to "Admin/Contents/Create/Event" - Then I should see "min=\"-10\"" - And I should see "max=\"100\"" - When I fill in - | name | value | - | Event.Guests.Value | -20 | - And I hit "Save" - Then I should see "The value must be greater than -10" - When I go to "Admin/Contents/Create/Event" - And I fill in - | name | value | - | Event.Guests.Value | 101 | - And I hit "Save" - Then I should see "The value must be less than 100" - - # Settings should be validated - When I go to "Admin/ContentTypes/Edit/Event" - And I fill in - | name | value | - | Fields[0].NumericFieldSettings.Minimum | a | - | Fields[0].NumericFieldSettings.Maximum | b | - And I hit "Save" - Then I should see "The value 'a' is not valid for Minimum." - And I should see "The value 'b' is not valid for Maximum." \ No newline at end of file + # The value should be bound + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].NumericFieldSettings.Minimum | -10 | + | Fields[0].NumericFieldSettings.Maximum | 100 | + And I hit "Save" + And I go to "Admin/Contents/Create/Event" + Then I should see "min=\"-10\"" + And I should see "max=\"100\"" + When I fill in + | name | value | + | Event.Guests.Value | -20 | + And I hit "Save" + Then I should see "The value must be greater than -10" + When I go to "Admin/Contents/Create/Event" + And I fill in + | name | value | + | Event.Guests.Value | 101 | + And I hit "Save" + Then I should see "The value must be less than 100" + + # Settings should be validated + When I go to "Admin/ContentTypes/Edit/Event" + And I fill in + | name | value | + | Fields[0].NumericFieldSettings.Minimum | a | + | Fields[0].NumericFieldSettings.Maximum | b | + And I hit "Save" + Then I should see "The value 'a' is not valid for Minimum." + And I should see "The value 'b' is not valid for Maximum." \ No newline at end of file diff --git a/src/Orchard.Specs/Numeric.feature.cs b/src/Orchard.Specs/Numeric.feature.cs index a676279cd..e90b2c1bf 100644 --- a/src/Orchard.Specs/Numeric.feature.cs +++ b/src/Orchard.Specs/Numeric.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -32,8 +32,8 @@ namespace Orchard.Specs public virtual void FeatureSetup() { testRunner = TechTalk.SpecFlow.TestRunnerManager.GetTestRunner(); - TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Numeric Field", " In order to add numeric content to my types\r\nAs an administrator\r\n I want to c" + - "reate, edit and publish numeric fields", ProgrammingLanguage.CSharp, ((string[])(null))); + TechTalk.SpecFlow.FeatureInfo featureInfo = new TechTalk.SpecFlow.FeatureInfo(new System.Globalization.CultureInfo("en-US"), "Numeric Field", " In order to add numeric content to my types\r\n As an administrator\r\n I want to" + + " create, edit and publish numeric fields", ProgrammingLanguage.CSharp, ((string[])(null))); testRunner.OnFeatureStart(featureInfo); } @@ -75,7 +75,7 @@ this.ScenarioSetup(scenarioInfo); #line 9 testRunner.Given("I have installed Orchard", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Given "); #line 10 - testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I have installed \"Orchard.Fields\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 11 testRunner.When("I go to \"Admin/ContentTypes\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 12 @@ -101,9 +101,9 @@ this.ScenarioSetup(scenarioInfo); #line 20 testRunner.Then("I should see \"Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 23 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 24 - testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I follow \"Add Field\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table2 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -118,17 +118,17 @@ this.ScenarioSetup(scenarioInfo); "FieldTypeName", "NumericField"}); #line 25 - testRunner.And("I fill in", ((string)(null)), table2, "And "); + testRunner.And("I fill in", ((string)(null)), table2, "And "); #line 30 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 31 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 32 - testRunner.Then("I should see \"The \\\"Guests\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The \\\"Guests\\\" field has been added.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 35 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 36 - testRunner.Then("I should see \"Guests\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Guests\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden TechTalk.SpecFlow.Table table3 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -137,21 +137,21 @@ this.ScenarioSetup(scenarioInfo); "Event.Guests.Value", "3"}); #line 37 - testRunner.When("I fill in", ((string)(null)), table3, "When "); + testRunner.When("I fill in", ((string)(null)), table3, "When "); #line 40 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 41 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 42 - testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Your Event has been created.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 43 - testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/List\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line 44 - testRunner.Then("I should see \"Guests:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Guests:\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 45 - testRunner.And("I should see \"3\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"3\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 48 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table4 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -160,15 +160,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].NumericFieldSettings.Hint", "Please enter a number"}); #line 49 - testRunner.And("I fill in", ((string)(null)), table4, "And "); + testRunner.And("I fill in", ((string)(null)), table4, "And "); #line 52 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 53 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 54 - testRunner.Then("I should see \"Please enter a number\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"Please enter a number\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 57 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table5 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -177,11 +177,11 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].NumericFieldSettings.Required", "true"}); #line 58 - testRunner.And("I fill in", ((string)(null)), table5, "And "); + testRunner.And("I fill in", ((string)(null)), table5, "And "); #line 61 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 62 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table6 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -190,13 +190,13 @@ this.ScenarioSetup(scenarioInfo); "Event.Guests.Value", ""}); #line 63 - testRunner.And("I fill in", ((string)(null)), table6, "And "); + testRunner.And("I fill in", ((string)(null)), table6, "And "); #line 66 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 67 - testRunner.Then("I should see \"The field Guests is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The field Guests is mandatory.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 70 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table7 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -208,15 +208,15 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].NumericFieldSettings.Maximum", "100"}); #line 71 - testRunner.And("I fill in", ((string)(null)), table7, "And "); + testRunner.And("I fill in", ((string)(null)), table7, "And "); #line 75 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 76 - testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 77 - testRunner.Then("I should see \"min=\\\"-10\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"min=\\\"-10\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 78 - testRunner.And("I should see \"max=\\\"100\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"max=\\\"100\\\"\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden TechTalk.SpecFlow.Table table8 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -225,13 +225,13 @@ this.ScenarioSetup(scenarioInfo); "Event.Guests.Value", "-20"}); #line 79 - testRunner.When("I fill in", ((string)(null)), table8, "When "); + testRunner.When("I fill in", ((string)(null)), table8, "When "); #line 82 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 83 - testRunner.Then("I should see \"The value must be greater than -10\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The value must be greater than -10\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 84 - testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/Contents/Create/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table9 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -240,13 +240,13 @@ this.ScenarioSetup(scenarioInfo); "Event.Guests.Value", "101"}); #line 85 - testRunner.And("I fill in", ((string)(null)), table9, "And "); + testRunner.And("I fill in", ((string)(null)), table9, "And "); #line 88 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 89 - testRunner.Then("I should see \"The value must be less than 100\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The value must be less than 100\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 92 - testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); + testRunner.When("I go to \"Admin/ContentTypes/Edit/Event\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "When "); #line hidden TechTalk.SpecFlow.Table table10 = new TechTalk.SpecFlow.Table(new string[] { "name", @@ -258,13 +258,13 @@ this.ScenarioSetup(scenarioInfo); "Fields[0].NumericFieldSettings.Maximum", "b"}); #line 93 - testRunner.And("I fill in", ((string)(null)), table10, "And "); + testRunner.And("I fill in", ((string)(null)), table10, "And "); #line 97 - testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I hit \"Save\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 98 - testRunner.Then("I should see \"The value 'a' is not valid for Minimum.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); + testRunner.Then("I should see \"The value 'a' is not valid for Minimum.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line 99 - testRunner.And("I should see \"The value 'b' is not valid for Maximum.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I should see \"The value 'b' is not valid for Maximum.\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line hidden this.ScenarioCleanup(); } diff --git a/src/Orchard.Specs/Pages.feature b/src/Orchard.Specs/Pages.feature index 5352b3df9..382d20e2d 100644 --- a/src/Orchard.Specs/Pages.feature +++ b/src/Orchard.Specs/Pages.feature @@ -25,7 +25,7 @@ Scenario: In the admin (menu) there is a link to create a Page And I fill in | name | value | | Title.Title | Super Duper | - | LayoutPart.LayoutEditor.Data | { "type": "Content", "data": "TypeName=Orchard.Layouts.Elements.Text&Content=This+is+super+number+two", "isTemplated": false, "contentType": "Orchard.Layouts.Elements.Text", "contentTypeLabel": "Text", "contentTypeClass": "text", "html": "This is super number two", "hasEditor": true } | + | AutoroutePart.LayoutEditor.Data | { "type": "Content", "data": "TypeName=Orchard.Layouts.Elements.Text&Content=This+is+super+number+two", "isTemplated": false, "contentType": "Orchard.Layouts.Elements.Text", "contentTypeLabel": "Text", "contentTypeClass": "text", "html": "This is super number two", "hasEditor": true } | And I hit "Publish Now" And I go to "super-duper-2" Then I should see "]*>.*?Super Duper.*?" @@ -36,7 +36,7 @@ Scenario: In the admin (menu) there is a link to create a Page And I fill in | name | value | | Title.Title | Another | - | Autoroute.PromoteToHomePage | true | + | AutoroutePart.PromoteToHomePage | true | And I hit "Publish Now" And I go to "/" Then I should see "

Another

" @@ -48,7 +48,7 @@ Scenario: In the admin (menu) there is a link to create a Page And I fill in | name | value | | Title.Title | Drafty | - | Autoroute.PromoteToHomePage | true | + | AutoroutePart.PromoteToHomePage | true | And I hit "Save" And I go to "/" Then I should see "

Another

" diff --git a/src/Orchard.Specs/Pages.feature.cs b/src/Orchard.Specs/Pages.feature.cs index 7ea3cf7d5..5a60be007 100644 --- a/src/Orchard.Specs/Pages.feature.cs +++ b/src/Orchard.Specs/Pages.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.0 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -110,7 +110,7 @@ this.ScenarioSetup(scenarioInfo); "Title.Title", "Super Duper"}); table2.AddRow(new string[] { - "LayoutPart.LayoutEditor.Data", + "AutoroutePart.LayoutEditor.Data", @"{ ""type"": ""Content"", ""data"": ""TypeName=Orchard.Layouts.Elements.Text&Content=This+is+super+number+two"", ""isTemplated"": false, ""contentType"": ""Orchard.Layouts.Elements.Text"", ""contentTypeLabel"": ""Text"", ""contentTypeClass"": ""text"", ""html"": ""This is super number two"", ""hasEditor"": true }"}); #line 25 testRunner.And("I fill in", ((string)(null)), table2, "And "); @@ -132,7 +132,7 @@ this.ScenarioSetup(scenarioInfo); "Title.Title", "Another"}); table3.AddRow(new string[] { - "Autoroute.PromoteToHomePage", + "AutoroutePart.PromoteToHomePage", "true"}); #line 36 testRunner.And("I fill in", ((string)(null)), table3, "And "); @@ -156,7 +156,7 @@ this.ScenarioSetup(scenarioInfo); "Title.Title", "Drafty"}); table4.AddRow(new string[] { - "Autoroute.PromoteToHomePage", + "AutoroutePart.PromoteToHomePage", "true"}); #line 48 testRunner.And("I fill in", ((string)(null)), table4, "And "); diff --git a/src/Orchard.Specs/PermissionModel.feature b/src/Orchard.Specs/PermissionModel.feature index aa480a614..e010048bb 100644 --- a/src/Orchard.Specs/PermissionModel.feature +++ b/src/Orchard.Specs/PermissionModel.feature @@ -25,5 +25,4 @@ Scenario: Anonymous user can see the home page but not the dashboard When I sign in as "bob" And I go to "/" Then I should see "this is the homepage of your new site" - And I should be denied access when I go to "admin" - + And I should be denied access when I go to "admin" \ No newline at end of file diff --git a/src/Orchard.Specs/PermissionModel.feature.cs b/src/Orchard.Specs/PermissionModel.feature.cs index 5d5cdd391..c58cf66f7 100644 --- a/src/Orchard.Specs/PermissionModel.feature.cs +++ b/src/Orchard.Specs/PermissionModel.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.34014 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. diff --git a/src/Orchard.Specs/Properties/AssemblyInfo.cs b/src/Orchard.Specs/Properties/AssemblyInfo.cs index 44da55c06..be2431218 100644 --- a/src/Orchard.Specs/Properties/AssemblyInfo.cs +++ b/src/Orchard.Specs/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Specs/Tags.feature b/src/Orchard.Specs/Tags.feature index af72b03e6..71440522a 100644 --- a/src/Orchard.Specs/Tags.feature +++ b/src/Orchard.Specs/Tags.feature @@ -26,5 +26,5 @@ Scenario: I can't add a tag with disallowed chars to a new Page | LayoutPart.LayoutEditor.Data | { "elements": [ { "typeName": "Orchard.Layouts.Elements.Text", "state": "Content=This+is+super."} ] } | | Tags.Tags | Foo, I <3 Orchard | And I hit "Publish Now" - And I am redirected + And I am redirected Then I should see "forbidden chars" diff --git a/src/Orchard.Specs/Tags.feature.cs b/src/Orchard.Specs/Tags.feature.cs index ef767cd61..71de7b3b9 100644 --- a/src/Orchard.Specs/Tags.feature.cs +++ b/src/Orchard.Specs/Tags.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.0 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. @@ -136,7 +136,7 @@ this.ScenarioSetup(scenarioInfo); #line 28 testRunner.And("I hit \"Publish Now\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 29 - testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); + testRunner.And("I am redirected", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "And "); #line 30 testRunner.Then("I should see \"forbidden chars\"", ((string)(null)), ((TechTalk.SpecFlow.Table)(null)), "Then "); #line hidden diff --git a/src/Orchard.Specs/Users.feature b/src/Orchard.Specs/Users.feature index dafcf1ebd..2378eddff 100644 --- a/src/Orchard.Specs/Users.feature +++ b/src/Orchard.Specs/Users.feature @@ -261,7 +261,7 @@ Scenario: I should be able to filter users by status Then I should see "User user1 disabled" When I fill in | name | value | - | Options.Search | | + | Options.Search | | | Options.Filter | Pending | And I hit "Filter" Then I should see "]*>user1" @@ -270,7 +270,7 @@ Scenario: I should be able to filter users by status When I fill in | name | value | | Options.Search | | - | Options.Filter | EmailPending | + | Options.Filter | EmailPending | And I hit "Filter" Then I should not see "]*>user1" And I should not see "]*>user2" @@ -278,14 +278,14 @@ Scenario: I should be able to filter users by status When I fill in | name | value | | Options.Search | | - | Options.Filter | Approved | + | Options.Filter | Approved | And I hit "Filter" Then I should not see "]*>user1" And I should see "]*>user2" And I should see "]*>admin" When I fill in | name | value | - | Options.Search | | + | Options.Search | | | Options.Filter | All | And I hit "Filter" Then I should see "]*>user1" diff --git a/src/Orchard.Specs/Users.feature.cs b/src/Orchard.Specs/Users.feature.cs index e1d2337e2..172ba791f 100644 --- a/src/Orchard.Specs/Users.feature.cs +++ b/src/Orchard.Specs/Users.feature.cs @@ -3,7 +3,7 @@ // This code was generated by SpecFlow (http://www.specflow.org/). // SpecFlow Version:1.9.0.77 // SpecFlow Generator Version:1.9.0.0 -// Runtime Version:4.0.30319.0 +// Runtime Version:4.0.30319.42000 // // Changes to this file may cause incorrect behavior and will be lost if // the code is regenerated. diff --git a/src/Orchard.Tests.Modules/App.config b/src/Orchard.Tests.Modules/App.config index a14743e2d..4febd68c7 100644 --- a/src/Orchard.Tests.Modules/App.config +++ b/src/Orchard.Tests.Modules/App.config @@ -24,7 +24,7 @@ - + diff --git a/src/Orchard.Tests.Modules/DesignerTools/Services/ObjectDumperTests.cs b/src/Orchard.Tests.Modules/DesignerTools/Services/ObjectDumperTests.cs index d78cd52bb..22b162ac1 100644 --- a/src/Orchard.Tests.Modules/DesignerTools/Services/ObjectDumperTests.cs +++ b/src/Orchard.Tests.Modules/DesignerTools/Services/ObjectDumperTests.cs @@ -264,9 +264,6 @@ namespace Orchard.Tests.Modules.DesignerTools.Services new JProperty("name", "TestingPart"), new JProperty("value", "ContentPart"), new JProperty("children", new JArray( - new JObject( - new JProperty("name", "Zones"), - new JProperty("value", "ZoneCollection")), new JObject( new JProperty("name", "Id"), new JProperty("value", "0")), @@ -327,9 +324,6 @@ namespace Orchard.Tests.Modules.DesignerTools.Services new JProperty("name", "TestingPart"), new JProperty("value", "ContentPart"), new JProperty("children", new JArray( - new JObject( - new JProperty("name", "Zones"), - new JProperty("value", "ZoneCollection")), new JObject( new JProperty("name", "Id"), new JProperty("value", "0")), diff --git a/src/Orchard.Tests.Modules/Indexing/LuceneIndexProviderTests.cs b/src/Orchard.Tests.Modules/Indexing/LuceneIndexProviderTests.cs index 41c6239fb..d7008dd6a 100644 --- a/src/Orchard.Tests.Modules/Indexing/LuceneIndexProviderTests.cs +++ b/src/Orchard.Tests.Modules/Indexing/LuceneIndexProviderTests.cs @@ -303,5 +303,24 @@ namespace Orchard.Tests.Modules.Indexing { _provider.Store("default", _provider.New(1).Add("field", "value2")); Assert.That(searchBuilder.WithField("id", "1").Count(), Is.EqualTo(1)); } + + [Test] + public void IndexProviderShouldDeleteMoreThanMaxTermsCount() { + _provider.CreateIndex("default"); + + var documents = Enumerable.Range(1, 1025).Select(i => _provider.New(i).Add("field", "value1")); + _provider.Store("default", documents); + + var searchBuilder = _provider.CreateSearchBuilder("default"); + + Assert.That(searchBuilder.Count(), Is.EqualTo(1025)); + Assert.That(searchBuilder.Get(1).ContentItemId, Is.EqualTo(1)); + Assert.That(searchBuilder.Get(1025).ContentItemId, Is.EqualTo(1025)); + + _provider.Delete("default", Enumerable.Range(1, 1025)); + + + Assert.That(searchBuilder.Count(), Is.EqualTo(0)); + } } } diff --git a/src/Orchard.Tests.Modules/Properties/AssemblyInfo.cs b/src/Orchard.Tests.Modules/Properties/AssemblyInfo.cs index 1ed9e5a14..569ed6ab3 100644 --- a/src/Orchard.Tests.Modules/Properties/AssemblyInfo.cs +++ b/src/Orchard.Tests.Modules/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Tests.Modules/Recipes/Services/RecipeManagerTests.cs b/src/Orchard.Tests.Modules/Recipes/Services/RecipeManagerTests.cs index c522aae81..7844dc917 100644 --- a/src/Orchard.Tests.Modules/Recipes/Services/RecipeManagerTests.cs +++ b/src/Orchard.Tests.Modules/Recipes/Services/RecipeManagerTests.cs @@ -110,23 +110,23 @@ namespace Orchard.Tests.Modules.Recipes.Services { [Test] public void HarvestRecipesFailsToFindRecipesWhenCalledWithNotExistingExtension() { - var recipes = (List)_recipeHarvester.HarvestRecipes("cantfindme"); + var recipes = _recipeHarvester.HarvestRecipes("cantfindme"); - Assert.That(recipes.Count, Is.EqualTo(0)); + Assert.That(recipes.Count(), Is.EqualTo(0)); } [Test] public void HarvestRecipesShouldHarvestRecipeXmlFiles() { - var recipes = (List)_recipeHarvester.HarvestRecipes("Sample1"); - Assert.That(recipes.Count, Is.EqualTo(1)); + var recipes = _recipeHarvester.HarvestRecipes("Sample1"); + Assert.That(recipes.Count(), Is.EqualTo(1)); } [Test] public void ParseRecipeLoadsRecipeMetaDataIntoModel() { - var recipes = (List)_recipeHarvester.HarvestRecipes("Sample1"); - Assert.That(recipes.Count, Is.EqualTo(1)); + var recipes = _recipeHarvester.HarvestRecipes("Sample1"); + Assert.That(recipes.Count(), Is.EqualTo(1)); - var sampleRecipe = recipes[0]; + var sampleRecipe = recipes.First(); Assert.That(sampleRecipe.Name, Is.EqualTo("cms")); Assert.That(sampleRecipe.Description, Is.EqualTo("a sample Orchard recipe describing a cms")); Assert.That(sampleRecipe.Author, Is.EqualTo("orchard")); diff --git a/src/Orchard.Tests/App.config b/src/Orchard.Tests/App.config index c8a59d19c..bbab85258 100644 --- a/src/Orchard.Tests/App.config +++ b/src/Orchard.Tests/App.config @@ -24,7 +24,7 @@ - + diff --git a/src/Orchard.Tests/Caching/CacheTests.cs b/src/Orchard.Tests/Caching/CacheTests.cs index 13af703fb..0201d3b26 100644 --- a/src/Orchard.Tests/Caching/CacheTests.cs +++ b/src/Orchard.Tests/Caching/CacheTests.cs @@ -1,4 +1,6 @@ using System; +using System.Linq; +using System.Threading; using Autofac; using NUnit.Framework; using Orchard.Caching; @@ -78,6 +80,42 @@ namespace Orchard.Tests.Caching { Is.Not.SameAs(c2.CacheManager.GetCache())); } + [Test] + public void CacheManagerIsNotBlocking() { + var hits = 0; + string result = ""; + + Enumerable.Range(0, 5).AsParallel().ForAll(x => + result = _cacheManager.Get("testItem", ctx => { + // by waiting for 100ms we expect all the calls to Get + // to enter this lambda + Thread.Sleep(100); + hits++; + return "testResult"; + }) + ); + + Assert.That(result, Is.EqualTo("testResult")); + Assert.That(hits, Is.GreaterThan(1)); + } + + [Test] + public void CacheManagerIsBlocking() { + var hits = 0; + string result = ""; + + Enumerable.Range(0, 5).AsParallel().ForAll(x => + result = _cacheManager.Get("testItem", true, ctx => { + Thread.Sleep(100); + hits++; + return "testResult"; + }) + ); + + Assert.That(result, Is.EqualTo("testResult")); + Assert.That(hits, Is.EqualTo(1)); + } + class ComponentOne { public ICacheManager CacheManager { get; set; } diff --git a/src/Orchard.Tests/DatabaseEnabledTestsBase.cs b/src/Orchard.Tests/DatabaseEnabledTestsBase.cs index e2f1fda3c..51e467618 100644 --- a/src/Orchard.Tests/DatabaseEnabledTestsBase.cs +++ b/src/Orchard.Tests/DatabaseEnabledTestsBase.cs @@ -24,7 +24,7 @@ namespace Orchard.Tests { protected string _databaseFilePath; protected ISessionFactory _sessionFactory; protected StubClock _clock; - + protected ShellSettings _shellSettings; [TestFixtureSetUp] public void InitFixture() { @@ -48,7 +48,7 @@ namespace Orchard.Tests { builder.RegisterInstance(new StubLocator(_session)).As(); builder.RegisterInstance(_clock).As(); builder.RegisterGeneric(typeof(Repository<>)).As(typeof(IRepository<>)).InstancePerLifetimeScope(); - builder.RegisterInstance(new ShellSettings { Name = ShellSettings.DefaultName, DataProvider = "SqlCe" }); + builder.RegisterInstance(_shellSettings = new ShellSettings { Name = ShellSettings.DefaultName, DataProvider = "SqlCe" }); builder.RegisterType().As().InstancePerLifetimeScope(); builder.Register(context => _sessionFactory.OpenSession()).As().InstancePerLifetimeScope(); diff --git a/src/Orchard.Tests/DisplayManagement/DefaultDisplayManagerTests.cs b/src/Orchard.Tests/DisplayManagement/DefaultDisplayManagerTests.cs index 82fb2f86c..c3e32d490 100644 --- a/src/Orchard.Tests/DisplayManagement/DefaultDisplayManagerTests.cs +++ b/src/Orchard.Tests/DisplayManagement/DefaultDisplayManagerTests.cs @@ -97,10 +97,18 @@ namespace Orchard.Tests.DisplayManagement { throw new NotImplementedException(); } + public override object Resolve(Type serviceType) { + throw new NotImplementedException(); + } + public override bool TryResolve(out T service) { throw new NotImplementedException(); } + public override bool TryResolve(Type serviceType, out object service) { + throw new NotImplementedException(); + } + public override T GetState(string name) { object value; return _state.TryGetValue(name, out value) ? (T)value : default(T); diff --git a/src/Orchard.Tests/Environment/DefaultOrchardHostTests.cs b/src/Orchard.Tests/Environment/DefaultOrchardHostTests.cs index 24594d570..8cfc8fbc5 100644 --- a/src/Orchard.Tests/Environment/DefaultOrchardHostTests.cs +++ b/src/Orchard.Tests/Environment/DefaultOrchardHostTests.cs @@ -95,6 +95,10 @@ namespace Orchard.Tests.Environment { var ext = new ExtensionDescriptor { Id = "Orchard.Framework" }; ext.Features = new[] { new FeatureDescriptor { Extension = ext, Id = ext.Id } }; yield return ext; + + var settings = new ExtensionDescriptor { Id = "Settings" }; + settings.Features = new[] { new FeatureDescriptor { Extension = settings, Id = settings.Id } }; + yield return settings; } public IEnumerable AvailableFeatures() { @@ -177,6 +181,7 @@ namespace Orchard.Tests.Environment { Assert.That(again2a, Is.SameAs(dep2a)); Assert.That(again2b, Is.SameAs(dep2b)); } + [Test] public void SingletonDependenciesShouldBeUniquePerShell() { var host = _lifetime.Resolve(); @@ -200,6 +205,7 @@ namespace Orchard.Tests.Environment { Assert.That(dep2, Is.SameAs(dep2a)); Assert.That(dep2, Is.SameAs(dep2b)); } + [Test] public void TransientDependenciesShouldBeUniquePerResolve() { var host = _lifetime.Resolve(); diff --git a/src/Orchard.Tests/Environment/DefaultWorkContextAccessorTests.cs b/src/Orchard.Tests/Environment/DefaultWorkContextAccessorTests.cs index 47f72ecb6..27ab4873b 100644 --- a/src/Orchard.Tests/Environment/DefaultWorkContextAccessorTests.cs +++ b/src/Orchard.Tests/Environment/DefaultWorkContextAccessorTests.cs @@ -28,13 +28,6 @@ namespace Orchard.Tests.Environment { container.Mock() .Setup(x => x.Current()) .Returns(() => _httpContextCurrent); - - container.Mock() - .Setup(x => x.CreateContext(It.IsAny())) - .Returns(() => new StubHttpContext()); - - container.Mock() - .Setup(x => x.Started()); } [Test] diff --git a/src/Orchard.Tests/Environment/Extensions/ExtensionManagerTests.cs b/src/Orchard.Tests/Environment/Extensions/ExtensionManagerTests.cs index 75a1d8ece..22e96abf9 100644 --- a/src/Orchard.Tests/Environment/Extensions/ExtensionManagerTests.cs +++ b/src/Orchard.Tests/Environment/Extensions/ExtensionManagerTests.cs @@ -512,7 +512,7 @@ Features: Dependencies: Beta "); - moduleExtensionFolder.Manifests.Add("Classic", @" + themeExtensionFolder.Manifests.Add("Classic", @" Name: Classic Version: 1.0.3 OrchardVersion: 1 diff --git a/src/Orchard.Tests/Environment/ShellBuilders/CompositionStrategyTests.cs b/src/Orchard.Tests/Environment/ShellBuilders/CompositionStrategyTests.cs index bb6780415..f36b467ae 100644 --- a/src/Orchard.Tests/Environment/ShellBuilders/CompositionStrategyTests.cs +++ b/src/Orchard.Tests/Environment/ShellBuilders/CompositionStrategyTests.cs @@ -9,6 +9,7 @@ using Orchard.Environment.Descriptor.Models; using Orchard.Environment.Extensions; using Orchard.Environment.Extensions.Models; using Orchard.Environment.ShellBuilders; +using Orchard.Logging; using Orchard.Tests.Environment.TestDependencies; using Orchard.Utility.Extensions; @@ -17,16 +18,22 @@ namespace Orchard.Tests.Environment.ShellBuilders { public class CompositionStrategyTests : ContainerTestBase { private CompositionStrategy _compositionStrategy; private Mock _extensionManager; + private IEnumerable _availableExtensions; + private IEnumerable _installedFeatures; + private Mock _loggerMock; protected override void Register(ContainerBuilder builder) { - _extensionManager = new Mock(MockBehavior.Loose); + _extensionManager = new Mock(); + _loggerMock = new Mock(); builder.RegisterType().AsSelf(); builder.RegisterInstance(_extensionManager.Object); + builder.RegisterInstance(_loggerMock.Object); } protected override void Resolve(ILifetimeScope container) { _compositionStrategy = container.Resolve(); + _compositionStrategy.Logger = container.Resolve(); var alphaExtension = new ExtensionDescriptor { Id = "Alpha", @@ -54,7 +61,11 @@ namespace Orchard.Tests.Environment.ShellBuilders { betaFeatureDescriptor }; - var features = new List { + _availableExtensions = new[] { + alphaExtension + }; + + _installedFeatures = new List { new Feature { Descriptor = alphaFeatureDescriptor, ExportedTypes = new List { @@ -69,16 +80,16 @@ namespace Orchard.Tests.Environment.ShellBuilders { } }; - _extensionManager.Setup(x => x.AvailableExtensions()).Returns(new List { - alphaExtension - }); + _loggerMock.Setup(x => x.IsEnabled(It.IsAny())).Returns(true); - _extensionManager.Setup(x => x.AvailableFeatures()).Returns( + _extensionManager.Setup(x => x.AvailableExtensions()).Returns(() => _availableExtensions); + + _extensionManager.Setup(x => x.AvailableFeatures()).Returns(() => _extensionManager.Object.AvailableExtensions() .SelectMany(ext => ext.Features) .ToReadOnlyCollection()); - _extensionManager.Setup(x => x.LoadFeatures(It.IsAny>())).Returns(features); + _extensionManager.Setup(x => x.LoadFeatures(It.IsAny>())).Returns(() => _installedFeatures); } [Test] @@ -87,7 +98,7 @@ namespace Orchard.Tests.Environment.ShellBuilders { var shellDescriptor = CreateShellDescriptor("Alpha", "Beta"); var shellBlueprint = _compositionStrategy.Compose(shellSettings, shellDescriptor); - Assert.That(shellBlueprint.Dependencies.Count(x => x.Type == typeof (AlphaDependency)), Is.EqualTo(1)); + Assert.That(shellBlueprint.Dependencies.Count(x => x.Type == typeof(AlphaDependency)), Is.EqualTo(1)); Assert.That(shellBlueprint.Dependencies.Count(x => x.Type == typeof(BetaDependency)), Is.EqualTo(1)); } @@ -101,6 +112,34 @@ namespace Orchard.Tests.Environment.ShellBuilders { Assert.That(shellDescriptor.Features.Count(x => x.Name == "Alpha"), Is.EqualTo(1)); } + [Test] + public void ComposeDoesNotThrowWhenFeatureStateRecordDoesNotExist() { + var shellSettings = CreateShell(); + var shellDescriptor = CreateShellDescriptor("MyFeature"); + + Assert.DoesNotThrow(() => _compositionStrategy.Compose(shellSettings, shellDescriptor)); + _loggerMock.Verify(x => x.Log(LogLevel.Warning, null, It.IsAny(), It.IsAny())); + } + + [Test] + public void ComposeThrowsWhenAutoEnabledDependencyDoesNotExist() { + var myModule = _availableExtensions.First(); + + myModule.Features = myModule.Features.Concat(new[] { + new FeatureDescriptor { + Extension = myModule, + Name = "MyFeature", + Id = "MyFeature", + Dependencies = new[] { "NonExistingFeature" } + } + }); + + var shellSettings = CreateShell(); + var shellDescriptor = CreateShellDescriptor("MyFeature"); + + Assert.Throws(() => _compositionStrategy.Compose(shellSettings, shellDescriptor)); + } + private ShellSettings CreateShell() { return new ShellSettings(); } diff --git a/src/Orchard.Tests/Environment/ShellBuilders/DefaultShellContextFactoryTests.cs b/src/Orchard.Tests/Environment/ShellBuilders/DefaultShellContextFactoryTests.cs index 7f3b1784e..dc0b13a26 100644 --- a/src/Orchard.Tests/Environment/ShellBuilders/DefaultShellContextFactoryTests.cs +++ b/src/Orchard.Tests/Environment/ShellBuilders/DefaultShellContextFactoryTests.cs @@ -56,11 +56,7 @@ namespace Orchard.Tests.Environment.ShellBuilders { _container.Mock() .Setup(x => x.Current()) - .Returns(httpContext); - - _container.Mock() - .Setup(x => x.CreateContext(It.IsAny())) - .Returns(httpContext); + .Returns(default(HttpContextBase)); var factory = _container.Resolve(); diff --git a/src/Orchard.Tests/Environment/State/DefaultProcessingEngineTests.cs b/src/Orchard.Tests/Environment/State/DefaultProcessingEngineTests.cs index c788e4b21..fb50ba290 100644 --- a/src/Orchard.Tests/Environment/State/DefaultProcessingEngineTests.cs +++ b/src/Orchard.Tests/Environment/State/DefaultProcessingEngineTests.cs @@ -43,10 +43,6 @@ namespace Orchard.Tests.Environment.State { _container.Mock() .Setup(x=>x.Current()) .Returns(httpContext); - _container.Mock() - .Setup(x => x.CreateContext(It.IsAny())) - .Returns(httpContext); - } [TearDown] diff --git a/src/Orchard.Tests/Localization/DateTimePartsTests.cs b/src/Orchard.Tests/Localization/DateTimePartsTests.cs index 4532b0448..6a3ee09e4 100644 --- a/src/Orchard.Tests/Localization/DateTimePartsTests.cs +++ b/src/Orchard.Tests/Localization/DateTimePartsTests.cs @@ -4,7 +4,7 @@ using Orchard.Localization.Models; namespace Orchard.Tests.Localization { - [TestFixture] + [TestFixture()] public class DateTimePartsTests { [Test] diff --git a/src/Orchard.Tests/Localization/DefaultDateFormatterTests.cs b/src/Orchard.Tests/Localization/DefaultDateFormatterTests.cs index 8a4226c84..a2135f09c 100644 --- a/src/Orchard.Tests/Localization/DefaultDateFormatterTests.cs +++ b/src/Orchard.Tests/Localization/DefaultDateFormatterTests.cs @@ -1,6 +1,5 @@ using System; using System.Collections.Concurrent; -using System.Collections.Generic; using System.Diagnostics; using System.Globalization; using System.Linq; @@ -13,7 +12,8 @@ using Orchard.Localization.Services; namespace Orchard.Tests.Localization { - [TestFixture] + [TestFixture()] + [Category("longrunning")] public class DefaultDateFormatterTests { [SetUp] diff --git a/src/Orchard.Tests/Localization/TestHelpers.cs b/src/Orchard.Tests/Localization/TestHelpers.cs index d9d268762..34cebb5fd 100644 --- a/src/Orchard.Tests/Localization/TestHelpers.cs +++ b/src/Orchard.Tests/Localization/TestHelpers.cs @@ -42,10 +42,18 @@ namespace Orchard.Tests.Localization { throw new NotImplementedException(); } + public override object Resolve(Type serviceType) { + throw new NotImplementedException(); + } + public override bool TryResolve(out T service) { throw new NotImplementedException(); } + public override bool TryResolve(Type serviceType, out object service) { + throw new NotImplementedException(); + } + public override T GetState(string name) { if (name == "CurrentCulture") return (T)((object)CultureName); if (name == "CurrentCalendar") return (T)((object)CalendarName); diff --git a/src/Orchard.Tests/Orchard.Framework.Tests.csproj b/src/Orchard.Tests/Orchard.Framework.Tests.csproj index 5dd02f1f0..924481c65 100644 --- a/src/Orchard.Tests/Orchard.Framework.Tests.csproj +++ b/src/Orchard.Tests/Orchard.Framework.Tests.csproj @@ -272,8 +272,7 @@ - - + @@ -295,7 +294,7 @@ - + diff --git a/src/Orchard.Tests/Properties/AssemblyInfo.cs b/src/Orchard.Tests/Properties/AssemblyInfo.cs index 41f87fe43..c3bb0f4b3 100644 --- a/src/Orchard.Tests/Properties/AssemblyInfo.cs +++ b/src/Orchard.Tests/Properties/AssemblyInfo.cs @@ -33,5 +33,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Tests/Stubs/StubMachineNameProvider.cs b/src/Orchard.Tests/Stubs/StubApplicationEnvironment.cs similarity index 56% rename from src/Orchard.Tests/Stubs/StubMachineNameProvider.cs rename to src/Orchard.Tests/Stubs/StubApplicationEnvironment.cs index c453ac6f7..2849ef6af 100644 --- a/src/Orchard.Tests/Stubs/StubMachineNameProvider.cs +++ b/src/Orchard.Tests/Stubs/StubApplicationEnvironment.cs @@ -1,12 +1,12 @@ using Orchard.Environment; namespace Orchard.Tests.Stubs { - public class StubMachineNameProvider : IMachineNameProvider { - public StubMachineNameProvider() { + public class StubApplicationEnvironment : IApplicationEnvironment { + public StubApplicationEnvironment() { MachineName = "Orchard Machine"; } public string MachineName { get; set; } - public string GetMachineName() { + public string GetEnvironmentIdentifier() { return MachineName; } } diff --git a/src/Orchard.Tests/Stubs/StubHttpContextAccessor.cs b/src/Orchard.Tests/Stubs/StubHttpContextAccessor.cs index b0368b6a7..892707b2f 100644 --- a/src/Orchard.Tests/Stubs/StubHttpContextAccessor.cs +++ b/src/Orchard.Tests/Stubs/StubHttpContextAccessor.cs @@ -1,5 +1,4 @@ using System.Web; -using Autofac; using Orchard.Mvc; namespace Orchard.Tests.Stubs { @@ -17,10 +16,6 @@ namespace Orchard.Tests.Stubs { return _httpContext; } - public HttpContextBase CreateContext(ILifetimeScope lifetimeScope) { - return _httpContext; - } - public void Set(HttpContextBase httpContext) { _httpContext = httpContext; } diff --git a/src/Orchard.Tests/Stubs/StubThreadProvider.cs b/src/Orchard.Tests/Stubs/StubThreadProvider.cs deleted file mode 100644 index 73d6ebcaf..000000000 --- a/src/Orchard.Tests/Stubs/StubThreadProvider.cs +++ /dev/null @@ -1,15 +0,0 @@ -using Orchard.Environment; - -namespace Orchard.Tests.Stubs { - public class StubThreadProvider : IThreadProvider { - public StubThreadProvider() { - ManagedThreadId = 1; - } - - public int ManagedThreadId { get; set; } - - public int GetCurrentThreadId() { - return ManagedThreadId; - } - } -} diff --git a/src/Orchard.Tests/Stubs/StubWorkContextAccessor.cs b/src/Orchard.Tests/Stubs/StubWorkContextAccessor.cs index b5fcdeeb6..76bbc5e22 100644 --- a/src/Orchard.Tests/Stubs/StubWorkContextAccessor.cs +++ b/src/Orchard.Tests/Stubs/StubWorkContextAccessor.cs @@ -13,7 +13,7 @@ namespace Orchard.Tests.Stubs { public StubWorkContextAccessor(ILifetimeScope lifetimeScope) { _lifetimeScope = lifetimeScope; - _workContext = new WorkContextImpl(_lifetimeScope); + _workContext = new WorkContextImpl(lifetimeScope); } public class WorkContextImpl : WorkContext { @@ -123,10 +123,18 @@ namespace Orchard.Tests.Stubs { return _lifetimeScope.Resolve(); } + public override object Resolve(Type serviceType) { + return _lifetimeScope.Resolve(serviceType); + } + public override bool TryResolve(out T service) { return _lifetimeScope.TryResolve(out service); } + public override bool TryResolve(Type serviceType, out object service) { + return _lifetimeScope.TryResolve(serviceType, out service); + } + public override T GetState(string name) { return (T) _contextDictonary[name]; } @@ -149,7 +157,9 @@ namespace Orchard.Tests.Stubs { } public IWorkContextScope CreateWorkContextScope() { - throw new NotSupportedException(); + var workLifetime = _lifetimeScope.BeginLifetimeScope("work"); + var workContext = new WorkContextImpl(workLifetime); + return new StubWorkContextScope(workContext, workLifetime); } } } diff --git a/src/Orchard.Tests/Stubs/StubWorkContextScope.cs b/src/Orchard.Tests/Stubs/StubWorkContextScope.cs new file mode 100644 index 000000000..a63c75957 --- /dev/null +++ b/src/Orchard.Tests/Stubs/StubWorkContextScope.cs @@ -0,0 +1,26 @@ +using Autofac; + +namespace Orchard.Tests.Stubs { + public class StubWorkContextScope : IWorkContextScope { + private readonly ILifetimeScope _lifetimeScope; + + public StubWorkContextScope(WorkContext workContext, ILifetimeScope lifetimeScope) { + _lifetimeScope = lifetimeScope; + WorkContext = workContext; + } + + public WorkContext WorkContext { get; private set; } + + public void Dispose() { + _lifetimeScope.Dispose(); + } + + public TService Resolve() { + return WorkContext.Resolve(); + } + + public bool TryResolve(out TService service) { + return WorkContext.TryResolve(out service); + } + } +} \ No newline at end of file diff --git a/src/Orchard.Tests/Tasks/DistributedLockServiceTests.cs b/src/Orchard.Tests/Tasks/DistributedLockServiceTests.cs index 5b0074c27..a0664bd60 100644 --- a/src/Orchard.Tests/Tasks/DistributedLockServiceTests.cs +++ b/src/Orchard.Tests/Tasks/DistributedLockServiceTests.cs @@ -1,10 +1,12 @@ using System; using System.Collections.Generic; +using System.Linq; using System.Threading.Tasks; using Autofac; using NUnit.Framework; using Orchard.Data; using Orchard.Environment; +using Orchard.Environment.Configuration; using Orchard.Services; using Orchard.Tasks.Locking.Records; using Orchard.Tasks.Locking.Services; @@ -15,11 +17,9 @@ namespace Orchard.Tests.Tasks { public class DistributedLockServiceTests : DatabaseEnabledTestsBase { private const string LockName = "Orchard Test Lock"; private DistributedLockService _distributedLockService; - private StubMachineNameProvider _machineNameProvider; - private StubThreadProvider _threadProvider; + private StubApplicationEnvironment _applicationEnvironment; private IRepository _distributedLockRepository; private ITransactionManager _transactionManager; - protected override IEnumerable DatabaseTypes { get { yield return typeof(DistributedLockRecord); } @@ -27,83 +27,71 @@ namespace Orchard.Tests.Tasks { public override void Register(ContainerBuilder builder) { builder.RegisterType().As(); - builder.RegisterType().As().SingleInstance(); - builder.RegisterType().As().SingleInstance(); + builder.RegisterType().As().SingleInstance(); builder.RegisterType().AsSelf(); } public override void Init() { base.Init(); _distributedLockService = _container.Resolve(); - _machineNameProvider = (StubMachineNameProvider)_container.Resolve(); - _threadProvider = (StubThreadProvider)_container.Resolve(); + _applicationEnvironment = (StubApplicationEnvironment)_container.Resolve(); _distributedLockRepository = _container.Resolve>(); _transactionManager = _container.Resolve(); } - [Test] - public void TryAcquiringLockSucceeds() { - DistributedLock @lock; - var lockAcquired = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - - Assert.That(lockAcquired, Is.True); - } - [Test] public void TryAcquiringLockTwiceOnSameMachineSucceeds() { - DistributedLock @lock; - var attempt1 = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - var attempt2 = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); + IDistributedLock @lock; + var attempt1 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out @lock); + var attempt2 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out @lock); Assert.That(attempt1, Is.True); Assert.That(attempt2, Is.True); } - [Test] - public void TryAcquiringLockTwiceOnSameMachineIncreasesLockCountTwice() { - DistributedLock @lock; - _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - var lockId = Int32.Parse(@lock.Id); - var lockRecord = _distributedLockRepository.Get(lockId); - Assert.That(lockRecord.Count, Is.EqualTo(2)); + [Test] + public void AcquiringTheLockOnTheSameMachineReturnsTheSameLock() { + IDistributedLock lock1, lock2; + _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out lock1); + _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out lock2); + + Assert.AreEqual(lock1, lock2); } [Test] - public void ReleasingLockDecreasesLockCount() { - DistributedLock @lock; - _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); + public void ReleasingSingleLockDeletesRecord() { + IDistributedLock lock1; + _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out lock1); - var lockId = Int32.Parse(@lock.Id); - var lockRecord = _distributedLockRepository.Get(lockId); - - _distributedLockService.ReleaseLock(@lock); - _session.Refresh(lockRecord); - Assert.That(lockRecord.Count, Is.EqualTo(1)); - } - - [Test] - public void ReleasingLockAndCountReachesZeroDeletesLock() - { - DistributedLock @lock; - _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - - var lockId = Int32.Parse(@lock.Id); - _distributedLockService.ReleaseLock(@lock); - var lockRecord = _distributedLockRepository.Get(lockId); + lock1.Dispose(); + var lockRecord = _distributedLockRepository.Table.FirstOrDefault(); Assert.That(lockRecord, Is.Null); } + [Test] + public void ReleasingFirstLockDoesntDeleteRecord() { + IDistributedLock lock1, lock2; + _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out lock1); + _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out lock2); + + lock1.Dispose(); + var lockRecord = _distributedLockRepository.Table.FirstOrDefault(); + Assert.That(lockRecord, Is.Not.Null); + + lock2.Dispose(); + lockRecord = _distributedLockRepository.Table.FirstOrDefault(); + Assert.That(lockRecord, Is.Null); + } + [Test] public void TryAcquiringLockTwiceFails() { - DistributedLock @lock; - _machineNameProvider.MachineName = "Orchard Test Machine 1"; - var attempt1 = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - _machineNameProvider.MachineName = "Orchard Test Machine 2"; - var attempt2 = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); + IDistributedLock @lock; + _applicationEnvironment.MachineName = "Orchard Test Machine 1"; + var attempt1 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out @lock); + _applicationEnvironment.MachineName = "Orchard Test Machine 2"; + var attempt2 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out @lock); Assert.That(attempt1, Is.True); Assert.That(attempt2, Is.False); @@ -111,66 +99,88 @@ namespace Orchard.Tests.Tasks { [Test] public void TryAcquiringNonExpiredActiveLockFails() { - DistributedLock @lock; - CreateNonExpiredActiveLock("Other Machine", threadId: null); - var success = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromHours(1), null, out @lock); + IDistributedLock @lock; + CreateNonExpiredActiveLock("Other Machine"); + var success = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromHours(1), out @lock); Assert.That(success, Is.False); } [Test] - public void TryAcquiringNonExpiredButInactiveLockSucceeds() { - DistributedLock @lock; - CreateNonExpiredButInactiveLock("Other Machine", threadId: null); - var success = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromHours(1), null, out @lock); + public void TryAcquiringNonExpiredButInactiveLockFromOtherMachineFails() { + IDistributedLock @lock; + CreateNonExpiredActiveLock("Other Machine"); + var success = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromHours(1), out @lock); + + Assert.That(success, Is.False); + } + + [Test] + public void TryAcquiringNonExpiredButInactiveLockFromSameMachineSucceeds() { + IDistributedLock @lock; + CreateNonExpiredActiveLock("Orchard Machine"); + var success = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromHours(1), out @lock); Assert.That(success, Is.True); } [Test] public void TryAcquiringExpiredButActiveLockSucceeds() { - DistributedLock @lock; - CreateExpiredButActiveLock("Other Machine", threadId: null); - var success = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromHours(1), null, out @lock); + IDistributedLock @lock; + CreateExpiredButActiveLock("Other Machine"); + var success = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromHours(1), out @lock); Assert.That(success, Is.True); } [Test] public void TryAcquiringNonExpiredAndActiveLockFromCurrentOwnerSucceeds() { - DistributedLock @lock; - CreateNonExpiredActiveLock(GetMachineName(), threadId: null); - var success = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromHours(1), null, out @lock); + IDistributedLock @lock; + CreateNonExpiredActiveLock(GetMachineName()); + var success = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromHours(1), out @lock); Assert.That(success, Is.True); } [Test] public void AcquiringNonExpiredAndActiveLockFromDifferentOwnerThrowsTimeoutException() { - CreateNonExpiredActiveLock("Other Machine", threadId: null); - Assert.Throws(() => _distributedLockService.AcquireLockForMachine(LockName, TimeSpan.FromHours(1), TimeSpan.Zero)); + CreateNonExpiredActiveLock("Other Machine"); + Assert.Throws(() => _distributedLockService.AcquireLock(LockName, TimeSpan.FromHours(1), TimeSpan.Zero)); } [Test] public void MultipleAcquisitionsFromDifferentMachinesShouldFail() { - DistributedLock @lock; - _machineNameProvider.MachineName = "Orchard Test Machine 1"; - var attempt1 = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); - _machineNameProvider.MachineName = "Orchard Test Machine 2"; - var attempt2 = _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock); + IDistributedLock @lock; + _applicationEnvironment.MachineName = "Orchard Test Machine 1"; + var attempt1 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromMinutes(60), out @lock); + _applicationEnvironment.MachineName = "Orchard Test Machine 2"; + var attempt2 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromMinutes(60), out @lock); Assert.That(attempt1, Is.True); Assert.That(attempt2, Is.False); } + [Test] + public void MultipleAcquisitionsFromDifferentMachinesOnDifferentTenantShouldSucceed() { + IDistributedLock @lock; + _applicationEnvironment.MachineName = "Orchard Test Machine 1"; + var attempt1 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out @lock); + _applicationEnvironment.MachineName = "Orchard Test Machine 2"; + _shellSettings.Name = "Foo"; + var attempt2 = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromSeconds(60), out @lock); + + Assert.That(attempt1, Is.True); + Assert.That(attempt2, Is.True); + } + [Test] public void MultithreadedAcquisitionsShouldNotCauseTransactionErrors() { var tasks = new List(); for (var i = 0; i < 10; i++) { var task = Task.Factory.StartNew(() => { - DistributedLock @lock; - Assert.DoesNotThrow(() => _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromHours(1), null, out @lock)); + IDistributedLock @lock; + Assert.DoesNotThrow(() => _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromHours(1), out @lock)); }); tasks.Add(task); @@ -179,61 +189,53 @@ namespace Orchard.Tests.Tasks { Task.WaitAll(tasks.ToArray()); } - [Test] - public void MixedScopeAcquisitionsShouldThrow() { - DistributedLock @lock; - Assert.DoesNotThrow(() => _distributedLockService.TryAcquireLockForMachine(LockName, TimeSpan.FromSeconds(60), null, out @lock)); - Assert.Throws(() => _distributedLockService.TryAcquireLockForThread(LockName, TimeSpan.FromSeconds(60), null, out @lock)); - } - [Test] public void TryAcquireActiveLockWithNullTimeoutReturnsFalseImmediately() { - CreateNonExpiredActiveLock("Other Machine", null); + CreateNonExpiredActiveLock("Other Machine"); - DistributedLock @lock; - var acquired = _distributedLockService.TryAcquireLockForThread(LockName, TimeSpan.FromMinutes(1), null, out @lock); + IDistributedLock @lock; + var acquired = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromMinutes(1), out @lock); Assert.That(acquired, Is.False); } [Test] public void ActiveLockWithUndefinedValidUntilNeverExpires() { - CreateNonExpiredActiveLockThatNeverExpires("Other Machine", null); + CreateNonExpiredActiveLockThatNeverExpires("Other Machine"); + _clock.Advance(DateTime.MaxValue - _clock.UtcNow); // Fast forward to the End of Time. - DistributedLock @lock; - var acquired = _distributedLockService.TryAcquireLockForThread(LockName, TimeSpan.FromMinutes(1), null, out @lock); + IDistributedLock @lock; + var acquired = _distributedLockService.TryAcquireLock(LockName, TimeSpan.FromMinutes(1), out @lock); Assert.That(acquired, Is.False); } [Test] public void ActiveLockWithUndefinedValidUntilNeverExpiresUntilReleased() { - DistributedLock @lock; + IDistributedLock @lock; // Create a never expiring lock. - _machineNameProvider.MachineName = "Orchard Test Machine 1"; - var attempt1 = _distributedLockService.TryAcquireLockForThread(LockName, maxValidFor: null, timeout: null, @lock: out @lock); - + _applicationEnvironment.MachineName = "Orchard Test Machine 1"; + var attempt1 = _distributedLockService.TryAcquireLock(LockName, maxValidFor: null, timeout: null, dLock: out @lock); + // Release the lock. - _distributedLockService.ReleaseLock(@lock); + @lock.Dispose(); // Acquire the lock from another machine. - _machineNameProvider.MachineName = "Orchard Test Machine 2"; - var attempt2 = _distributedLockService.TryAcquireLockForThread(LockName, maxValidFor: null, timeout: null, @lock: out @lock); + _applicationEnvironment.MachineName = "Orchard Test Machine 2"; + var attempt2 = _distributedLockService.TryAcquireLock(LockName, maxValidFor: null, timeout: null, dLock: out @lock); // Validate the results. Assert.That(attempt1, Is.True); Assert.That(attempt2, Is.True); } - private DistributedLockRecord CreateLockRecord(int count, DateTime createdUtc, DateTime? validUntilUtc, string machineName, int? threadId) { + private DistributedLockRecord CreateLockRecord(DateTime createdUtc, DateTime? validUntilUtc, string machineName) { var record = new DistributedLockRecord { - Name = LockName, - Count = count, + Name = String.Format("DistributedLock:{0}:{1}", ShellSettings.DefaultName, LockName), CreatedUtc = createdUtc, ValidUntilUtc = validUntilUtc, MachineName = machineName, - ThreadId = threadId }; _distributedLockRepository.Create(record); @@ -241,32 +243,23 @@ namespace Orchard.Tests.Tasks { return record; } - private DistributedLockRecord CreateNonExpiredActiveLock(string machineName, int? threadId) { + private DistributedLockRecord CreateNonExpiredActiveLock(string machineName) { var now = _clock.UtcNow; - return CreateLockRecord(1, now, now + TimeSpan.FromHours(1), machineName, threadId); + return CreateLockRecord(now, now + TimeSpan.FromHours(1), machineName); } - private DistributedLockRecord CreateNonExpiredButInactiveLock(string machineName, int? threadId) { + private DistributedLockRecord CreateExpiredButActiveLock(string machineName) { var now = _clock.UtcNow; - return CreateLockRecord(0, now, now + TimeSpan.FromHours(1), machineName, threadId); + return CreateLockRecord(now, now - TimeSpan.FromHours(1), machineName); } - private DistributedLockRecord CreateExpiredButActiveLock(string machineName, int? threadId) { + private DistributedLockRecord CreateNonExpiredActiveLockThatNeverExpires(string machineName) { var now = _clock.UtcNow; - return CreateLockRecord(1, now, now - TimeSpan.FromHours(1), machineName, threadId); - } - - private DistributedLockRecord CreateNonExpiredActiveLockThatNeverExpires(string machineName, int? threadId) { - var now = _clock.UtcNow; - return CreateLockRecord(1, now, null, machineName, threadId); + return CreateLockRecord(now, null, machineName); } private string GetMachineName() { - return _machineNameProvider.GetMachineName(); - } - - private int GetThreadId() { - return _threadProvider.GetCurrentThreadId(); + return _applicationEnvironment.GetEnvironmentIdentifier(); } } } diff --git a/src/Orchard.Tests/Tasks/LockTests.cs b/src/Orchard.Tests/Tasks/LockTests.cs deleted file mode 100644 index d94218428..000000000 --- a/src/Orchard.Tests/Tasks/LockTests.cs +++ /dev/null @@ -1,39 +0,0 @@ -using Autofac; -using Moq; -using NUnit.Framework; -using Orchard.Tasks.Locking.Services; - -namespace Orchard.Tests.Tasks { - [TestFixture] - public class LockTests : ContainerTestBase { - private const string LockName = "Orchard Test Lock"; - private const string LockId = "1"; - private Mock _distributedLockServiceMock; - private DistributedLock _lock; - - protected override void Register(ContainerBuilder builder) { - _distributedLockServiceMock = new Mock(); - builder.RegisterInstance(_distributedLockServiceMock.Object); - } - - protected override void Resolve(ILifetimeScope container) { - _lock = DistributedLock.ForMachine(_distributedLockServiceMock.Object, LockName, "Orchard Test Machine", LockId); - } - - [Test] - public void DisposeInvokesDistributedLockServiceDisposeLock() { - _lock.Dispose(); - - _distributedLockServiceMock.Verify(service => service.ReleaseLock(_lock), Times.Exactly(1)); - } - - [Test] - public void DisposingMultipleTimesInvokesDistributedLockServiceDisposeLockOnce() { - _lock.Dispose(); - _lock.Dispose(); - _lock.Dispose(); - - _distributedLockServiceMock.Verify(service => service.ReleaseLock(_lock), Times.Exactly(1)); - } - } -} \ No newline at end of file diff --git a/src/Orchard.Tests/Tasks/SweepGeneratorTests.cs b/src/Orchard.Tests/Tasks/SweepGeneratorTests.cs index e851df66c..8e468800a 100644 --- a/src/Orchard.Tests/Tasks/SweepGeneratorTests.cs +++ b/src/Orchard.Tests/Tasks/SweepGeneratorTests.cs @@ -23,10 +23,6 @@ namespace Orchard.Tests.Tasks { .Setup(x => x.Current()) .Returns(() => null); - container.Mock() - .Setup(x => x.CreateContext(It.IsAny())) - .Returns(() => new StubHttpContext()); - container.Mock() .Setup(x => x.Started()); } diff --git a/src/Orchard.Tests/Time/TimeZoneSelectorTests.cs b/src/Orchard.Tests/Time/TimeZoneSelectorTests.cs index b0cee916b..18b089be0 100644 --- a/src/Orchard.Tests/Time/TimeZoneSelectorTests.cs +++ b/src/Orchard.Tests/Time/TimeZoneSelectorTests.cs @@ -69,10 +69,18 @@ namespace Orchard.Tests.Time { throw new NotImplementedException(); } + public override object Resolve(Type serviceType) { + throw new NotImplementedException(); + } + public override bool TryResolve(out T service) { throw new NotImplementedException(); } + public override bool TryResolve(Type serviceType, out object service) { + throw new NotImplementedException(); + } + public override T GetState(string name) { return default(T); } diff --git a/src/Orchard.WarmupStarter/Properties/AssemblyInfo.cs b/src/Orchard.WarmupStarter/Properties/AssemblyInfo.cs index 4f4ee0785..3bf0779c8 100644 --- a/src/Orchard.WarmupStarter/Properties/AssemblyInfo.cs +++ b/src/Orchard.WarmupStarter/Properties/AssemblyInfo.cs @@ -32,8 +32,8 @@ using System.Security; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] // Enable web application to call this assembly in Full Trust [assembly: AllowPartiallyTrustedCallers] diff --git a/src/Orchard.WarmupStarter/Starter.cs b/src/Orchard.WarmupStarter/Starter.cs index ffa1f0763..b0b328da4 100644 --- a/src/Orchard.WarmupStarter/Starter.cs +++ b/src/Orchard.WarmupStarter/Starter.cs @@ -89,9 +89,9 @@ namespace Orchard.WarmupStarter { var result = _initialization(application); _initializationResult = result; } - catch (Exception e) { + catch (Exception ex) { lock (_synLock) { - _error = e; + _error = ex; _previousError = null; } } diff --git a/src/Orchard.Web.Tests/Properties/AssemblyInfo.cs b/src/Orchard.Web.Tests/Properties/AssemblyInfo.cs index 788066b9f..0338f1a5b 100644 --- a/src/Orchard.Web.Tests/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web.Tests/Properties/AssemblyInfo.cs @@ -33,5 +33,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Config/Host.config b/src/Orchard.Web/Config/Host.config index 9e6b2d119..94eefe177 100644 --- a/src/Orchard.Web/Config/Host.config +++ b/src/Orchard.Web/Config/Host.config @@ -8,7 +8,7 @@ - + diff --git a/src/Orchard.Web/Config/HostComponents.config b/src/Orchard.Web/Config/HostComponents.config index dd1a321d6..f91398632 100644 --- a/src/Orchard.Web/Config/HostComponents.config +++ b/src/Orchard.Web/Config/HostComponents.config @@ -106,5 +106,19 @@ + + + + + + + + + + + + + + diff --git a/src/Orchard.Web/Config/Sites.config b/src/Orchard.Web/Config/Sites.config index b64e22c3f..852db503e 100644 --- a/src/Orchard.Web/Config/Sites.config +++ b/src/Orchard.Web/Config/Sites.config @@ -8,7 +8,7 @@ /,o=/(^[\s]+|[\s]+$)/g,a=[];t=t.split(r),n=t.length;for(var s=0;n>s;s++){var u=t[s];if(u.length>0&&"WEBVTT"!==u&&u.match(i)){var l=u.split(i),d=t[s+1];a.push({start:e(l[0].replace(o,"")),end:e(l[1].replace(o,"")),data:d})}}return Q.when(a)}}},MediaPlayer.rules.BaseRulesCollection=function(){ +"use strict";var e=[];return{downloadRatioRule:void 0,insufficientBufferRule:void 0,getRules:function(){return Q.when(e)},setup:function(){var e=this;e.getRules().then(function(t){t.push(e.downloadRatioRule),t.push(e.insufficientBufferRule)})}}},MediaPlayer.rules.BaseRulesCollection.prototype={constructor:MediaPlayer.rules.BaseRulesCollection},MediaPlayer.rules.DownloadRatioRule=function(){"use strict";var e=function(e,t,n){var r=this,i=Q.defer();return r.manifestExt.getRepresentationFor(e,n).then(function(e){r.manifestExt.getBandwidth(e).then(function(e){i.resolve(e/t)})}),i.promise};return{debug:void 0,manifestExt:void 0,checkIndex:function(t,n,r){var i,o,a,s,u,l,d,c,f,g,h=this,p=n.HttpList,m=.75;return h.debug.log("Checking download ratio rule..."),n?null===p||void 0===p||0===p.length?(h.debug.log("No requests made for this stream yet, bailing."),Q.when(new MediaPlayer.rules.SwitchRequest)):(i=p[p.length-1],a=(i.tfinish.getTime()-i.trequest.getTime())/1e3,o=(i.tfinish.getTime()-i.tresponse.getTime())/1e3,0>=a?(h.debug.log("Don't know how long the download of the last fragment took, bailing."),Q.when(new MediaPlayer.rules.SwitchRequest)):null===i.mediaduration||void 0===i.mediaduration||i.mediaduration<=0?(h.debug.log("Don't know the duration of the last media fragment, bailing."),Q.when(new MediaPlayer.rules.SwitchRequest)):(d=Q.defer(),u=i.mediaduration/a,s=i.mediaduration/o*m,isNaN(s)||isNaN(u)?(h.debug.log("Total time: "+a+"s"),h.debug.log("Download time: "+o+"s"),h.debug.log("The ratios are NaN, bailing."),Q.when(new MediaPlayer.rules.SwitchRequest)):(h.debug.log("Total ratio: "+u),h.debug.log("Download ratio: "+s),h.debug.log("Download ratio: "+s),isNaN(s)?(h.debug.log("Invalid ratio, bailing."),d.resolve(new MediaPlayer.rules.SwitchRequest)):1>s?(h.debug.log("Download ratio is poor."),t>0?(h.debug.log("We are not at the lowest bitrate, so switch down."),h.manifestExt.getRepresentationFor(t-1,r).then(function(e){h.manifestExt.getBandwidth(e).then(function(e){h.manifestExt.getRepresentationFor(t,r).then(function(n){h.manifestExt.getBandwidth(n).then(function(n){l=e/n,h.debug.log("Switch ratio: "+l),l>s?(h.debug.log("Things must be going pretty bad, switch all the way down."),d.resolve(new MediaPlayer.rules.SwitchRequest(0))):(h.debug.log("Things could be better, so just switch down one index."),d.resolve(new MediaPlayer.rules.SwitchRequest(t-1)))})})})})):(h.debug.log("We are at the lowest bitrate and cannot switch down, use current."),d.resolve(new MediaPlayer.rules.SwitchRequest(t)))):(h.debug.log("Download ratio is good."),h.manifestExt.getRepresentationCount(r).then(function(n){n-=1,n>t?(h.debug.log("We are not at the highest bitrate, so switch up."),h.manifestExt.getRepresentationFor(t+1,r).then(function(i){h.manifestExt.getBandwidth(i).then(function(i){h.manifestExt.getRepresentationFor(t,r).then(function(o){h.manifestExt.getBandwidth(o).then(function(o){if(l=i/o,h.debug.log("Switch ratio: "+l),s>=l)if(s>1e3)h.debug.log("Tons of bandwidth available, go all the way up."),d.resolve(new MediaPlayer.rules.SwitchRequest(n-1));else if(s>100)h.debug.log("Just enough bandwidth available, switch up one."),d.resolve(new MediaPlayer.rules.SwitchRequest(t+1));else{for(h.debug.log("Not exactly sure where to go, so do some math."),f=-1,c=[];(f+=1)f&&!(st&&(u=MediaPlayer.rules.SwitchRequest.prototype.STRONG,a.debug.log("Apply STRONG to buffer rule.")),s?(a.debug.log("The buffer ran dry recently, switch down."),Q.when(new MediaPlayer.rules.SwitchRequest(n-1,u))):e>t?(a.debug.log("Too many dry buffer hits, quit switching bitrates."),Q.when(new MediaPlayer.rules.SwitchRequest(n,u))):Q.when(new MediaPlayer.rules.SwitchRequest(MediaPlayer.rules.SwitchRequest.prototype.NO_CHANGE,u)))))}}},MediaPlayer.rules.InsufficientBufferRule.prototype={constructor:MediaPlayer.rules.InsufficientBufferRule},MediaPlayer.rules.LimitSwitchesRule=function(){"use strict";var e=10,t=2e4,n=5,r=0;return{debug:void 0,checkIndex:function(i,o){if(r>0)return r-=1,Q.when(new MediaPlayer.rules.SwitchRequest(i,MediaPlayer.rules.SwitchRequest.prototype.STRONG));var a,s,u,l=this,d=!1,c=(new Date).getTime(),f=o.RepSwitchList.length;for(l.debug.log("Checking limit switches rule..."),u=f-1;u>=0;u-=1){if(a=o.RepSwitchList[u],s=c-a.t.getTime(),s>=t){l.debug.log("Reached time limit, bailing.");break}if(u>=e){l.debug.log("Found too many switches within validation time, force the stream to not change."),d=!0;break}}return d?(l.debug.log("Wait some time before allowing another switch."),r=n,Q.when(new MediaPlayer.rules.SwitchRequest(i,MediaPlayer.rules.SwitchRequest.prototype.STRONG))):Q.when(new MediaPlayer.rules.SwitchRequest(MediaPlayer.rules.SwitchRequest.prototype.NO_CHANGE,MediaPlayer.rules.SwitchRequest.prototype.STRONG))}}},MediaPlayer.rules.LimitSwitchesRule.prototype={constructor:MediaPlayer.rules.LimitSwitchesRule},MediaPlayer.rules.SwitchRequest=function(e,t){"use strict";this.quality=e,this.priority=t,void 0===this.quality&&(this.quality=999),void 0===this.priority&&(this.priority=.5)},MediaPlayer.rules.SwitchRequest.prototype={constructor:MediaPlayer.rules.SwitchRequest,NO_CHANGE:999,DEFAULT:.5,STRONG:1,WEAK:0},MediaPlayer.models.MetricsList=function(){"use strict";return{TcpList:[],HttpList:[],RepSwitchList:[],BufferLevel:[],PlayList:[],DroppedFrames:[]}},MediaPlayer.models.MetricsList.prototype={constructor:MediaPlayer.models.MetricsList},MediaPlayer.vo.SegmentRequest=function(){"use strict";this.action="download",this.startTime=0/0,this.streamType=null,this.type=null,this.duration=0/0,this.timescale=0/0,this.range=null,this.url=null,this.requestStartDate=null,this.firstByteDate=null,this.requestEndDate=null,this.deferred=null,this.quality=0/0,this.index=0/0},MediaPlayer.vo.SegmentRequest.prototype={constructor:MediaPlayer.vo.SegmentRequest,ACTION_DOWNLOAD:"download",ACTION_COMPLETE:"complete"},MediaPlayer.vo.metrics.BufferLevel=function(){"use strict";this.t=null,this.level=null},MediaPlayer.vo.metrics.BufferLevel.prototype={constructor:MediaPlayer.vo.metrics.BufferLevel},MediaPlayer.vo.metrics.DroppedFrames=function(){"use strict";this.time=null,this.droppedFrames=null},MediaPlayer.vo.metrics.DroppedFrames.prototype={constructor:MediaPlayer.vo.metrics.DroppedFrames},MediaPlayer.vo.metrics.HTTPRequest=function(){"use strict";this.tcpid=null,this.type=null,this.url=null,this.actualurl=null,this.range=null,this.trequest=null,this.tresponse=null,this.tfinish=null,this.responsecode=null,this.interval=null,this.mediaduration=null,this.trace=[]},MediaPlayer.vo.metrics.HTTPRequest.prototype={constructor:MediaPlayer.vo.metrics.HTTPRequest},MediaPlayer.vo.metrics.HTTPRequest.Trace=function(){"use strict";this.s=null,this.d=null,this.b=[]},MediaPlayer.vo.metrics.HTTPRequest.Trace.prototype={constructor:MediaPlayer.vo.metrics.HTTPRequest.Trace},MediaPlayer.vo.metrics.PlayList=function(){"use strict";this.start=null,this.mstart=null,this.starttype=null,this.trace=[]},MediaPlayer.vo.metrics.PlayList.Trace=function(){"use strict";this.representationid=null,this.subreplevel=null,this.start=null,this.mstart=null,this.duration=null,this.playbackspeed=null,this.stopreason=null},MediaPlayer.vo.metrics.PlayList.prototype={constructor:MediaPlayer.vo.metrics.PlayList},MediaPlayer.vo.metrics.PlayList.INITIAL_PLAY_START_REASON="initial_start",MediaPlayer.vo.metrics.PlayList.SEEK_START_REASON="seek",MediaPlayer.vo.metrics.PlayList.Trace.prototype={constructor:MediaPlayer.vo.metrics.PlayList.Trace()},MediaPlayer.vo.metrics.PlayList.Trace.USER_REQUEST_STOP_REASON="user_request",MediaPlayer.vo.metrics.PlayList.Trace.REPRESENTATION_SWITCH_STOP_REASON="representation_switch",MediaPlayer.vo.metrics.PlayList.Trace.END_OF_CONTENT_STOP_REASON="end_of_content",MediaPlayer.vo.metrics.PlayList.Trace.REBUFFERING_REASON="rebuffering",MediaPlayer.vo.metrics.RepresentationSwitch=function(){"use strict";this.t=null,this.mt=null,this.to=null,this.lto=null},MediaPlayer.vo.metrics.RepresentationSwitch.prototype={constructor:MediaPlayer.vo.metrics.RepresentationSwitch},MediaPlayer.vo.metrics.TCPConnection=function(){"use strict";this.tcpid=null,this.dest=null,this.topen=null,this.tclose=null,this.tconnect=null},MediaPlayer.vo.metrics.TCPConnection.prototype={constructor:MediaPlayer.vo.metrics.TCPConnection}; \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Services/Jobs/JobProcessor.cs b/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Services/Jobs/JobProcessor.cs index 4a066e68a..4fd8487fe 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Services/Jobs/JobProcessor.cs +++ b/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Services/Jobs/JobProcessor.cs @@ -2,7 +2,6 @@ using System.Collections.Generic; using System.Linq; using System.Text; -using System.Threading; using Microsoft.WindowsAzure.MediaServices.Client; using Orchard.Azure.MediaServices.Helpers; using Orchard.Azure.MediaServices.Models; @@ -18,8 +17,6 @@ using Orchard.Tasks.Locking.Services; namespace Orchard.Azure.MediaServices.Services.Jobs { public class JobProcessor : Component, IBackgroundTask { - private static readonly object _sweepLock = new object(); - private readonly IWamsClient _wamsClient; private readonly IAssetManager _assetManager; private readonly IJobManager _jobManager; @@ -41,126 +38,125 @@ namespace Orchard.Azure.MediaServices.Services.Jobs { } public void Sweep() { - if (Monitor.TryEnter(_sweepLock)) { - try { - Logger.Debug("Beginning sweep."); + Logger.Debug("Beginning sweep."); - if (!_orchardServices.WorkContext.CurrentSite.As().IsValid()) { - Logger.Debug("Settings are invalid; going back to sleep."); - return; - } + try { + if (!_orchardServices.WorkContext.CurrentSite.As().IsValid()) { + Logger.Debug("Settings are invalid; going back to sleep."); + return; + } - // Only allow this task to run on one farm node at a time. - DistributedLock @lock; - if (_distributedLockService.TryAcquireLockForMachine(GetType().FullName, TimeSpan.FromHours(1), out @lock)) { - using (@lock) { - var jobs = _jobManager.GetActiveJobs().ToDictionary(job => job.WamsJobId); + // Only allow this task to run on one farm node at a time. + IDistributedLock @lock; + if (_distributedLockService.TryAcquireLock(GetType().FullName, TimeSpan.FromHours(1), out @lock)) { + using (@lock) { + var jobs = _jobManager.GetActiveJobs().ToDictionary(job => job.WamsJobId); - if (!jobs.Any()) { - Logger.Debug("No open jobs were found; going back to sleep."); - return; + if (!jobs.Any()) { + Logger.Debug("No open jobs were found; going back to sleep."); + return; + } + + Logger.Information("Beginning processing of {0} open jobs.", jobs.Count()); + + var wamsJobs = _wamsClient.GetJobsById(jobs.Keys); + + foreach (var wamsJob in wamsJobs) { + Logger.Information("Processing job '{0}'...", wamsJob.Name); + + var job = jobs[wamsJob.Id]; + var tasks = job.Tasks.ToDictionary(task => task.WamsTaskId); + var wamsTasks = wamsJob.Tasks.ToArray(); + + foreach (var wamsTask in wamsTasks) { + var task = tasks[wamsTask.Id]; + task.Status = MapWamsJobState(wamsTask.State); + task.PercentComplete = (int)wamsTask.Progress; } - Logger.Information("Beginning processing of {0} open jobs.", jobs.Count()); + var previousStatus = job.Status; + var wamsJobErrors = HarvestWamsJobErrors(wamsJob).ToArray(); - var wamsJobs = _wamsClient.GetJobsById(jobs.Keys); + job.CreatedUtc = wamsJob.Created; + job.StartedUtc = wamsJob.StartTime; + job.FinishedUtc = wamsJob.EndTime; + job.Status = MapWamsJobState(wamsJob.State); + job.ErrorMessage = GetAggregateErrorMessage(wamsJobErrors); - foreach (var wamsJob in wamsJobs) { - Logger.Information("Processing job '{0}'...", wamsJob.Name); + LogWamsJobErrors(wamsJobErrors); - var job = jobs[wamsJob.Id]; - var tasks = job.Tasks.ToDictionary(task => task.WamsTaskId); - var wamsTasks = wamsJob.Tasks.ToArray(); + if (job.Status != previousStatus) { + if (job.Status == JobStatus.Finished) { + Logger.Information("Job '{0}' was finished in WAMS; creating locators.", wamsJob.Name); - foreach (var wamsTask in wamsTasks) { - var task = tasks[wamsTask.Id]; - task.Status = MapWamsJobState(wamsTask.State); - task.PercentComplete = (int)wamsTask.Progress; - } + var lastTask = job.Tasks.Last(); + var lastWamsTask = wamsTasks.Single(task => task.Id == lastTask.WamsTaskId); + var outputAsset = lastWamsTask.OutputAssets.First(); + var outputAssetName = !String.IsNullOrWhiteSpace(job.OutputAssetName) ? job.OutputAssetName : lastWamsTask.Name; + var outputAssetDescription = job.OutputAssetDescription.TrimSafe(); + var encoderMetadataXml = _wamsClient.GetEncoderMetadataXml(outputAsset).Result; + var cloudVideoPart = job.CloudVideoPart; + var wamsLocators = _wamsClient.CreateLocatorsAsync(outputAsset, WamsLocatorCategory.Private).Result; - var previousStatus = job.Status; - var wamsJobErrors = HarvestWamsJobErrors(wamsJob).ToArray(); + // HACK: Temporary workaround to disable dynamic packaging for VC1-based assets. In future versions + // this will be implemented more robustly by testing all the dynamic URLs to see which ones work + // and only store and use the working ones. + var forceNonDynamicAsset = lastWamsTask.Configuration.StartsWith("VC1"); - job.CreatedUtc = wamsJob.Created; - job.StartedUtc = wamsJob.StartTime; - job.FinishedUtc = wamsJob.EndTime; - job.Status = MapWamsJobState(wamsJob.State); - job.ErrorMessage = GetAggregateErrorMessage(wamsJobErrors); + if (wamsLocators.OnDemandLocator != null && !forceNonDynamicAsset) { + _assetManager.CreateAssetFor(cloudVideoPart, asset => { + asset.IncludeInPlayer = true; + asset.Name = outputAssetName; + asset.Description = outputAssetDescription; + asset.EncodingPreset = lastTask.HarvestAssetName; + asset.WamsPrivateLocatorId = wamsLocators.SasLocator.Id; + asset.WamsPrivateLocatorUrl = wamsLocators.SasLocator.Url; + asset.WamsPrivateOnDemandLocatorId = wamsLocators.OnDemandLocator.Id; + asset.WamsPrivateOnDemandLocatorUrl = wamsLocators.OnDemandLocator.Url; + asset.WamsManifestFilename = wamsLocators.OnDemandManifestFilename; + asset.WamsAssetId = outputAsset.Id; + asset.WamsEncoderMetadataXml = encoderMetadataXml; + asset.UploadState.Status = AssetUploadStatus.Uploaded; + asset.PublishState.Status = AssetPublishStatus.None; + }); + } + else { + _assetManager.CreateAssetFor(cloudVideoPart, asset => { + asset.IncludeInPlayer = true; + asset.Name = outputAssetName; + asset.Description = outputAssetDescription; + asset.EncodingPreset = lastTask.HarvestAssetName; + asset.WamsPrivateLocatorId = wamsLocators.SasLocator.Id; + asset.WamsPrivateLocatorUrl = wamsLocators.SasLocator.Url; + asset.WamsAssetId = outputAsset.Id; + asset.WamsEncoderMetadataXml = encoderMetadataXml; + asset.UploadState.Status = AssetUploadStatus.Uploaded; + asset.PublishState.Status = AssetPublishStatus.None; + }); + } - LogWamsJobErrors(wamsJobErrors); - - if (job.Status != previousStatus) { - if (job.Status == JobStatus.Finished) { - Logger.Information("Job '{0}' was finished in WAMS; creating locators.", wamsJob.Name); - - var lastTask = job.Tasks.Last(); - var lastWamsTask = wamsTasks.Single(task => task.Id == lastTask.WamsTaskId); - var outputAsset = lastWamsTask.OutputAssets.First(); - var outputAssetName = !String.IsNullOrWhiteSpace(job.OutputAssetName) ? job.OutputAssetName : lastWamsTask.Name; - var outputAssetDescription = job.OutputAssetDescription.TrimSafe(); - var encoderMetadataXml = _wamsClient.GetEncoderMetadataXml(outputAsset).Result; - var cloudVideoPart = job.CloudVideoPart; - var wamsLocators = _wamsClient.CreateLocatorsAsync(outputAsset, WamsLocatorCategory.Private).Result; - - // HACK: Temporary workaround to disable dynamic packaging for VC1-based assets. In future versions - // this will be implemented more robustly by testing all the dynamic URLs to see which ones work - // and only store and use the working ones. - var forceNonDynamicAsset = lastWamsTask.Configuration.StartsWith("VC1"); - - if (wamsLocators.OnDemandLocator != null && !forceNonDynamicAsset) { - _assetManager.CreateAssetFor(cloudVideoPart, asset => { - asset.IncludeInPlayer = true; - asset.Name = outputAssetName; - asset.Description = outputAssetDescription; - asset.EncodingPreset = lastTask.HarvestAssetName; - asset.WamsPrivateLocatorId = wamsLocators.SasLocator.Id; - asset.WamsPrivateLocatorUrl = wamsLocators.SasLocator.Url; - asset.WamsPrivateOnDemandLocatorId = wamsLocators.OnDemandLocator.Id; - asset.WamsPrivateOnDemandLocatorUrl = wamsLocators.OnDemandLocator.Url; - asset.WamsManifestFilename = wamsLocators.OnDemandManifestFilename; - asset.WamsAssetId = outputAsset.Id; - asset.WamsEncoderMetadataXml = encoderMetadataXml; - asset.UploadState.Status = AssetUploadStatus.Uploaded; - asset.PublishState.Status = AssetPublishStatus.None; - }); - } - else { - _assetManager.CreateAssetFor(cloudVideoPart, asset => { - asset.IncludeInPlayer = true; - asset.Name = outputAssetName; - asset.Description = outputAssetDescription; - asset.EncodingPreset = lastTask.HarvestAssetName; - asset.WamsPrivateLocatorId = wamsLocators.SasLocator.Id; - asset.WamsPrivateLocatorUrl = wamsLocators.SasLocator.Url; - asset.WamsAssetId = outputAsset.Id; - asset.WamsEncoderMetadataXml = encoderMetadataXml; - asset.UploadState.Status = AssetUploadStatus.Uploaded; - asset.PublishState.Status = AssetPublishStatus.None; - }); - } - - try { - if (cloudVideoPart.IsPublished()) - _assetManager.PublishAssetsFor(cloudVideoPart); - } - catch (Exception ex) { - Logger.Warning(ex, "Processing of job '{0}' was completed but an error occurred while publishing the cloud video item with ID {1} after processing.", wamsJob.Name, cloudVideoPart.Id); - } + try { + if (cloudVideoPart.IsPublished()) + _assetManager.PublishAssetsFor(cloudVideoPart); + } + catch (Exception ex) { + Logger.Warning(ex, "Processing of job '{0}' was completed but an error occurred while publishing the cloud video item with ID {1} after processing.", wamsJob.Name, cloudVideoPart.Id); } } - - Logger.Information("Processing of job '{0}' was successfully completed.", wamsJob.Name); } + + Logger.Information("Processing of job '{0}' was successfully completed.", wamsJob.Name); } } } - catch (Exception ex) { - Logger.Error(ex, "Error during sweep."); - } - finally { - Monitor.Exit(_sweepLock); - Logger.Debug("Ending sweep."); - } + else + Logger.Debug("Distributed lock could not be acquired; going back to sleep."); + } + catch (Exception ex) { + Logger.Error(ex, "Error during sweep."); + } + finally { + Logger.Debug("Ending sweep."); } } diff --git a/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Web.config b/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Web.config index f585f8311..c3a205ab2 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Web.config +++ b/src/Orchard.Web/Modules/Orchard.Azure.MediaServices/Web.config @@ -45,7 +45,7 @@ - + @@ -57,15 +57,15 @@ - + - + - + diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Module.txt b/src/Orchard.Web/Modules/Orchard.Azure/Module.txt index 9c8eb1aac..c8016dd21 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Azure/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides a set of Orchard service implementations targeting Microsoft Azure services. Category: Hosting diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Orchard.Azure.csproj b/src/Orchard.Web/Modules/Orchard.Azure/Orchard.Azure.csproj index 8587c2796..12d87b822 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure/Orchard.Azure.csproj +++ b/src/Orchard.Web/Modules/Orchard.Azure/Orchard.Azure.csproj @@ -107,7 +107,7 @@ - + diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Azure/Properties/AssemblyInfo.cs index 5edcbfb73..39c27bbae 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Azure/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/AzureApplicationEnvironment.cs b/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/AzureApplicationEnvironment.cs new file mode 100644 index 000000000..0b6c5bc0d --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/AzureApplicationEnvironment.cs @@ -0,0 +1,10 @@ +using Microsoft.WindowsAzure.ServiceRuntime; +using Orchard.Environment; + +namespace Orchard.Azure.Services.Environment { + public class AzureApplicationEnvironment : IApplicationEnvironment { + public string GetEnvironmentIdentifier() { + return RoleEnvironment.CurrentRoleInstance.Id; + } + } +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/AzureBlobShellSettingsManager.cs b/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/AzureBlobShellSettingsManager.cs index 2e1ad499f..6e11a98d9 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/AzureBlobShellSettingsManager.cs +++ b/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/AzureBlobShellSettingsManager.cs @@ -2,11 +2,11 @@ using System.Collections.Generic; using System.IO; using System.Linq; -using Microsoft.Azure; using Orchard.Azure.Services.FileSystems; using Orchard.Environment.Configuration; using Orchard.FileSystems.Media; using Orchard.Logging; +using Microsoft.Azure; namespace Orchard.Azure.Services.Environment.Configuration { diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/DefaultPlatformConfigurationAccessor.cs b/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/DefaultPlatformConfigurationAccessor.cs index ba596ce66..71639b174 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/DefaultPlatformConfigurationAccessor.cs +++ b/src/Orchard.Web/Modules/Orchard.Azure/Services/Environment/Configuration/DefaultPlatformConfigurationAccessor.cs @@ -1,6 +1,6 @@ -using Microsoft.Azure; -using System.Configuration; +using System.Configuration; using System; +using Microsoft.Azure; namespace Orchard.Azure.Services.Environment.Configuration { diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Services/TaskLease/AzureMachineNameProvider.cs b/src/Orchard.Web/Modules/Orchard.Azure/Services/TaskLease/AzureMachineNameProvider.cs deleted file mode 100644 index 52a6dd2ab..000000000 --- a/src/Orchard.Web/Modules/Orchard.Azure/Services/TaskLease/AzureMachineNameProvider.cs +++ /dev/null @@ -1,13 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using Microsoft.WindowsAzure.ServiceRuntime; -using Orchard.Environment; - -namespace Orchard.Azure.Services.TaskLease { - public class AzureMachineNameProvider : IMachineNameProvider { - public string GetMachineName() { - return RoleEnvironment.CurrentRoleInstance.Id; - } - } -} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Azure/Web.config b/src/Orchard.Web/Modules/Orchard.Azure/Web.config index 15d8508e3..254f46733 100644 --- a/src/Orchard.Web/Modules/Orchard.Azure/Web.config +++ b/src/Orchard.Web/Modules/Orchard.Azure/Web.config @@ -41,7 +41,7 @@ - + @@ -53,15 +53,15 @@ - + - + - + diff --git a/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogArchivesPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogArchivesPartDriver.cs index 640c11181..f2783f000 100644 --- a/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogArchivesPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogArchivesPartDriver.cs @@ -53,10 +53,14 @@ namespace Orchard.Blogs.Drivers { } protected override void Importing(BlogArchivesPart part, ImportContentContext context) { - var blog = context.Attribute(part.PartDefinition.Name, "Blog"); - if (blog != null) { - part.BlogId = context.GetItemFromSession(blog).Id; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } + + context.ImportAttribute(part.PartDefinition.Name, "Blog", blog => + part.BlogId = context.GetItemFromSession(blog).Id + ); } protected override void Exporting(BlogArchivesPart part, ExportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogPartDriver.cs index db0030450..2fdc29313 100644 --- a/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/BlogPartDriver.cs @@ -43,20 +43,22 @@ namespace Orchard.Blogs.Drivers { } protected override void Importing(BlogPart part, ContentManagement.Handlers.ImportContentContext context) { - var description = context.Attribute(part.PartDefinition.Name, "Description"); - if (description != null) { - part.Description = description; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - var postCount = context.Attribute(part.PartDefinition.Name, "PostCount"); - if (postCount != null) { - part.PostCount = Convert.ToInt32(postCount); - } + context.ImportAttribute(part.PartDefinition.Name, "Description", description => + part.Description = description + ); - var feedProxyUrl = context.Attribute(part.PartDefinition.Name, "FeedProxyUrl"); - if (feedProxyUrl != null) { - part.FeedProxyUrl = feedProxyUrl; - } + context.ImportAttribute(part.PartDefinition.Name, "PostCount", postCount => + part.PostCount = Convert.ToInt32(postCount) + ); + + context.ImportAttribute(part.PartDefinition.Name, "FeedProxyUrl", feedProxyUrl => + part.FeedProxyUrl = feedProxyUrl + ); } protected override void Exporting(BlogPart part, ContentManagement.Handlers.ExportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/RecentBlogPostsPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/RecentBlogPostsPartDriver.cs index 8ab1b9268..e44f6fce6 100644 --- a/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/RecentBlogPostsPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Blogs/Drivers/RecentBlogPostsPartDriver.cs @@ -66,22 +66,25 @@ namespace Orchard.Blogs.Drivers { } protected override void Importing(RecentBlogPostsPart part, ImportContentContext context) { - var blog = context.Attribute(part.PartDefinition.Name, "Blog"); - if (blog != null) { - part.BlogId = context.GetItemFromSession(blog).Id; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - var count = context.Attribute(part.PartDefinition.Name, "Count"); - if (count != null) { - part.Count = Convert.ToInt32(count); - } + context.ImportAttribute(part.PartDefinition.Name, "Blog", blog => + part.BlogId = context.GetItemFromSession(blog).Id + ); + + context.ImportAttribute(part.PartDefinition.Name, "Count", count => + part.Count = Convert.ToInt32(count) + ); } protected override void Exporting(RecentBlogPostsPart part, ExportContentContext context) { var blog = _contentManager.Get(part.BlogId); var blogIdentity = _contentManager.GetItemMetadata(blog).Identity; - context.Element(part.PartDefinition.Name).SetAttributeValue("Blog", blogIdentity); + context.Element(part.PartDefinition.Name).SetAttributeValue("Blog", blogIdentity); context.Element(part.PartDefinition.Name).SetAttributeValue("Count", part.Count); } } diff --git a/src/Orchard.Web/Modules/Orchard.Blogs/Module.txt b/src/Orchard.Web/Modules/Orchard.Blogs/Module.txt index 37df5063f..c5d8c1e48 100644 --- a/src/Orchard.Web/Modules/Orchard.Blogs/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Blogs/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The Orchard Blogs module is implementing basic blogging features. FeatureDescription: A simple web log. diff --git a/src/Orchard.Web/Modules/Orchard.Blogs/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Blogs/Properties/AssemblyInfo.cs index 51684d725..669976e1b 100644 --- a/src/Orchard.Web/Modules/Orchard.Blogs/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Blogs/Properties/AssemblyInfo.cs @@ -29,6 +29,6 @@ using System.Runtime.InteropServices; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Caching/Module.txt b/src/Orchard.Web/Modules/Orchard.Caching/Module.txt index d047dcf80..dd1e26e2b 100644 --- a/src/Orchard.Web/Modules/Orchard.Caching/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Caching/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: Sébastien Ros Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides an API to cache business data. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Caching/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Caching/Properties/AssemblyInfo.cs index be2a0894d..94e0fe6a8 100644 --- a/src/Orchard.Web/Modules/Orchard.Caching/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Caching/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.CodeGeneration/Module.txt b/src/Orchard.Web/Modules/Orchard.CodeGeneration/Module.txt index 53f661672..b268ada56 100644 --- a/src/Orchard.Web/Modules/Orchard.CodeGeneration/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.CodeGeneration/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Tools to create Orchard components. FeatureDescription: Tools to create Orchard components. diff --git a/src/Orchard.Web/Modules/Orchard.CodeGeneration/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.CodeGeneration/Properties/AssemblyInfo.cs index a21dc27e2..285c3afa6 100644 --- a/src/Orchard.Web/Modules/Orchard.CodeGeneration/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.CodeGeneration/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentPartDriver.cs index a2488a61f..b27c3ced1 100644 --- a/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentPartDriver.cs @@ -117,7 +117,7 @@ namespace Orchard.Comments.Drivers { // prevent users from commenting on a closed thread by hijacking the commentedOn property var commentsPart = commentedOn.As(); - if (!commentsPart.CommentsActive) { + if (commentsPart == null || !commentsPart.CommentsActive) { _orchardServices.TransactionManager.Cancel(); return Editor(part, shapeHelper); } @@ -132,71 +132,65 @@ namespace Orchard.Comments.Drivers { } protected override void Importing(CommentPart part, ContentManagement.Handlers.ImportContentContext context) { - var author = context.Attribute(part.PartDefinition.Name, "Author"); - if (author != null) { - part.Record.Author = author; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - var siteName = context.Attribute(part.PartDefinition.Name, "SiteName"); - if (siteName != null) { - part.Record.SiteName = siteName; - } + context.ImportAttribute(part.PartDefinition.Name, "Author", author => + part.Record.Author = author + ); - var userName = context.Attribute(part.PartDefinition.Name, "UserName"); - if (userName != null) { - part.Record.UserName = userName; - } + context.ImportAttribute(part.PartDefinition.Name, "SiteName", siteName => + part.Record.SiteName = siteName + ); - var email = context.Attribute(part.PartDefinition.Name, "Email"); - if (email != null) { - part.Record.Email = email; - } + context.ImportAttribute(part.PartDefinition.Name, "UserName", userName => + part.Record.UserName = userName + ); - var position = context.Attribute(part.PartDefinition.Name, "Position"); - if (position != null) { - part.Record.Position = decimal.Parse(position, CultureInfo.InvariantCulture); - } + context.ImportAttribute(part.PartDefinition.Name, "Email", email => + part.Record.Email = email + ); - var status = context.Attribute(part.PartDefinition.Name, "Status"); - if (status != null) { - part.Record.Status = (CommentStatus)Enum.Parse(typeof(CommentStatus), status); - } + context.ImportAttribute(part.PartDefinition.Name, "Position", position => + part.Record.Position = decimal.Parse(position, CultureInfo.InvariantCulture) + ); - var commentDate = context.Attribute(part.PartDefinition.Name, "CommentDateUtc"); - if (commentDate != null) { - part.Record.CommentDateUtc = XmlConvert.ToDateTime(commentDate, XmlDateTimeSerializationMode.Utc); - } + context.ImportAttribute(part.PartDefinition.Name, "Status", status => + part.Record.Status = (CommentStatus)Enum.Parse(typeof(CommentStatus), status) + ); - var text = context.Attribute(part.PartDefinition.Name, "CommentText"); - if (text != null) { - part.Record.CommentText = text; - } + context.ImportAttribute(part.PartDefinition.Name, "CommentDateUtc", commentDate => + part.Record.CommentDateUtc = XmlConvert.ToDateTime(commentDate, XmlDateTimeSerializationMode.Utc) + ); - var commentedOn = context.Attribute(part.PartDefinition.Name, "CommentedOn"); - if (commentedOn != null) { + context.ImportAttribute(part.PartDefinition.Name, "CommentText", text => + part.Record.CommentText = text + ); + + context.ImportAttribute(part.PartDefinition.Name, "CommentedOn", commentedOn => { var contentItem = context.GetItemFromSession(commentedOn); if (contentItem != null) { part.Record.CommentedOn = contentItem.Id; } contentItem.As().Record.CommentPartRecords.Add(part.Record); - } + }); - var repliedOn = context.Attribute(part.PartDefinition.Name, "RepliedOn"); - if (repliedOn != null) { + context.ImportAttribute(part.PartDefinition.Name, "RepliedOn", repliedOn => { var contentItem = context.GetItemFromSession(repliedOn); if (contentItem != null) { part.Record.RepliedOn = contentItem.Id; } - } + }); - var commentedOnContainer = context.Attribute(part.PartDefinition.Name, "CommentedOnContainer"); - if (commentedOnContainer != null) { + context.ImportAttribute(part.PartDefinition.Name, "CommentedOnContainer", commentedOnContainer => { var container = context.GetItemFromSession(commentedOnContainer); if (container != null) { part.Record.CommentedOnContainer = container.Id; } - } + }); } protected override void Exporting(CommentPart part, ContentManagement.Handlers.ExportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentsPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentsPartDriver.cs index 97966f3f9..d7175898a 100644 --- a/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentsPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Comments/Drivers/CommentsPartDriver.cs @@ -111,20 +111,22 @@ namespace Orchard.Comments.Drivers { } protected override void Importing(CommentsPart part, ContentManagement.Handlers.ImportContentContext context) { - var commentsShown = context.Attribute(part.PartDefinition.Name, "CommentsShown"); - if (commentsShown != null) { - part.CommentsShown = Convert.ToBoolean(commentsShown); + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - var commentsActive = context.Attribute(part.PartDefinition.Name, "CommentsActive"); - if (commentsActive != null) { - part.CommentsActive = Convert.ToBoolean(commentsActive); - } + context.ImportAttribute(part.PartDefinition.Name, "CommentsShown", commentsShown => + part.CommentsShown = Convert.ToBoolean(commentsShown) + ); - var threadedComments = context.Attribute(part.PartDefinition.Name, "ThreadedComments"); - if (threadedComments != null) { - part.ThreadedComments = Convert.ToBoolean(threadedComments); - } + context.ImportAttribute(part.PartDefinition.Name, "CommentsActive", commentsActive => + part.CommentsActive = Convert.ToBoolean(commentsActive) + ); + + context.ImportAttribute(part.PartDefinition.Name, "ThreadedComments", threadedComments => + part.ThreadedComments = Convert.ToBoolean(threadedComments) + ); } protected override void Exporting(CommentsPart part, ContentManagement.Handlers.ExportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Comments/Module.txt b/src/Orchard.Web/Modules/Orchard.Comments/Module.txt index db08b140b..d474d622a 100644 --- a/src/Orchard.Web/Modules/Orchard.Comments/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Comments/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The comments system implemented by this module can be applied to arbitrary Orchard content types, such as blogs and pages. It includes comment validation and spam protection through the Akismet service. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Comments/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Comments/Properties/AssemblyInfo.cs index decf91004..d7cd0582f 100644 --- a/src/Orchard.Web/Modules/Orchard.Comments/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Comments/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Comments/Services/CommentService.cs b/src/Orchard.Web/Modules/Orchard.Comments/Services/CommentService.cs index 816f2720a..18d288fa1 100644 --- a/src/Orchard.Web/Modules/Orchard.Comments/Services/CommentService.cs +++ b/src/Orchard.Web/Modules/Orchard.Comments/Services/CommentService.cs @@ -118,7 +118,8 @@ namespace Orchard.Comments.Services { } public void DeleteComment(int commentId) { - _orchardServices.ContentManager.Remove(_orchardServices.ContentManager.Get(commentId).ContentItem); + // Get latest because the comment may be unpublished if the anti-spam module has caught it + _orchardServices.ContentManager.Remove(_orchardServices.ContentManager.Get(commentId, VersionOptions.Latest).ContentItem); } public bool CommentsDisabledForCommentedContent(int id) { diff --git a/src/Orchard.Web/Modules/Orchard.ContentPermissions/Drivers/ContentPermissionsPartDriver.cs b/src/Orchard.Web/Modules/Orchard.ContentPermissions/Drivers/ContentPermissionsPartDriver.cs index 444d72ac3..22e6f9998 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentPermissions/Drivers/ContentPermissionsPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.ContentPermissions/Drivers/ContentPermissionsPartDriver.cs @@ -162,6 +162,11 @@ namespace Orchard.ContentPermissions.Drivers { } protected override void Importing(ContentPermissionsPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + context.ImportAttribute(part.PartDefinition.Name, "Enabled", s => part.Enabled = XmlConvert.ToBoolean(s)); context.ImportAttribute(part.PartDefinition.Name, "ViewContent", s => part.ViewContent = s); context.ImportAttribute(part.PartDefinition.Name, "EditContent", s => part.EditContent = s); diff --git a/src/Orchard.Web/Modules/Orchard.ContentPermissions/Module.txt b/src/Orchard.Web/Modules/Orchard.ContentPermissions/Module.txt index 7d895333c..ae22aa617 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentPermissions/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.ContentPermissions/Module.txt @@ -2,7 +2,7 @@ Name: Orchard.ContentPermissions AntiForgery: enabled Author: Chris Pyle, Sbastien Ros Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Allows item-level front end view permissions. Features: diff --git a/src/Orchard.Web/Modules/Orchard.ContentPermissions/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.ContentPermissions/Properties/AssemblyInfo.cs index 41952f985..9d53413a9 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentPermissions/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.ContentPermissions/Properties/AssemblyInfo.cs @@ -31,5 +31,5 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.ContentPicker/Drivers/ContentMenuItemPartDriver.cs b/src/Orchard.Web/Modules/Orchard.ContentPicker/Drivers/ContentMenuItemPartDriver.cs index 4a6808484..ee75f144a 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentPicker/Drivers/ContentMenuItemPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.ContentPicker/Drivers/ContentMenuItemPartDriver.cs @@ -60,14 +60,18 @@ namespace Orchard.ContentPicker.Drivers { } protected override void Importing(ContentMenuItemPart part, ImportContentContext context) { - var contentItemId = context.Attribute(part.PartDefinition.Name, "ContentItem"); - if (contentItemId != null) { - var contentItem = context.GetItemFromSession(contentItemId); - part.Content = contentItem; - } - else { - part.Content = null; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } + + context.ImportAttribute(part.PartDefinition.Name, "ContentItem", + contentItemId => { + var contentItem = context.GetItemFromSession(contentItemId); + part.Content = contentItem; + }, () => + part.Content = null + ); } protected override void Exporting(ContentMenuItemPart part, ExportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.ContentPicker/Fields/ContentPickerField.cs b/src/Orchard.Web/Modules/Orchard.ContentPicker/Fields/ContentPickerField.cs index b08c95247..8b7c15194 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentPicker/Fields/ContentPickerField.cs +++ b/src/Orchard.Web/Modules/Orchard.ContentPicker/Fields/ContentPickerField.cs @@ -17,7 +17,7 @@ namespace Orchard.ContentPicker.Fields { public IEnumerable ContentItems { get { - return _contentItems.Value; + return _contentItems.Value ?? Enumerable.Empty(); } } diff --git a/src/Orchard.Web/Modules/Orchard.ContentPicker/Module.txt b/src/Orchard.Web/Modules/Orchard.ContentPicker/Module.txt index 1feba3586..42a678254 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentPicker/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.ContentPicker/Module.txt @@ -2,7 +2,7 @@ Name: Orchard.ContentPicker AntiForgery: enabled Author: The Orchard Team Website: http://orchardcontentpicker.codeplex.com -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: UI for selecting Content Items. Features: diff --git a/src/Orchard.Web/Modules/Orchard.ContentPicker/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.ContentPicker/Properties/AssemblyInfo.cs index ca5eacf22..c9a2beb06 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentPicker/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.ContentPicker/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.ContentTypes/Module.txt b/src/Orchard.Web/Modules/Orchard.ContentTypes/Module.txt index c1db7247b..c7fb423a5 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentTypes/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.ContentTypes/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: ContentTypes modules enables the creation and alteration of content types not based on code. Dependencies: Contents diff --git a/src/Orchard.Web/Modules/Orchard.ContentTypes/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.ContentTypes/Properties/AssemblyInfo.cs index 904c28a48..06128c3bf 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentTypes/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.ContentTypes/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.ContentTypes/Services/StereotypeService.cs b/src/Orchard.Web/Modules/Orchard.ContentTypes/Services/StereotypeService.cs index c00221328..4b4e8471a 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentTypes/Services/StereotypeService.cs +++ b/src/Orchard.Web/Modules/Orchard.ContentTypes/Services/StereotypeService.cs @@ -22,7 +22,7 @@ namespace Orchard.ContentTypes.Services { } public IEnumerable GetStereotypes() { - return _cacheManager.Get("ContentType.Stereotypes", context => { + return _cacheManager.Get("ContentType.Stereotypes", true, context => { // TODO: Implement a signal in ContentDefinitionManager that gets raised whenever a type definition is updated. // For now, we'll just cache the stereotypes for 1 minute. diff --git a/src/Orchard.Web/Modules/Orchard.ContentTypes/Views/Admin/EditPlacement.cshtml b/src/Orchard.Web/Modules/Orchard.ContentTypes/Views/Admin/EditPlacement.cshtml index 4cd178cc0..e9b798834 100644 --- a/src/Orchard.Web/Modules/Orchard.ContentTypes/Views/Admin/EditPlacement.cshtml +++ b/src/Orchard.Web/Modules/Orchard.ContentTypes/Views/Admin/EditPlacement.cshtml @@ -43,7 +43,7 @@
- +
} diff --git a/src/Orchard.Web/Modules/Orchard.CustomForms/Drivers/CustomFormPartDriver.cs b/src/Orchard.Web/Modules/Orchard.CustomForms/Drivers/CustomFormPartDriver.cs index ee942bd46..109675fb7 100644 --- a/src/Orchard.Web/Modules/Orchard.CustomForms/Drivers/CustomFormPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.CustomForms/Drivers/CustomFormPartDriver.cs @@ -59,21 +59,20 @@ namespace Orchard.CustomForms.Drivers { } protected override void Importing(CustomFormPart part, ImportContentContext context) { - IfNotNull(context.Attribute(part.PartDefinition.Name, "ContentType"), x => part.Record.ContentType = x); - IfNotNull(context.Attribute(part.PartDefinition.Name, "SaveContentItem"), x => part.Record.SaveContentItem = Boolean.Parse(x)); - IfNotNull(context.Attribute(part.PartDefinition.Name, "CustomMessage"), x => part.Record.CustomMessage = Boolean.Parse(x)); - IfNotNull(context.Attribute(part.PartDefinition.Name, "Message"), x => part.Record.Message = x); - IfNotNull(context.Attribute(part.PartDefinition.Name, "Redirect"), x => part.Record.Redirect = Boolean.Parse(x)); - IfNotNull(context.Attribute(part.PartDefinition.Name, "RedirectUrl"), x => part.Record.RedirectUrl = x); - IfNotNull(context.Attribute(part.PartDefinition.Name, "SubmitButtonText"), x => part.Record.SubmitButtonText = x); - } - - private static void IfNotNull(T value, Action then) { - if (value != null) { - then(value); + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - } + context.ImportAttribute(part.PartDefinition.Name, "ContentType", x => part.Record.ContentType = x); + context.ImportAttribute(part.PartDefinition.Name, "SaveContentItem", x => part.Record.SaveContentItem = Boolean.Parse(x)); + context.ImportAttribute(part.PartDefinition.Name, "CustomMessage", x => part.Record.CustomMessage = Boolean.Parse(x)); + context.ImportAttribute(part.PartDefinition.Name, "Message", x => part.Record.Message = x); + context.ImportAttribute(part.PartDefinition.Name, "Redirect", x => part.Record.Redirect = Boolean.Parse(x)); + context.ImportAttribute(part.PartDefinition.Name, "RedirectUrl", x => part.Record.RedirectUrl = x); + context.ImportAttribute(part.PartDefinition.Name, "SubmitButtonText", x => part.Record.SubmitButtonText = x); + } + protected override void Exporting(CustomFormPart part, ExportContentContext context) { context.Element(part.PartDefinition.Name).SetAttributeValue("ContentType", part.Record.ContentType); context.Element(part.PartDefinition.Name).SetAttributeValue("SaveContentItem", part.Record.SaveContentItem); diff --git a/src/Orchard.Web/Modules/Orchard.CustomForms/Module.txt b/src/Orchard.Web/Modules/Orchard.CustomForms/Module.txt index 3fff3687d..af58241ea 100644 --- a/src/Orchard.Web/Modules/Orchard.CustomForms/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.CustomForms/Module.txt @@ -2,7 +2,7 @@ Name: Custom Forms AntiForgery: enabled Author: The Orchard Team Website: http://orchardcustomforms.codeplex.com -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 LifecycleStatus: Deprecated Description: Create custom forms like contact forms or content contributions. diff --git a/src/Orchard.Web/Modules/Orchard.CustomForms/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.CustomForms/Properties/AssemblyInfo.cs index 32c55e753..5095e0451 100644 --- a/src/Orchard.Web/Modules/Orchard.CustomForms/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.CustomForms/Properties/AssemblyInfo.cs @@ -31,5 +31,5 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Dashboards/Orchard.Dashboards.csproj b/src/Orchard.Web/Modules/Orchard.Dashboards/Orchard.Dashboards.csproj index ec35906e7..2914e7645 100644 --- a/src/Orchard.Web/Modules/Orchard.Dashboards/Orchard.Dashboards.csproj +++ b/src/Orchard.Web/Modules/Orchard.Dashboards/Orchard.Dashboards.csproj @@ -72,7 +72,7 @@ - + diff --git a/src/Orchard.Web/Modules/Orchard.Dashboards/Services/DefaultDashboardSelector.cs b/src/Orchard.Web/Modules/Orchard.Dashboards/Services/DefaultDashboardSelector.cs index 55b7ca415..f303de0ed 100644 --- a/src/Orchard.Web/Modules/Orchard.Dashboards/Services/DefaultDashboardSelector.cs +++ b/src/Orchard.Web/Modules/Orchard.Dashboards/Services/DefaultDashboardSelector.cs @@ -6,7 +6,7 @@ using Orchard.Layouts.Models; namespace Orchard.Dashboards.Services { public class DefaultDashboardSelector : IDashboardSelector { private readonly IOrchardServices _services; - public const string DefaultLayout = "{\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Canvas\",\"data\":\"\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Grid\",\"data\":\"\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Row\",\"data\":\"\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-cog%22%3e%3c%2fi%3e+Get+up+and+running%3c%2fh2%3e%0d%0a%3cp%3eStart+by+exploring+the+menu+on+the+left+and+familiarize+yourself+with+Orchard.+As+for+the+basics%2c+we+suggest+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dOrchard.Themes%7d%22%3echanging+the+theme%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cAdmin%2carea%3dContents%2cid%3dPage%7d%22%3eadding+some+pages%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cBlogAdmin%2carea%3dOrchard.Blogs%7d%22%3esetup+up+a+blog%3c%2fa%3e%2c+and+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dSettings%2cgroupInfoId%3dIndex%7d%22%3econfiguring+basic+settings%3c%2fa%3e.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-cog%22%3e%3c%2fi%3e+Get+up+and+running%3c%2fh2%3e%0d%0a%3cp%3eStart+by+exploring+the+menu+on+the+left+and+familiarize+yourself+with+Orchard.+As+for+the+basics%2c+we+suggest+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dOrchard.Themes%7d%22%3echanging+the+theme%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cAdmin%2carea%3dContents%2cid%3dPage%7d%22%3eadding+some+pages%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cBlogAdmin%2carea%3dOrchard.Blogs%7d%22%3esetup+up+a+blog%3c%2fa%3e%2c+and+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dSettings%2cgroupInfoId%3dIndex%7d%22%3econfiguring+basic+settings%3c%2fa%3e.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":1,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-th-large%22%3e%3c%2fi%3eGet+more+goodies%3c%2fh2%3e%0d%0a%3cp%3eChange+the+way+your+site+works+and+looks+with+%3ca+href%3d%22%23%7bUrl.Action%3aThemes%2cGallery%2carea%3dOrchard.Packaging%7d%22%3ethemes%3c%2fa%3e+and+%3ca+href%3d%22%23%7bUrl.Action%3aModules%2cGallery%2carea%3dOrchard.Packaging%7d%22%3emodules%3c%2fa%3e.+There%26rsquo%3bs+plenty+to+choose+from+in+the+%3ca+href%3d%22http%3a%2f%2fgallery.orchardproject.net%22%3eOrchard+Gallery%3c%2fa%3e.+We%26rsquo%3bre+always+adding+things%2c+so+be+sure+to+check+back+often+to+see+what%26rsquo%3bs+new.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-th-large%22%3e%3c%2fi%3eGet+more+goodies%3c%2fh2%3e%0d%0a%3cp%3eChange+the+way+your+site+works+and+looks+with+%3ca+href%3d%22%23%7bUrl.Action%3aThemes%2cGallery%2carea%3dOrchard.Packaging%7d%22%3ethemes%3c%2fa%3e+and+%3ca+href%3d%22%23%7bUrl.Action%3aModules%2cGallery%2carea%3dOrchard.Packaging%7d%22%3emodules%3c%2fa%3e.+There%26rsquo%3bs+plenty+to+choose+from+in+the+%3ca+href%3d%22http%3a%2f%2fgallery.orchardproject.net%22%3eOrchard+Gallery%3c%2fa%3e.+We%26rsquo%3bre+always+adding+things%2c+so+be+sure+to+check+back+often+to+see+what%26rsquo%3bs+new.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":2,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-book%22%3e%3c%2fi%3eRead+the+Docs%3c%2fh2%3e%0d%0a%3cp%3eAre+you+ready+to+go+deeper+and+become+an+Orchard+expert%3f+Take+a+look+at+the+%3ca+href%3d%22http%3a%2f%2fdocs.orchardproject.net%22%3eOrchard+Documentation%3c%2fa%3e+to+learn+about+how+everything+connects+together+and+what+makes+Orchard+tick.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-book%22%3e%3c%2fi%3eRead+the+Docs%3c%2fh2%3e%0d%0a%3cp%3eAre+you+ready+to+go+deeper+and+become+an+Orchard+expert%3f+Take+a+look+at+the+%3ca+href%3d%22http%3a%2f%2fdocs.orchardproject.net%22%3eOrchard+Documentation%3c%2fa%3e+to+learn+about+how+everything+connects+together+and+what+makes+Orchard+tick.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":3,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-users%22%3e%3c%2fi%3eMake+friends%3c%2fh2%3e%0d%0a%3cp%3eFind+friends+that+share+your+interest+of+Orchard.+There+are+a+couple+ways+that+you+can%26nbsp%3b%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fdiscussions%22%3ediscuss+and+get+connected%3c%2fa%3e+to+the+project+including+mailing+lists%2c+forums+and+IRC.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-users%22%3e%3c%2fi%3eMake+friends%3c%2fh2%3e%0d%0a%3cp%3eFind+friends+that+share+your+interest+of+Orchard.+There+are+a+couple+ways+that+you+can%26nbsp%3b%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fdiscussions%22%3ediscuss+and+get+connected%3c%2fa%3e+to+the+project+including+mailing+lists%2c+forums+and+IRC.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":4,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-pencil%22%3e%3c%2fi%3eContribute+back%3c%2fh2%3e%0d%0a%3cp%3eHelp+grow+Orchard.+We+encourage+contributions+of+all+sorts%2c+including+code+submissions%2c+documentation%2c+translations%2c+feature+recommendations%2c+and+more.Here+are+some+ways+to+%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fcontribution%22%3egive+back+to+the+project%3c%2fa%3e.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-pencil%22%3e%3c%2fi%3eContribute+back%3c%2fh2%3e%0d%0a%3cp%3eHelp+grow+Orchard.+We+encourage+contributions+of+all+sorts%2c+including+code+submissions%2c+documentation%2c+translations%2c+feature+recommendations%2c+and+more.Here+are+some+ways+to+%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fcontribution%22%3egive+back+to+the+project%3c%2fa%3e.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":5,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Row\",\"data\":\"\",\"exportableData\":\"\",\"index\":1,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=10&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-clock-o%22%3e%3c%2fi%3eStay+up+to+date%3c%2fh2%3e%0d%0a%3cp%3e%3ciframe+width%3d%22100%25%22+height%3d%22100%25%22+scrolling%3d%22no%22+frameborder%3d%220%22+src%3d%22http%3a%2f%2fwww.orchardproject.net%2fadvisory%22+id%3d%22advisory%22%3e%0d%0a+%26lt%3bp%26gt%3bYour+browser+does+not+support+iframes.+You+can%27t+see+advisory+messages.%26lt%3b%2fp%26gt%3b%3c%2fiframe%3e%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-clock-o%22%3e%3c%2fi%3eStay+up+to+date%3c%2fh2%3e%0d%0a%3cp%3e%3ciframe+width%3d%22100%25%22+height%3d%22100%25%22+scrolling%3d%22no%22+frameborder%3d%220%22+src%3d%22http%3a%2f%2fwww.orchardproject.net%2fadvisory%22+id%3d%22advisory%22%3e%0d%0a+%26lt%3bp%26gt%3bYour+browser+does+not+support+iframes.+You+can%27t+see+advisory+messages.%26lt%3b%2fp%26gt%3b%3c%2fiframe%3e%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":1,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}]}"; + public const string DefaultLayout = "{\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Canvas\",\"data\":\"\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Grid\",\"data\":\"\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Row\",\"data\":\"\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-cog%22%3e%3c%2fi%3e+Get+up+and+running%3c%2fh2%3e%0d%0a%3cp%3eStart+by+exploring+the+menu+on+the+left+and+familiarize+yourself+with+Orchard.+As+for+the+basics%2c+we+suggest+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dOrchard.Themes%7d%22%3echanging+the+theme%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cAdmin%2carea%3dContents%2cid%3dPage%7d%22%3eadding+some+pages%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cBlogAdmin%2carea%3dOrchard.Blogs%7d%22%3esetup+up+a+blog%3c%2fa%3e%2c+and+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dSettings%2cgroupInfoId%3dIndex%7d%22%3econfiguring+basic+settings%3c%2fa%3e.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-cog%22%3e%3c%2fi%3e+Get+up+and+running%3c%2fh2%3e%0d%0a%3cp%3eStart+by+exploring+the+menu+on+the+left+and+familiarize+yourself+with+Orchard.+As+for+the+basics%2c+we+suggest+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dOrchard.Themes%7d%22%3echanging+the+theme%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cAdmin%2carea%3dContents%2cid%3dPage%7d%22%3eadding+some+pages%3c%2fa%3e%2c+%3ca+href%3d%22%23%7bUrl.Action%3aCreate%2cBlogAdmin%2carea%3dOrchard.Blogs%7d%22%3esetup+up+a+blog%3c%2fa%3e%2c+and+%3ca+href%3d%22%23%7bUrl.Action%3aIndex%2cAdmin%2carea%3dSettings%2cgroupInfoId%3dIndex%7d%22%3econfiguring+basic+settings%3c%2fa%3e.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":1,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-th-large%22%3e%3c%2fi%3eGet+more+goodies%3c%2fh2%3e%0d%0a%3cp%3eChange+the+way+your+site+works+and+looks+with+%3ca+href%3d%22%23%7bUrl.Action%3aThemes%2cGallery%2carea%3dOrchard.Packaging%7d%22%3ethemes%3c%2fa%3e+and+%3ca+href%3d%22%23%7bUrl.Action%3aModules%2cGallery%2carea%3dOrchard.Packaging%7d%22%3emodules%3c%2fa%3e.+There%26rsquo%3bs+plenty+to+choose+from+in+the+%3ca+href%3d%22http%3a%2f%2fgallery.orchardproject.net%22%3eOrchard+Gallery%3c%2fa%3e.+We%26rsquo%3bre+always+adding+things%2c+so+be+sure+to+check+back+often+to+see+what%26rsquo%3bs+new.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-th-large%22%3e%3c%2fi%3eGet+more+goodies%3c%2fh2%3e%0d%0a%3cp%3eChange+the+way+your+site+works+and+looks+with+%3ca+href%3d%22%23%7bUrl.Action%3aThemes%2cGallery%2carea%3dOrchard.Packaging%7d%22%3ethemes%3c%2fa%3e+and+%3ca+href%3d%22%23%7bUrl.Action%3aModules%2cGallery%2carea%3dOrchard.Packaging%7d%22%3emodules%3c%2fa%3e.+There%26rsquo%3bs+plenty+to+choose+from+in+the+%3ca+href%3d%22http%3a%2f%2fgallery.orchardproject.net%22%3eOrchard+Gallery%3c%2fa%3e.+We%26rsquo%3bre+always+adding+things%2c+so+be+sure+to+check+back+often+to+see+what%26rsquo%3bs+new.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":2,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-book%22%3e%3c%2fi%3eRead+the+Docs%3c%2fh2%3e%0d%0a%3cp%3eAre+you+ready+to+go+deeper+and+become+an+Orchard+expert%3f+Take+a+look+at+the+%3ca+href%3d%22http%3a%2f%2fdocs.orchardproject.net%22%3eOrchard+Documentation%3c%2fa%3e+to+learn+about+how+everything+connects+together+and+what+makes+Orchard+tick.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-book%22%3e%3c%2fi%3eRead+the+Docs%3c%2fh2%3e%0d%0a%3cp%3eAre+you+ready+to+go+deeper+and+become+an+Orchard+expert%3f+Take+a+look+at+the+%3ca+href%3d%22http%3a%2f%2fdocs.orchardproject.net%22%3eOrchard+Documentation%3c%2fa%3e+to+learn+about+how+everything+connects+together+and+what+makes+Orchard+tick.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":3,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-users%22%3e%3c%2fi%3eMake+friends%3c%2fh2%3e%0d%0a%3cp%3eFind+friends+that+share+your+interest+of+Orchard.+There+are+a+couple+ways+that+you+can%26nbsp%3b%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fdiscussions%22%3ediscuss+and+get+connected%3c%2fa%3e+to+the+project+including+mailing+lists%2c+forums+and+IRC.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-users%22%3e%3c%2fi%3eMake+friends%3c%2fh2%3e%0d%0a%3cp%3eFind+friends+that+share+your+interest+of+Orchard.+There+are+a+couple+ways+that+you+can%26nbsp%3b%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fdiscussions%22%3ediscuss+and+get+connected%3c%2fa%3e+to+the+project+including+mailing+lists%2c+forums+and+IRC.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":4,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-pencil%22%3e%3c%2fi%3eContribute+back%3c%2fh2%3e%0d%0a%3cp%3eHelp+grow+Orchard.+We+encourage+contributions+of+all+sorts%2c+including+code+submissions%2c+documentation%2c+translations%2c+feature+recommendations%2c+and+more.Here+are+some+ways+to+%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fcontribution%22%3egive+back+to+the+project%3c%2fa%3e.%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-pencil%22%3e%3c%2fi%3eContribute+back%3c%2fh2%3e%0d%0a%3cp%3eHelp+grow+Orchard.+We+encourage+contributions+of+all+sorts%2c+including+code+submissions%2c+documentation%2c+translations%2c+feature+recommendations%2c+and+more.Here+are+some+ways+to+%3ca+href%3d%22http%3a%2f%2forchardproject.net%2fcontribution%22%3egive+back+to+the+project%3c%2fa%3e.%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":5,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Row\",\"data\":\"\",\"exportableData\":\"\",\"index\":1,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=10&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":0,\"elements\":[{\"typeName\":\"Orchard.Layouts.Elements.Html\",\"data\":\"TypeName=Orchard.Layouts.Elements.Html&Text=%3ch2%3e%3ci+class%3d%22fa+fa-clock-o%22%3e%3c%2fi%3eStay+up+to+date%3c%2fh2%3e%0d%0a%3cp%3e%3ciframe+width%3d%22100%25%22+height%3d%22100%25%22+scrolling%3d%22no%22+frameborder%3d%220%22+src%3d%22http%3a%2f%2fwww.orchardproject.net%2fadvisory%22+id%3d%22advisory%22%3e%0d%0a+%26lt%3bp%26gt%3bYour+browser+does+not+support+iframes.+You+can%27t+see+advisory+messages.%26lt%3b%2fp%26gt%3b%3c%2fiframe%3e%3c%2fp%3e&Content=%3ch2%3e%3ci+class%3d%22fa+fa-clock-o%22%3e%3c%2fi%3eStay+up+to+date%3c%2fh2%3e%0d%0a%3cp%3e%3ciframe+width%3d%22100%25%22+height%3d%22100%25%22+scrolling%3d%22no%22+frameborder%3d%220%22+src%3d%22http%3a%2f%2fwww.orchardproject.net%2fadvisory%22+id%3d%22advisory%22%3e%0d%0a+%26lt%3bp%26gt%3bYour+browser+does+not+support+iframes.+You+can%27t+see+advisory+messages.%26lt%3b%2fp%26gt%3b%3c%2fiframe%3e%3c%2fp%3e\",\"exportableData\":\"\",\"index\":0,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"},{\"typeName\":\"Orchard.Layouts.Elements.Column\",\"data\":\"Width=2&Offset=0&Collapsible=null\",\"exportableData\":\"\",\"index\":1,\"elements\":[],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}],\"isTemplated\":false,\"htmlId\":\"\",\"htmlClass\":\"\",\"htmlStyle\":\"\",\"rule\":\"\"}]}"; public DefaultDashboardSelector(IOrchardServices services) { _services = services; diff --git a/src/Orchard.Web/Modules/Orchard.DesignerTools/Module.txt b/src/Orchard.Web/Modules/Orchard.DesignerTools/Module.txt index 9cf6763e0..6cd2a40b2 100644 --- a/src/Orchard.Web/Modules/Orchard.DesignerTools/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.DesignerTools/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Contains designer tools to ease the Themes development process FeatureName: Shape Tracing diff --git a/src/Orchard.Web/Modules/Orchard.DesignerTools/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.DesignerTools/Properties/AssemblyInfo.cs index e7d9be4bd..c725f836f 100644 --- a/src/Orchard.Web/Modules/Orchard.DesignerTools/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.DesignerTools/Properties/AssemblyInfo.cs @@ -31,5 +31,5 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.DesignerTools/Services/ShapeTracingFactory.cs b/src/Orchard.Web/Modules/Orchard.DesignerTools/Services/ShapeTracingFactory.cs index cec0f009a..ae80ad879 100644 --- a/src/Orchard.Web/Modules/Orchard.DesignerTools/Services/ShapeTracingFactory.cs +++ b/src/Orchard.Web/Modules/Orchard.DesignerTools/Services/ShapeTracingFactory.cs @@ -2,6 +2,7 @@ using System.Linq; using System.Text; using System.Web.Routing; +using System.Xml.Linq; using Orchard.DisplayManagement.Descriptors; using Orchard.DisplayManagement.Implementation; using Orchard.DisplayManagement.Shapes; @@ -11,15 +12,16 @@ using Orchard.Security; using Orchard.Themes; using Orchard.UI; using Orchard.UI.Admin; +using System.Web; namespace Orchard.DesignerTools.Services { [OrchardFeature("Orchard.DesignerTools")] public class ShapeTracingFactory : IShapeFactoryEvents, IShapeDisplayEvents { + private readonly WorkContext _workContext; private readonly IShapeTableManager _shapeTableManager; private readonly IThemeManager _themeManager; private readonly IWebSiteFolder _webSiteFolder; private readonly IAuthorizer _authorizer; - private readonly IWorkContextAccessor _workContextAccessor; private bool _processing; private int _shapeId; @@ -31,7 +33,7 @@ namespace Orchard.DesignerTools.Services { IWebSiteFolder webSiteFolder, IAuthorizer authorizer ) { - _workContextAccessor = workContextAccessor; + _workContext = workContextAccessor.GetContext(); _shapeTableManager = shapeTableManager; _themeManager = themeManager; _webSiteFolder = webSiteFolder; @@ -39,17 +41,13 @@ namespace Orchard.DesignerTools.Services { } private bool IsActivable() { - var workContext = _workContextAccessor.GetContext(); - // activate on front-end only - if (AdminFilter.IsApplied(new RequestContext(workContext.HttpContext, new RouteData()))) + if (AdminFilter.IsApplied(new RequestContext(_workContext.HttpContext, new RouteData()))) return false; // if not logged as a site owner, still activate if it's a local request (development machine) - if (!_authorizer.Authorize(StandardPermissions.SiteOwner)) { - - return workContext.HttpContext.Request.IsLocal; - } + if (!_authorizer.Authorize(StandardPermissions.SiteOwner)) + return _workContext.HttpContext.Request.IsLocal; return true; } @@ -78,8 +76,7 @@ namespace Orchard.DesignerTools.Services { && context.ShapeType != "DateTimeRelative") { var shapeMetadata = (ShapeMetadata)context.Shape.Metadata; - var workContext = _workContextAccessor.GetContext(); - var currentTheme = workContext.CurrentTheme; + var currentTheme = _workContext.CurrentTheme; var shapeTable = _shapeTableManager.GetShapeTable(currentTheme.Id); if (!shapeTable.Descriptors.ContainsKey(shapeMetadata.Type)) { @@ -102,8 +99,7 @@ namespace Orchard.DesignerTools.Services { var shape = context.Shape; var shapeMetadata = (ShapeMetadata) context.Shape.Metadata; - var workContext = _workContextAccessor.GetContext(); - var currentTheme = _themeManager.GetRequestTheme(workContext.HttpContext.Request.RequestContext); + var currentTheme = _themeManager.GetRequestTheme(_workContext.HttpContext.Request.RequestContext); var shapeTable = _shapeTableManager.GetShapeTable(currentTheme.Id); if (!shapeMetadata.Wrappers.Contains("ShapeTracingWrapper")) { diff --git a/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/QueryElementDriver.cs b/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/QueryElementDriver.cs index f5a492657..0b284e293 100644 --- a/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/QueryElementDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/QueryElementDriver.cs @@ -136,7 +136,7 @@ namespace Orchard.DynamicForms.Drivers { var runtimeValues = GetRuntimeValues(element); if (!String.IsNullOrWhiteSpace(optionLabel)) { - yield return new SelectListItem { Text = displayType != "Design" ? _tokenizer.Replace(optionLabel, tokenData) : optionLabel }; + yield return new SelectListItem { Text = displayType != "Design" ? _tokenizer.Replace(optionLabel, tokenData) : optionLabel, Value = string.Empty }; } if (queryId == null) diff --git a/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/TaxonomyElementDriver.cs b/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/TaxonomyElementDriver.cs index c398f5726..806e72c0a 100644 --- a/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/TaxonomyElementDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.DynamicForms/Drivers/TaxonomyElementDriver.cs @@ -138,7 +138,7 @@ namespace Orchard.DynamicForms.Drivers { var runtimeValues = GetRuntimeValues(element); if (!String.IsNullOrWhiteSpace(optionLabel)) { - yield return new SelectListItem { Text = displayType != "Design" ? _tokenizer.Replace(optionLabel, tokenData) : optionLabel }; + yield return new SelectListItem { Text = displayType != "Design" ? _tokenizer.Replace(optionLabel, tokenData) : optionLabel, Value = string.Empty }; } if (taxonomyId == null) diff --git a/src/Orchard.Web/Modules/Orchard.DynamicForms/Migrations.cs b/src/Orchard.Web/Modules/Orchard.DynamicForms/Migrations.cs index e65035a55..16acf01e1 100644 --- a/src/Orchard.Web/Modules/Orchard.DynamicForms/Migrations.cs +++ b/src/Orchard.Web/Modules/Orchard.DynamicForms/Migrations.cs @@ -1,10 +1,15 @@ using System; +using System.Linq; +using System.Security.Cryptography; using Orchard.ContentManagement.MetaData; using Orchard.Core.Contents.Extensions; +using Orchard.Data; using Orchard.Data.Migration; namespace Orchard.DynamicForms { public class Migrations : DataMigrationImpl { + private readonly byte[] _oldLayoutHash = new byte[] { 0x91, 0x10, 0x3b, 0x97, 0xce, 0x1e, 0x1e, 0xc7, 0x7a, 0x41, 0xf7, 0x82, 0xe8, 0x58, 0x85, 0x91 }; + private const string DefaultFormLayoutData = @"{ ""elements"": [ @@ -59,7 +64,52 @@ namespace Orchard.DynamicForms { .WithSetting("LayoutTypePartSettings.DefaultLayoutData", DefaultFormLayoutData)) .WithSetting("Stereotype", "Widget") .DisplayedAs("Form Widget")); - return 1; + return 2; + } + + public int UpdateFrom1() { + // if the default layout data was unchanged, fix it with the new default + + var formLayoutPart = ContentDefinitionManager + .GetTypeDefinition("Form") + .Parts + .FirstOrDefault(x => x.PartDefinition.Name == "LayoutPart"); + + if (formLayoutPart != null && + formLayoutPart.Settings["LayoutTypePartSettings.DefaultLayoutData"] != null) { + var layout = formLayoutPart.Settings["LayoutTypePartSettings.DefaultLayoutData"]; + + if(GetMD5(layout) == _oldLayoutHash) { + ContentDefinitionManager.AlterTypeDefinition("Form", type => type + .WithPart("LayoutPart", p => p + .WithSetting("LayoutTypePartSettings.DefaultLayoutData", DefaultFormLayoutData)) + ); + } + } + + var formWidgetLayoutPart = ContentDefinitionManager + .GetTypeDefinition("FormWidget") + .Parts + .FirstOrDefault(x => x.PartDefinition.Name == "LayoutPart"); + + if (formWidgetLayoutPart != null && + formWidgetLayoutPart.Settings["LayoutTypePartSettings.DefaultLayoutData"] != null) { + var layout = formWidgetLayoutPart.Settings["LayoutTypePartSettings.DefaultLayoutData"]; + + if (GetMD5(layout) == _oldLayoutHash) { + ContentDefinitionManager.AlterTypeDefinition("FormWidget", type => type + .WithPart("LayoutPart", p => p + .WithSetting("LayoutTypePartSettings.DefaultLayoutData", DefaultFormLayoutData)) + ); + } + } + + return 2; + } + + private byte[] GetMD5(string text) { + byte[] encodedText = System.Text.Encoding.UTF8.GetBytes(text); + return ((HashAlgorithm)CryptoConfig.CreateFromName("MD5")).ComputeHash(encodedText); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.DynamicForms/Module.txt b/src/Orchard.Web/Modules/Orchard.DynamicForms/Module.txt index 50a66b14d..130adc262 100644 --- a/src/Orchard.Web/Modules/Orchard.DynamicForms/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.DynamicForms/Module.txt @@ -1,8 +1,8 @@ -Name: Custom Forms +Name: Dynamic Forms AntiForgery: enabled Author: The Orchard Team -Website: http://orchardcustomforms.codeplex.com -Version: 1.9.1 +Website: http://www.orchardproject.net/ +Version: 1.9.2 OrchardVersion: 1.9 Description: Create custom forms like contact forms using layouts. Features: diff --git a/src/Orchard.Web/Modules/Orchard.DynamicForms/Tokens/FormTokens.cs b/src/Orchard.Web/Modules/Orchard.DynamicForms/Tokens/FormTokens.cs index e26cb2626..68fbdb5c4 100644 --- a/src/Orchard.Web/Modules/Orchard.DynamicForms/Tokens/FormTokens.cs +++ b/src/Orchard.Web/Modules/Orchard.DynamicForms/Tokens/FormTokens.cs @@ -7,7 +7,7 @@ namespace Orchard.DynamicForms.Tokens { public void Describe(DescribeContext context) { context.For("FormSubmission", T("Dynamic Form submission"), T("Dynamic Form Submission tokens for use in workflows handling the Dynamic Form Submitted event.")) - .Token("Field:*", T("Field:"), T("The posted field value to access.")) + .Token("Field:*", T("Field:"), T("The posted field value to access."), "Text") .Token("IsValid:*", T("IsValid:"), T("The posted field validation status.")) ; } @@ -15,10 +15,20 @@ namespace Orchard.DynamicForms.Tokens { public void Evaluate(EvaluateContext context) { context.For("FormSubmission") .Token(token => token.StartsWith("Field:", StringComparison.OrdinalIgnoreCase) ? token.Substring("Field:".Length) : null, GetFieldValue) + .Chain(FilterChainParam, "Text", GetFieldValue) .Token(token => token.StartsWith("IsValid:", StringComparison.OrdinalIgnoreCase) ? token.Substring("IsValid:".Length) : null, GetFieldValidationStatus); } - private object GetFieldValue(string fieldName, FormSubmissionTokenContext context) { + private static Tuple FilterChainParam(string token) { + int tokenLength = "Field:".Length; + int chainIndex = token.IndexOf('.'); + if (token.StartsWith("Field:", StringComparison.OrdinalIgnoreCase) && chainIndex > tokenLength) + return new Tuple(token.Substring(tokenLength, chainIndex - tokenLength), token.Substring(chainIndex + 1)); + else + return null; + } + + private string GetFieldValue(string fieldName, FormSubmissionTokenContext context) { return context.PostedValues[fieldName]; } diff --git a/src/Orchard.Web/Modules/Orchard.Email/Module.txt b/src/Orchard.Web/Modules/Orchard.Email/Module.txt index 58fedf6d9..d81676eca 100644 --- a/src/Orchard.Web/Modules/Orchard.Email/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Email/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The Email Messaging module adds Email sending functionalities. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Email/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Email/Properties/AssemblyInfo.cs index 56a441495..165086d69 100644 --- a/src/Orchard.Web/Modules/Orchard.Email/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Email/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/BooleanFieldDriver.cs b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/BooleanFieldDriver.cs index 4e0afb916..50d0b98af 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/BooleanFieldDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/BooleanFieldDriver.cs @@ -61,7 +61,8 @@ namespace Orchard.Fields.Drivers { } protected override void Exporting(ContentPart part, BooleanField field, ExportContentContext context) { - context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", field.Value); + if (field.Value.HasValue) + context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", field.Value); } protected override void Describe(DescribeMembersContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/DateTimeFieldDriver.cs b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/DateTimeFieldDriver.cs index 1cda2654a..85b59a5d1 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/DateTimeFieldDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/DateTimeFieldDriver.cs @@ -167,7 +167,9 @@ namespace Orchard.Fields.Drivers { } protected override void Exporting(ContentPart part, DateTimeField field, ExportContentContext context) { - context.Element(GetPrefix(field, part)).SetAttributeValue("Value", XmlConvert.ToString(field.Storage.Get(null), XmlDateTimeSerializationMode.Utc)); + var value = field.Storage.Get(null); + if (value != DateTime.MinValue) + context.Element(GetPrefix(field, part)).SetAttributeValue("Value", XmlConvert.ToString(value, XmlDateTimeSerializationMode.Utc)); } protected override void Describe(DescribeMembersContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/EnumerationFieldDriver.cs b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/EnumerationFieldDriver.cs index 3d1607f39..4ba91fbbe 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/EnumerationFieldDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/EnumerationFieldDriver.cs @@ -4,6 +4,7 @@ using Orchard.ContentManagement.Handlers; using Orchard.Fields.Settings; using Orchard.Fields.Fields; using Orchard.Localization; +using System; namespace Orchard.Fields.Drivers { public class EnumerationFieldDriver : ContentFieldDriver { @@ -51,7 +52,8 @@ namespace Orchard.Fields.Drivers { } protected override void Exporting(ContentPart part, EnumerationField field, ExportContentContext context) { - context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", field.Value); + if (!String.IsNullOrEmpty(field.Value)) + context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", field.Value); } protected override void Describe(DescribeMembersContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/InputFieldDriver.cs b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/InputFieldDriver.cs index aefd92b55..8a85cf904 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/InputFieldDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/InputFieldDriver.cs @@ -56,7 +56,8 @@ namespace Orchard.Fields.Drivers { } protected override void Exporting(ContentPart part, InputField field, ExportContentContext context) { - context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", field.Value); + if (!String.IsNullOrEmpty(field.Value)) + context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", field.Value); } protected override void Describe(DescribeMembersContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/LinkFieldDriver.cs b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/LinkFieldDriver.cs index 24727c63f..ef92dd6e0 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/LinkFieldDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/LinkFieldDriver.cs @@ -62,9 +62,11 @@ namespace Orchard.Fields.Drivers { } protected override void Exporting(ContentPart part, LinkField field, ExportContentContext context) { - context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Text", field.Text); - context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Url", field.Value); - context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Target", field.Target); + if (!String.IsNullOrEmpty(field.Text) || !String.IsNullOrEmpty(field.Value) || !String.IsNullOrEmpty(field.Target)) { + context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Text", field.Text); + context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Url", field.Value); + context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Target", field.Target); + } } protected override void Describe(DescribeMembersContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/NumericFieldDriver.cs b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/NumericFieldDriver.cs index 95114fe2b..326c25a97 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Drivers/NumericFieldDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Fields/Drivers/NumericFieldDriver.cs @@ -104,7 +104,8 @@ namespace Orchard.Fields.Drivers { } protected override void Exporting(ContentPart part, NumericField field, ExportContentContext context) { - context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", !field.Value.HasValue ? String.Empty : field.Value.Value.ToString(CultureInfo.InvariantCulture)); + if (field.Value.HasValue) + context.Element(field.FieldDefinition.Name + "." + field.Name).SetAttributeValue("Value", field.Value.Value.ToString(CultureInfo.InvariantCulture)); } protected override void Describe(DescribeMembersContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Module.txt b/src/Orchard.Web/Modules/Orchard.Fields/Module.txt index 5e8bec54d..bda7b95f0 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Fields/Module.txt @@ -2,7 +2,7 @@ Name: Fields AntiForgery: enabled Author: Antoine Griffard, Sbastien Ros Website: http://orchardfields.codeplex.com -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Some content fields Features: diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Fields/Properties/AssemblyInfo.cs index 0243b4fab..4ed61055a 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Fields/Properties/AssemblyInfo.cs @@ -31,5 +31,5 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Enumeration.Edit.cshtml b/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Enumeration.Edit.cshtml index b5fec779d..ca39d9ade 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Enumeration.Edit.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Enumeration.Edit.cshtml @@ -8,34 +8,34 @@ @switch (settings.ListMode) { case ListMode.Dropdown: - @Html.DropDownListFor(m => m.Value, new SelectList(options, Model.Value)) + @Html.DropDownListFor(m => m.Value, new SelectList(options, Model.Value), settings.Required ? new {required = "required"} : null) break; case ListMode.Radiobutton: foreach (var option in options) { - if (string.IsNullOrWhiteSpace(option)) { - - } + if (string.IsNullOrWhiteSpace(option)) { + + } else { - - } + + } } break; case ListMode.Listbox: - @Html.ListBoxFor(m => m.SelectedValues, new MultiSelectList(options, Model.SelectedValues)) - break; + @Html.ListBoxFor(m => m.SelectedValues, new MultiSelectList(options, Model.SelectedValues), settings.Required ? new {required = "required"} : null) + break; case ListMode.Checkbox: - int index = 0; + int index = 0; foreach (var option in options) { index++; if (!string.IsNullOrWhiteSpace(option)) {
- - + required="required" } /> +
} } diff --git a/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Numeric.Edit.cshtml b/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Numeric.Edit.cshtml index 369dd5658..9a7a400a9 100644 --- a/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Numeric.Edit.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Fields/Views/EditorTemplates/Fields/Numeric.Edit.cshtml @@ -4,7 +4,9 @@
- @Html.TextBoxFor(m => m.Value, new { @class = "text small", type = "text", min = (Model.Settings.Minimum.HasValue) ? Model.Settings.Minimum.Value : 0, max = (Model.Settings.Maximum.HasValue) ? Model.Settings.Maximum.Value : 1000000, step = Math.Pow(10, 0 - Model.Settings.Scale).ToString(CultureInfo.InvariantCulture) }) + @(Model.Settings.Required + ? Html.TextBoxFor(m => m.Value, new {@class = "text-small", type = "text", min = (Model.Settings.Minimum.HasValue) ? Model.Settings.Minimum.Value : 0, max = (Model.Settings.Maximum.HasValue) ? Model.Settings.Maximum.Value : 1000000, step = Math.Pow(10, 0 - Model.Settings.Scale).ToString(CultureInfo.InvariantCulture), required = "required"}) + : Html.TextBoxFor(m => m.Value, new {@class = "text-small", type = "text", min = (Model.Settings.Minimum.HasValue) ? Model.Settings.Minimum.Value : 0, max = (Model.Settings.Maximum.HasValue) ? Model.Settings.Maximum.Value : 1000000, step = Math.Pow(10, 0 - Model.Settings.Scale).ToString(CultureInfo.InvariantCulture)})) @Html.ValidationMessageFor(m => m.Value) @if (HasText(Model.Settings.Hint)) { @Model.Settings.Hint diff --git a/src/Orchard.Web/Modules/Orchard.Forms/Module.txt b/src/Orchard.Web/Modules/Orchard.Forms/Module.txt index 2e0319fd3..4a8fed3d7 100644 --- a/src/Orchard.Web/Modules/Orchard.Forms/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Forms/Module.txt @@ -2,7 +2,7 @@ Name: Forms AntiForgery: enabled Author: The Orchard Team Website: http://orchardforms.codeplex.com -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides a system to publish and alter html forms. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Forms/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Forms/Properties/AssemblyInfo.cs index 5cbaaaa4d..1fc377313 100644 --- a/src/Orchard.Web/Modules/Orchard.Forms/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Forms/Properties/AssemblyInfo.cs @@ -31,5 +31,5 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.ImageEditor/Controllers/AdminController.cs b/src/Orchard.Web/Modules/Orchard.ImageEditor/Controllers/AdminController.cs index 595db7cf3..ffe66206d 100644 --- a/src/Orchard.Web/Modules/Orchard.ImageEditor/Controllers/AdminController.cs +++ b/src/Orchard.Web/Modules/Orchard.ImageEditor/Controllers/AdminController.cs @@ -45,6 +45,15 @@ namespace Orchard.ImageEditor.Controllers { [Themed(false)] public ActionResult Edit(string folderPath, string filename) { + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia)) + return new HttpUnauthorizedResult(); + + // Check permission. + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { + return new HttpUnauthorizedResult(); + } + var media = Services.ContentManager.Query().Where(x => x.FolderPath == folderPath && x.FileName == filename).Slice(0, 1).FirstOrDefault(); if (media == null) { @@ -64,12 +73,21 @@ namespace Orchard.ImageEditor.Controllers { [HttpPost] public ActionResult Upload(int id, string content, int width, int height) { + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia)) + return new HttpUnauthorizedResult(); + var media = Services.ContentManager.Get(id).As(); if (media == null) { return HttpNotFound(); } + // Check permission. + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(media.FolderPath)) { + return new HttpUnauthorizedResult(); + } + const string signature = "data:image/jpeg;base64,"; if (!content.StartsWith(signature, StringComparison.OrdinalIgnoreCase)) { @@ -96,7 +114,7 @@ namespace Orchard.ImageEditor.Controllers { } public ActionResult Proxy(string url) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent)) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia)) return HttpNotFound(); var sslFailureCallback = new RemoteCertificateValidationCallback((o, cert, chain, errors) => true); diff --git a/src/Orchard.Web/Modules/Orchard.ImageEditor/Module.txt b/src/Orchard.Web/Modules/Orchard.ImageEditor/Module.txt index fde3fe0df..24825db85 100644 --- a/src/Orchard.Web/Modules/Orchard.ImageEditor/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.ImageEditor/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Adds a client side image editor for Media Library Features: diff --git a/src/Orchard.Web/Modules/Orchard.ImageEditor/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.ImageEditor/Properties/AssemblyInfo.cs index db9ee8615..0ff254799 100644 --- a/src/Orchard.Web/Modules/Orchard.ImageEditor/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.ImageEditor/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Models/ExportContext.cs b/src/Orchard.Web/Modules/Orchard.ImportExport/Models/ExportContext.cs index 6d702593b..f829d360f 100644 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Models/ExportContext.cs +++ b/src/Orchard.Web/Modules/Orchard.ImportExport/Models/ExportContext.cs @@ -1,8 +1,10 @@ -using System.Xml.Linq; +using System; +using System.Xml.Linq; namespace Orchard.ImportExport.Models { public class ExportContext { public XDocument Document { get; set; } + [Obsolete] public ExportOptions ExportOptions { get; set; } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Models/SetupContext.cs b/src/Orchard.Web/Modules/Orchard.ImportExport/Models/SetupContext.cs deleted file mode 100644 index 0727bd888..000000000 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Models/SetupContext.cs +++ /dev/null @@ -1,15 +0,0 @@ -using System.Collections.Generic; -using System.Xml.Linq; - -namespace Orchard.ImportExport.Models { - public class SetupContext { - public string SiteName { get; set; } - public string AdminUsername { get; set; } - public string AdminPassword { get; set; } - public string DatabaseProvider { get; set; } - public string DatabaseConnectionString { get; set; } - public string DatabaseTablePrefix { get; set; } - public IEnumerable EnabledFeatures { get; set; } - public XDocument RecipeDocument { get; set; } - } -} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Module.txt b/src/Orchard.Web/Modules/Orchard.ImportExport/Module.txt index 3b6960348..7275d2c0d 100644 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.ImportExport/Module.txt @@ -3,7 +3,7 @@ Path: ImportExport AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides content item data import and export capability. Features: @@ -11,4 +11,4 @@ Features: Name: Import Export Description: Imports and exports content item data. Category: Content - Dependencies: Orchard.jQuery, Orchard.Recipes \ No newline at end of file + Dependencies: Orchard.jQuery, Orchard.Recipes, Orchard.Setup.Services diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Orchard.ImportExport.csproj b/src/Orchard.Web/Modules/Orchard.ImportExport/Orchard.ImportExport.csproj index 74c14c623..df4c6767f 100644 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Orchard.ImportExport.csproj +++ b/src/Orchard.Web/Modules/Orchard.ImportExport/Orchard.ImportExport.csproj @@ -90,9 +90,6 @@ - - - @@ -139,6 +136,10 @@ {fc1d74e8-7a4d-48f4-83de-95c6173780c4} Orchard.Recipes + + {8c7fcbc2-e6e1-405e-bfb5-d8d9e67a09c4} + Orchard.Setup + diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.ImportExport/Properties/AssemblyInfo.cs index 6f1259ddc..f0aa0453b 100644 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.ImportExport/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Providers/ImportActions/ExecuteRecipeAction.cs b/src/Orchard.Web/Modules/Orchard.ImportExport/Providers/ImportActions/ExecuteRecipeAction.cs index 9b52d8678..fa3cdb7cd 100644 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Providers/ImportActions/ExecuteRecipeAction.cs +++ b/src/Orchard.Web/Modules/Orchard.ImportExport/Providers/ImportActions/ExecuteRecipeAction.cs @@ -12,6 +12,7 @@ using Orchard.ImportExport.ViewModels; using Orchard.Mvc; using Orchard.Recipes.Models; using Orchard.Recipes.Services; +using Orchard.Setup.Services; using Orchard.Tasks; using Orchard.UI.Notify; @@ -50,13 +51,16 @@ namespace Orchard.ImportExport.Providers.ImportActions { _sweepGenerator = sweepGenerator; _recipeStepQueue = recipeStepQueue; _recipeStepResultRepository = recipeStepResultRepository; - } + + RecipeExecutionTimeout = 600; + } public override string Name { get { return "ExecuteRecipe"; } } public XDocument RecipeDocument { get; set; } public bool ResetSite { get; set; } public string SuperUserPassword { get; set; } + public int RecipeExecutionTimeout { get; set; } public override dynamic BuildEditor(dynamic shapeFactory) { return UpdateEditor(shapeFactory, null); @@ -148,8 +152,8 @@ namespace Orchard.ImportExport.Providers.ImportActions { // Give each execution step a chance to augment the recipe step before it will be scheduled. PrepareRecipe(recipeDocument); - // Sets the request timeout to 10 minutes to give enough time to execute custom recipes. - _orchardServices.WorkContext.HttpContext.Server.ScriptTimeout = 600; + // Sets the request timeout to a configurable amount of seconds to give enough time to execute custom recipes. + _orchardServices.WorkContext.HttpContext.Server.ScriptTimeout = RecipeExecutionTimeout; // Suspend background task execution. _sweepGenerator.Terminate(); @@ -181,7 +185,7 @@ namespace Orchard.ImportExport.Providers.ImportActions { private string Setup(XDocument recipeDocument) { // Prepare Setup. var setupContext = new SetupContext { - RecipeDocument = recipeDocument, + Recipe = _recipeParser.ParseRecipe(recipeDocument), AdminPassword = SuperUserPassword, AdminUsername = _orchardServices.WorkContext.CurrentSite.SuperUser, DatabaseConnectionString = _shellSettings.DataConnectionString, diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Services/ISetupService.cs b/src/Orchard.Web/Modules/Orchard.ImportExport/Services/ISetupService.cs deleted file mode 100644 index 176bf128c..000000000 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Services/ISetupService.cs +++ /dev/null @@ -1,7 +0,0 @@ -using Orchard.ImportExport.Models; - -namespace Orchard.ImportExport.Services { - public interface ISetupService : IDependency { - string Setup(SetupContext context); - } -} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.ImportExport/Services/SetupService.cs b/src/Orchard.Web/Modules/Orchard.ImportExport/Services/SetupService.cs deleted file mode 100644 index c125a21c9..000000000 --- a/src/Orchard.Web/Modules/Orchard.ImportExport/Services/SetupService.cs +++ /dev/null @@ -1,180 +0,0 @@ -using System; -using System.Linq; -using System.Security.Cryptography; -using System.Web; -using Orchard.ContentManagement; -using Orchard.Core.Settings.Models; -using Orchard.Data; -using Orchard.Data.Migration; -using Orchard.Data.Migration.Interpreters; -using Orchard.Data.Migration.Schema; -using Orchard.Environment; -using Orchard.Environment.Configuration; -using Orchard.Environment.Descriptor; -using Orchard.Environment.Descriptor.Models; -using Orchard.Environment.ShellBuilders; -using Orchard.Environment.State; -using Orchard.ImportExport.Models; -using Orchard.Localization.Services; -using Orchard.Logging; -using Orchard.Recipes.Services; -using Orchard.Security; -using Orchard.Settings; -using Orchard.Utility.Extensions; - -namespace Orchard.ImportExport.Services -{ - public class SetupService : Component, ISetupService { - private readonly ShellSettings _shellSettings; - private readonly IOrchardHost _orchardHost; - private readonly IShellSettingsManager _shellSettingsManager; - private readonly IShellContainerFactory _shellContainerFactory; - private readonly ICompositionStrategy _compositionStrategy; - private readonly IProcessingEngine _processingEngine; - - public SetupService( - ShellSettings shellSettings, - IOrchardHost orchardHost, - IShellSettingsManager shellSettingsManager, - IShellContainerFactory shellContainerFactory, - ICompositionStrategy compositionStrategy, - IProcessingEngine processingEngine) { - - _shellSettings = shellSettings; - _orchardHost = orchardHost; - _shellSettingsManager = shellSettingsManager; - _shellContainerFactory = shellContainerFactory; - _compositionStrategy = compositionStrategy; - _processingEngine = processingEngine; - } - - public string Setup(SetupContext context) { - string executionId; - - Logger.Information("Running setup for tenant '{0}'.", _shellSettings.Name); - - // The vanilla Orchard distibution has the following features enabled. - string[] hardcoded = { - // Framework - "Orchard.Framework", - // Core - "Common", "Containers", "Contents", "Dashboard", "Feeds", "Navigation","Scheduling", "Settings", "Shapes", "Title", - // Modules - "Orchard.Pages", "Orchard.ContentPicker", "Orchard.Themes", "Orchard.Users", "Orchard.Roles", "Orchard.Modules", - "PackagingServices", "Orchard.Packaging", "Gallery", "Orchard.Recipes" - }; - - context.EnabledFeatures = hardcoded.Union(context.EnabledFeatures ?? Enumerable.Empty()).Distinct().ToList(); - - var shellSettings = new ShellSettings(_shellSettings); - - if (String.IsNullOrEmpty(shellSettings.DataProvider)) { - shellSettings.DataProvider = context.DatabaseProvider; - shellSettings.DataConnectionString = context.DatabaseConnectionString; - shellSettings.DataTablePrefix = context.DatabaseTablePrefix; - } - - shellSettings.EncryptionAlgorithm = "AES"; - shellSettings.EncryptionKey = SymmetricAlgorithm.Create(shellSettings.EncryptionAlgorithm).Key.ToHexString(); - shellSettings.HashAlgorithm = "HMACSHA256"; - shellSettings.HashKey = HMAC.Create(shellSettings.HashAlgorithm).Key.ToHexString(); - - var shellDescriptor = new ShellDescriptor { - Features = context.EnabledFeatures.Select(name => new ShellFeature { Name = name }) - }; - - var shellBlueprint = _compositionStrategy.Compose(shellSettings, shellDescriptor); - - // Initialize database explicitly, and store shell descriptor. - using (var bootstrapLifetimeScope = _shellContainerFactory.CreateContainer(shellSettings, shellBlueprint)) { - using (var environment = bootstrapLifetimeScope.CreateWorkContextScope()) { - // Workaround to avoid a Transaction issue with PostgreSQL. - environment.Resolve().RequireNew(); - - var schemaBuilder = new SchemaBuilder(environment.Resolve()); - - schemaBuilder.CreateTable("Orchard_Framework_DataMigrationRecord", table => table - .Column("Id", column => column.PrimaryKey().Identity()) - .Column("DataMigrationClass") - .Column("Version")); - - schemaBuilder.AlterTable("Orchard_Framework_DataMigrationRecord", - table => table.AddUniqueConstraint("UC_DMR_DataMigrationClass_Version", "DataMigrationClass", "Version")); - - var dataMigrationManager = environment.Resolve(); - dataMigrationManager.Update("Settings"); - - foreach (var feature in context.EnabledFeatures) { - dataMigrationManager.Update(feature); - } - - var descriptorManager = environment.Resolve(); - descriptorManager.UpdateShellDescriptor(0, shellDescriptor.Features, shellDescriptor.Parameters); - } - } - - // In effect "pump messages" see PostMessage circa 1980. - while ( _processingEngine.AreTasksPending() ) - _processingEngine.ExecuteNextTask(); - - // Create a standalone environment. - // Must mark state as Running - otherwise standalone environment is created "for setup". - shellSettings.State = TenantState.Running; - using (var environment = _orchardHost.CreateStandaloneEnvironment(shellSettings)) { - try { - executionId = CreateTenantData(context, environment); - } - catch { - environment.Resolve().Cancel(); - throw; - } - } - - _shellSettingsManager.SaveSettings(shellSettings); - - return executionId; - } - - private string CreateTenantData(SetupContext context, IWorkContextScope environment) { - // Create superuser. - var membershipService = environment.Resolve(); - var user = membershipService.CreateUser( - new CreateUserParams( - context.AdminUsername, - context.AdminPassword, - email: String.Empty, - passwordQuestion: String.Empty, - passwordAnswer: String.Empty, - isApproved: true)); - - // Set superuser as current user for request (it will be set as the owner of all content items). - var authenticationService = environment.Resolve(); - authenticationService.SetAuthenticatedUserForRequest(user); - - // Set site name and settings. - var siteService = environment.Resolve(); - var siteSettings = siteService.GetSiteSettings().As(); - siteSettings.SiteSalt = Guid.NewGuid().ToString("N"); - siteSettings.SiteName = context.SiteName; - siteSettings.SuperUser = context.AdminUsername; - siteSettings.SiteCulture = "en-US"; - - // Add default culture. - var cultureManager = environment.Resolve(); - cultureManager.AddCulture("en-US"); - - // Execute recipe - var recipeParser = environment.Resolve(); - var recipe = recipeParser.ParseRecipe(context.RecipeDocument); - var recipeManager = environment.Resolve(); - var executionId = recipeManager.Execute(recipe); - - // Null check: temporary fix for running setup in command line. - if (HttpContext.Current != null) { - authenticationService.SignIn(user, true); - } - - return executionId; - } - } -} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Indexing/Module.txt b/src/Orchard.Web/Modules/Orchard.Indexing/Module.txt index 99ab8cfca..122669177 100644 --- a/src/Orchard.Web/Modules/Orchard.Indexing/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Indexing/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The Indexing module enables the site to be indexed. The index generated by this module can then be used by the search module to provide an integrated full-text search experience to a web site. FeatureDescription: Indexing infrastructure. Requires an index implementation like the Lucene module. diff --git a/src/Orchard.Web/Modules/Orchard.Indexing/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Indexing/Properties/AssemblyInfo.cs index 999336b2c..eefbb569d 100644 --- a/src/Orchard.Web/Modules/Orchard.Indexing/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Indexing/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.JobsQueue/Module.txt b/src/Orchard.Web/Modules/Orchard.JobsQueue/Module.txt index 05186f82d..48b0c9042 100644 --- a/src/Orchard.Web/Modules/Orchard.JobsQueue/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.JobsQueue/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: This module provides a jobs queue to process jobs asynchronously. Features: diff --git a/src/Orchard.Web/Modules/Orchard.JobsQueue/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.JobsQueue/Properties/AssemblyInfo.cs index 43830bba5..fa1bc9edb 100644 --- a/src/Orchard.Web/Modules/Orchard.JobsQueue/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.JobsQueue/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.JobsQueue/Services/JobsQueueProcessor.cs b/src/Orchard.Web/Modules/Orchard.JobsQueue/Services/JobsQueueProcessor.cs index 039bfbc75..74bec9bb6 100644 --- a/src/Orchard.Web/Modules/Orchard.JobsQueue/Services/JobsQueueProcessor.cs +++ b/src/Orchard.Web/Modules/Orchard.JobsQueue/Services/JobsQueueProcessor.cs @@ -1,7 +1,6 @@ using System; using System.Collections.Generic; using System.Linq; -using System.Threading; using Newtonsoft.Json.Linq; using Orchard.Environment; using Orchard.Events; @@ -13,12 +12,11 @@ namespace Orchard.JobsQueue.Services { public class JobsQueueProcessor : IJobsQueueProcessor { private readonly Work _jobsQueueManager; private readonly Work _eventBus; - private readonly ReaderWriterLockSlim _rwl = new ReaderWriterLockSlim(); private readonly IDistributedLockService _distributedLockService; public JobsQueueProcessor( Work jobsQueueManager, - Work eventBus, + Work eventBus, IDistributedLockService distributedLockService) { _jobsQueueManager = jobsQueueManager; @@ -28,26 +26,19 @@ namespace Orchard.JobsQueue.Services { } public ILogger Logger { get; set; } - public void ProcessQueue() { - // prevent two threads on the same machine to process the message queue - if (_rwl.TryEnterWriteLock(0)) { - try { - DistributedLock @lock; - if(_distributedLockService.TryAcquireLockForMachine(GetType().FullName, TimeSpan.FromMinutes(5), out @lock)){ - using (@lock) { - IEnumerable messages; - while ((messages = _jobsQueueManager.Value.GetJobs(0, 10).ToArray()).Any()) { - foreach (var message in messages) { - ProcessMessage(message); - } - } + public void ProcessQueue() { + IDistributedLock @lock; + if (_distributedLockService.TryAcquireLock(GetType().FullName, TimeSpan.FromMinutes(5), out @lock)) { + using (@lock) { + IEnumerable messages; + + while ((messages = _jobsQueueManager.Value.GetJobs(0, 10).ToArray()).Any()) { + foreach (var message in messages) { + ProcessMessage(message); } } } - finally { - _rwl.ExitWriteLock(); - } } } diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/AdminMenu.cs b/src/Orchard.Web/Modules/Orchard.Layouts/AdminMenu.cs index 35db9afee..0a675af3d 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/AdminMenu.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/AdminMenu.cs @@ -11,9 +11,9 @@ namespace Orchard.Layouts { builder .AddImageSet("layouts") .Add(T("Layouts"), "8.5", layouts => layouts - .Action("List", "Admin", new {id = "Layout", area = "Contents"}) + .Action("List", "Admin", new {id = "Layout", area = "Contents"}).Permission(Permissions.ManageLayouts) .LinkToFirstChild(false) - .Add(T("Elements"), "1", elements => elements.Action("Index", "BlueprintAdmin", new {area = "Orchard.Layouts"}))); + .Add(T("Elements"), "1", elements => elements.Action("Index", "BlueprintAdmin", new {area = "Orchard.Layouts"}).Permission(Permissions.ManageLayouts))); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Editor.js b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Editor.js index 8595615e8..b7dcde8aa 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Editor.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Editor.js @@ -74,45 +74,6 @@ return host.addElement(contentType); }; - $scope.toggleInlineEditing = function () { - if (!$scope.element.inlineEditingIsActive) { - $scope.element.inlineEditingIsActive = true; - $element.find(".layout-toolbar-container").show(); - var selector = "#layout-editor-" + $scope.$id + " .layout-html .layout-content-markup[data-templated=false]"; - var firstContentEditorId = $(selector).first().attr("id"); - tinymce.init({ - selector: selector, - theme: "modern", - schema: "html5", - plugins: [ - "advlist autolink lists link image charmap print preview hr anchor pagebreak", - "searchreplace wordcount visualblocks visualchars code fullscreen", - "insertdatetime media nonbreaking table contextmenu directionality", - "emoticons template paste textcolor colorpicker textpattern", - "fullscreen autoresize" - ], - toolbar: "undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close", - convert_urls: false, - valid_elements: "*[*]", - // Shouldn't be needed due to the valid_elements setting, but TinyMCE would strip script.src without it. - extended_valid_elements: "script[type|defer|src|language]", - statusbar: false, - skin: "orchardlightgray", - inline: true, - fixed_toolbar_container: "#layout-editor-" + $scope.$id + " .layout-toolbar-container", - init_instance_callback: function (editor) { - if (editor.id == firstContentEditorId) - tinymce.execCommand("mceFocus", false, editor.id); - } - }); - } - else { - tinymce.remove("#layout-editor-" + $scope.$id + " .layout-content-markup"); - $element.find(".layout-toolbar-container").hide(); - $scope.element.inlineEditingIsActive = false; - } - }; - $(document).on("cut copy paste", function (e) { // If the pseudo clipboard was already invoked (which happens on the first clipboard // operation after page load even if native clipboard support exists) then sit this @@ -164,23 +125,12 @@ element.find(".layout-toolbar-container").click(function (e) { e.stopPropagation(); }); - // Intercept mousedown on editor while in inline editing mode to - // prevent current editor from losing focus. - element.mousedown(function (e) { - if (scope.element.inlineEditingIsActive) { - e.preventDefault(); - e.stopPropagation(); - } - }) // Unfocus and unselect everything on click outside of canvas. $(window).click(function (e) { - // Except when in inline editing mode. - if (!scope.element.inlineEditingIsActive) { - scope.$apply(function () { - scope.element.activeElement = null; - scope.element.focusedElement = null; - }); - } + scope.$apply(function () { + scope.element.activeElement = null; + scope.element.focusedElement = null; + }); }); } }; diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Html.js b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Html.js index 4dc6a4f1e..94890f1d6 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Html.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Directives/Html.js @@ -35,13 +35,6 @@ templateUrl: environment.templateUrl("Html"), replace: true, link: function (scope, element) { - // Mouse down events must not be intercepted by drag and drop while inline editing is active, - // otherwise clicks in inline editors will have no effect. - element.find(".layout-content-markup").mousedown(function (e) { - if (scope.element.editor.inlineEditingIsActive) { - e.stopPropagation(); - } - }); } }; } diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Services/ScopeConfigurator.js b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Services/ScopeConfigurator.js index 988fdde43..a2ce3a6ba 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Services/ScopeConfigurator.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/LayoutEditor/Services/ScopeConfigurator.js @@ -15,7 +15,7 @@ var resetFocus = false; var element = $scope.element; - if (element.editor.isDragging || element.editor.inlineEditingIsActive) + if (element.editor.isDragging) return; // If native clipboard support exists, the pseudo-clipboard will have been disabled. diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Editor.js b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Editor.js index a93081991..86b664804 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Editor.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Editor.js @@ -9,7 +9,6 @@ this.focusedElement = null; this.dropTargetElement = null; this.isDragging = false; - this.inlineEditingIsActive = false; this.isResizing = false; this.resetToolboxElements = function () { diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Element.js b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Element.js index 6f30d5508..6a983cf25 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Element.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/JavaScript/Models/Element.js @@ -57,7 +57,7 @@ this.setIsActive = function (value) { if (!this.editor) return; - if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing) + if (this.editor.isDragging || this.editor.isResizing) return; if (value) @@ -77,7 +77,7 @@ return; if (!this.children && this.isTemplated) return; - if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing) + if (this.editor.isDragging || this.editor.isResizing) return; this.editor.focusedElement = this; diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/Less/LayoutEditor/Canvas.less b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/Less/LayoutEditor/Canvas.less index aab898205..ec7aba16f 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Assets/Less/LayoutEditor/Canvas.less +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Assets/Less/LayoutEditor/Canvas.less @@ -6,4 +6,4 @@ display: none; } } -} +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/BlueprintAdminController.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/BlueprintAdminController.cs index 1dcac2e31..6c5b8ae36 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/BlueprintAdminController.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/BlueprintAdminController.cs @@ -34,7 +34,8 @@ namespace Orchard.Layouts.Controllers { ICultureAccessor cultureAccessor, IShapeFactory shapeFactory, ITransactionManager transactionManager, - ISignals signals) { + ISignals signals, + IOrchardServices orchardServices) { _elementBlueprintService = elementBlueprintService; _notifier = notifier; @@ -43,12 +44,19 @@ namespace Orchard.Layouts.Controllers { _shapeFactory = shapeFactory; _transactionManager = transactionManager; _signals = signals; + Services = orchardServices; + T = NullLocalizer.Instance; } + public IOrchardServices Services { get; set; } public Localizer T { get; set; } public ActionResult Index() { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var blueprints = _elementBlueprintService.GetBlueprints().ToArray(); var viewModel = new BlueprintsIndexViewModel { Blueprints = blueprints @@ -57,6 +65,10 @@ namespace Orchard.Layouts.Controllers { } public ActionResult Browse() { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var categories = RemoveBlueprints(_elementManager.GetCategories(DescribeElementsContext.Empty)).ToArray(); var viewModel = new BrowseElementsViewModel { Categories = categories @@ -65,6 +77,10 @@ namespace Orchard.Layouts.Controllers { } public ActionResult Create(string id) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + if (String.IsNullOrWhiteSpace(id)) return RedirectToAction("Browse"); @@ -80,6 +96,10 @@ namespace Orchard.Layouts.Controllers { [HttpPost] public ActionResult Create(string id, CreateElementBlueprintViewModel model) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var describeContext = DescribeElementsContext.Empty; var descriptor = _elementManager.GetElementDescriptorByTypeName(describeContext, id); var baseElement = _elementManager.ActivateElement(descriptor); @@ -100,7 +120,11 @@ namespace Orchard.Layouts.Controllers { return RedirectToAction("Edit", new { id = blueprint.Id }); } - public ViewResult Edit(int id) { + public ActionResult Edit(int id) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var blueprint = _elementBlueprintService.GetBlueprint(id); var describeContext = DescribeElementsContext.Empty; var descriptor = _elementManager.GetElementDescriptorByTypeName(describeContext, blueprint.BaseElementTypeName); @@ -125,6 +149,10 @@ namespace Orchard.Layouts.Controllers { [HttpPost] [ValidateInput(false)] public ActionResult Edit(int id, ElementDataViewModel model) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var blueprint = _elementBlueprintService.GetBlueprint(id); var describeContext = DescribeElementsContext.Empty; var descriptor = _elementManager.GetElementDescriptorByTypeName(describeContext, blueprint.BaseElementTypeName); @@ -154,6 +182,10 @@ namespace Orchard.Layouts.Controllers { } public ActionResult Properties(int id) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var blueprint = _elementBlueprintService.GetBlueprint(id); var describeContext = DescribeElementsContext.Empty; var descriptor = _elementManager.GetElementDescriptorByTypeName(describeContext, blueprint.BaseElementTypeName); @@ -171,6 +203,10 @@ namespace Orchard.Layouts.Controllers { [HttpPost] public ActionResult Properties(int id, ElementBlueprintPropertiesViewModel model) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var blueprint = _elementBlueprintService.GetBlueprint(id); var describeContext = DescribeElementsContext.Empty; var descriptor = _elementManager.GetElementDescriptorByTypeName(describeContext, blueprint.BaseElementTypeName); @@ -191,7 +227,12 @@ namespace Orchard.Layouts.Controllers { return RedirectToAction("Index"); } + [HttpPost] public ActionResult Delete(int id) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + var blueprint = _elementBlueprintService.GetBlueprint(id); if (blueprint == null) @@ -204,7 +245,12 @@ namespace Orchard.Layouts.Controllers { [FormValueRequired("submit.BulkEdit")] [ActionName("Index")] + [HttpPost] public ActionResult BulkDelete(IEnumerable blueprintIds) { + if (!Services.Authorizer.Authorize(Permissions.ManageLayouts, T("Not authorized to manage layouts."))) { + return new HttpUnauthorizedResult(); + } + if (blueprintIds == null || !blueprintIds.Any()) { _notifier.Error(T("Please select the blueprints to delete.")); } diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/ElementController.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/ElementController.cs index 170bdfb19..38dffa4ce 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/ElementController.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/ElementController.cs @@ -142,18 +142,7 @@ namespace Orchard.Layouts.Controllers { _objectStore.Set(session, state); return RedirectToAction("Edit", new {session = session}); } - - public RedirectToRouteResult Add(string session, string typeName, int? contentId = null, string contentType = null) { - var state = new ElementSessionState { - TypeName = typeName, - ContentId = contentId, - ContentType = contentType - }; - - _objectStore.Set(session, state); - return RedirectToAction("Edit", new { session = session }); - } - + public ViewResult Edit(string session) { var sessionState = _objectStore.Get(session); var contentId = sessionState.ContentId; diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/LayoutController.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/LayoutController.cs index 753bd9aa8..a72f1783b 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/LayoutController.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Controllers/LayoutController.cs @@ -6,6 +6,7 @@ using Orchard.ContentManagement; using Orchard.Layouts.Elements; using Orchard.Layouts.Framework.Elements; using Orchard.Layouts.Services; +using Orchard.Localization; using Orchard.UI.Admin; namespace Orchard.Layouts.Controllers { @@ -15,15 +16,25 @@ namespace Orchard.Layouts.Controllers { private readonly ILayoutManager _layoutManager; private readonly ILayoutModelMapper _mapper; - public LayoutController(IContentManager contentManager, ILayoutManager layoutManager, ILayoutModelMapper mapper) { + public LayoutController( + IContentManager contentManager, + ILayoutManager layoutManager, + ILayoutModelMapper mapper, + IOrchardServices orchardServices) { _contentManager = contentManager; _layoutManager = layoutManager; _mapper = mapper; + Services = orchardServices; + + T = NullLocalizer.Instance; } + public IOrchardServices Services { get; set; } + public Localizer T { get; set; } + [HttpPost, ValidateInput(enableValidation: false)] - public ContentResult ApplyTemplate(int? templateId = null, string layoutData = null, int? contentId = null, string contentType = null) { + public ActionResult ApplyTemplate(int? templateId = null, string layoutData = null, int? contentId = null, string contentType = null) { var template = templateId != null ? _layoutManager.GetLayout(templateId.Value) : null; var templateElements = template != null ? _layoutManager.LoadElements(template).ToList() : default(IEnumerable); var describeContext = CreateDescribeElementsContext(contentId, contentType); diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/LayoutPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/LayoutPartDriver.cs index 116ab44bc..ace0e73fa 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/LayoutPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/LayoutPartDriver.cs @@ -90,9 +90,18 @@ namespace Orchard.Layouts.Drivers { protected override DriverResult Editor(LayoutPart part, IUpdateModel updater, dynamic shapeHelper) { return ContentShape("Parts_Layout_Edit", () => { - if (part.Id == 0 && String.IsNullOrWhiteSpace(part.LayoutData)) { - part.LayoutData = part.TypePartDefinition.Settings.GetModel().DefaultLayoutData; + + var settings = part.TypePartDefinition.Settings.GetModel(); + + // If the default layout setting is left empty, use the one from the service + if (String.IsNullOrWhiteSpace(settings.DefaultLayoutData)) { + var defaultData = _serializer.Serialize(_layoutManager.CreateDefaultLayout()); + part.LayoutData = defaultData; + } + else { + part.LayoutData = settings.DefaultLayoutData; + } } var viewModel = new LayoutPartViewModel { @@ -140,6 +149,11 @@ namespace Orchard.Layouts.Drivers { } protected override void Importing(LayoutPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + context.ImportChildEl(part.PartDefinition.Name, "LayoutData", s => { part.LayoutData = s; _layoutManager.Importing(new ImportLayoutContext { diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/ProjectionElementDriver.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/ProjectionElementDriver.cs index d42f56128..ec8f4631c 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/ProjectionElementDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Drivers/ProjectionElementDriver.cs @@ -290,11 +290,11 @@ namespace Orchard.Layouts.Drivers { var query = element.QueryId != null ? _contentManager.Get(element.QueryId.Value) : default(QueryPart); var layout = query != null && element.LayoutId != null ? _layoutRepository.Get(element.LayoutId.Value) : default(LayoutRecord); var queryIdentity = query != null ? _contentManager.GetItemMetadata(query).Identity.ToString() : default(string); - var layoutIndex = layout != null ? query.Layouts.IndexOf(layout) : default(int?); + var layoutIndex = layout != null ? query.Layouts.IndexOf(layout) : -1; // -1 is the Default Layout. - if (queryIdentity != null && layoutIndex != null) { + if (queryIdentity != null) { context.ExportableData["QueryId"] = queryIdentity; - context.ExportableData["LayoutIndex"] = layoutIndex.Value.ToString(); + context.ExportableData["LayoutIndex"] = layoutIndex.ToString(); } } @@ -307,10 +307,9 @@ namespace Orchard.Layouts.Drivers { var queryPart = query.As(); var layoutIndex = XmlHelper.Parse(context.ExportableData.Get("LayoutIndex")); - var layout = queryPart.Layouts[layoutIndex]; element.QueryId = queryPart.Id; - element.LayoutId = layout.Id; + element.LayoutId = layoutIndex != -1 ? queryPart.Layouts[layoutIndex].Id : -1; } private static string GetLayoutDescription(IEnumerable layouts, LayoutRecord l) { diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Elements/Column.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Elements/Column.cs index 7f87b3e0d..f0b8d3e0a 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Elements/Column.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Elements/Column.cs @@ -21,7 +21,7 @@ namespace Orchard.Layouts.Elements { } public int? Width { - get { return this.Retrieve("Width") ?? this.Retrieve("ColumnSpan") ?? 0; } // Falling back on "ColumnSpan" for backward compatibility. + get { return this.Retrieve("Width") ?? this.Retrieve("ColumnSpan") ?? 12; } // Falling back on "ColumnSpan" for backward compatibility. set { this.Store(x => x.Width, value); } } diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Framework/Elements/Element.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Framework/Elements/Element.cs index 87959efaf..16aa590e6 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Framework/Elements/Element.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Framework/Elements/Element.cs @@ -3,7 +3,7 @@ using Orchard.Localization; using Orchard.Utility.Extensions; namespace Orchard.Layouts.Framework.Elements { - public abstract class Element { + public abstract class Element : IElement { protected Element() { T = NullLocalizer.Instance; Data = new ElementDataDictionary(); diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Framework/Elements/IElement.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Framework/Elements/IElement.cs new file mode 100644 index 000000000..0cf0c8f1f --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Framework/Elements/IElement.cs @@ -0,0 +1,3 @@ +namespace Orchard.Layouts.Framework.Elements { + public interface IElement : ITransientDependency {} +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Handlers/LayoutPartHandler.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Handlers/LayoutPartHandler.cs index 78f9a595b..0e2c3b37a 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Handlers/LayoutPartHandler.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Handlers/LayoutPartHandler.cs @@ -16,7 +16,6 @@ namespace Orchard.Layouts.Handlers { private readonly IContentPartDisplay _contentPartDisplay; private readonly IShapeDisplay _shapeDisplay; private readonly ILayoutSerializer _serializer; - private readonly IStaticHttpContextScopeFactory _staticHttpContextScopeFactory; private readonly IAliasService _aliasService; public LayoutPartHandler( @@ -26,7 +25,6 @@ namespace Orchard.Layouts.Handlers { IContentPartDisplay contentPartDisplay, IShapeDisplay shapeDisplay, ILayoutSerializer serializer, - IStaticHttpContextScopeFactory staticHttpContextScopeFactory, IAliasService aliasService) { _layoutManager = layoutManager; @@ -34,7 +32,6 @@ namespace Orchard.Layouts.Handlers { _contentPartDisplay = contentPartDisplay; _shapeDisplay = shapeDisplay; _serializer = serializer; - _staticHttpContextScopeFactory = staticHttpContextScopeFactory; _aliasService = aliasService; Filters.Add(StorageFilter.For(repository)); @@ -44,22 +41,13 @@ namespace Orchard.Layouts.Handlers { private void IndexLayout(IndexContentContext context, LayoutPart part) { var layoutShape = _contentPartDisplay.BuildDisplay(part); - var layoutHtml = RenderShape(layoutShape); + var layoutHtml = _shapeDisplay.Display(layoutShape); context.DocumentIndex .Add("body", layoutHtml).RemoveTags().Analyze() .Add("format", "html").Store(); } - /// - /// This method of rendering is safe even in background tasks. - /// - private string RenderShape(dynamic shape) { - using (_staticHttpContextScopeFactory.CreateStaticScope()) { - return _shapeDisplay.Display(shape); - } - } - private void UpdateTemplateClients(PublishContentContext context, LayoutPart part) { UpdateTemplateClients(part); } diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Migrations.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Migrations.cs index c87ade13c..746120763 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Migrations.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Migrations.cs @@ -33,7 +33,6 @@ namespace Orchard.Layouts { .WithPart("LayoutPart", p => p .WithSetting("LayoutTypePartSettings.IsTemplate", "True")) .DisplayedAs("Layout") - .Listable() .Draftable()); ContentDefinitionManager.AlterTypeDefinition("LayoutWidget", type => type diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Models/ElementSessionState.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Models/ElementSessionState.cs index 26859a51f..6fcff6c7f 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Models/ElementSessionState.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Models/ElementSessionState.cs @@ -1,9 +1,12 @@ +using System; + namespace Orchard.Layouts.Models { - public class ElementSessionState { + [Serializable] + public class ElementSessionState { public string TypeName { get; set; } public string ElementData { get; set; } public string ElementEditorData { get; set; } public int? ContentId { get; set; } public string ContentType { get; set; } } -} \ No newline at end of file +} diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Module.txt b/src/Orchard.Web/Modules/Orchard.Layouts/Module.txt index 6e61893e6..4f21508df 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Module.txt @@ -10,7 +10,7 @@ Features: Orchard.Layouts: Name: Layouts Description: Provides tools to create layouts. - Dependencies: Common, Orchard.jQuery, Orchard.Forms, Orchard.Tokens, Orchard.MediaLibrary, TinyMce, Orchard.Widgets + Dependencies: Common, Orchard.jQuery, Orchard.Forms, Orchard.Tokens, Orchard.MediaLibrary, Orchard.Widgets Category: Layout Orchard.Layouts.Snippets: Name: Layout Snippets diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Orchard.Layouts.csproj b/src/Orchard.Web/Modules/Orchard.Layouts/Orchard.Layouts.csproj index 39ace50e4..c25bf3fb8 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Orchard.Layouts.csproj +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Orchard.Layouts.csproj @@ -358,6 +358,7 @@ + @@ -367,6 +368,7 @@ + diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Permissions.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Permissions.cs new file mode 100644 index 000000000..a7905be9b --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Permissions.cs @@ -0,0 +1,40 @@ +using System.Collections.Generic; +using Orchard.Environment.Extensions.Models; +using Orchard.Security.Permissions; + +namespace Orchard.Layouts { + public class Permissions : IPermissionProvider { + public static readonly Permission ManageLayouts = new Permission { Description = "Managing Layouts", Name = "ManageLayouts" }; + + public virtual Feature Feature { get; set; } + + public IEnumerable GetPermissions() { + return new[] { + ManageLayouts, + }; + } + + public IEnumerable GetDefaultStereotypes() { + return new[] { + new PermissionStereotype { + Name = "Administrator", + Permissions = new[] { ManageLayouts } + }, + new PermissionStereotype { + Name = "Editor", + Permissions = new[] { ManageLayouts } + }, + new PermissionStereotype { + Name = "Moderator", + }, + new PermissionStereotype { + Name = "Author" + }, + new PermissionStereotype { + Name = "Contributor", + }, + }; + } + + } +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js index 405a8f9e7..2be99a094 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js @@ -62,7 +62,7 @@ angular var resetFocus = false; var element = $scope.element; - if (element.editor.isDragging || element.editor.inlineEditingIsActive) + if (element.editor.isDragging) return; // If native clipboard support exists, the pseudo-clipboard will have been disabled. @@ -443,45 +443,6 @@ angular return host.addElement(contentType); }; - $scope.toggleInlineEditing = function () { - if (!$scope.element.inlineEditingIsActive) { - $scope.element.inlineEditingIsActive = true; - $element.find(".layout-toolbar-container").show(); - var selector = "#layout-editor-" + $scope.$id + " .layout-html .layout-content-markup[data-templated=false]"; - var firstContentEditorId = $(selector).first().attr("id"); - tinymce.init({ - selector: selector, - theme: "modern", - schema: "html5", - plugins: [ - "advlist autolink lists link image charmap print preview hr anchor pagebreak", - "searchreplace wordcount visualblocks visualchars code fullscreen", - "insertdatetime media nonbreaking table contextmenu directionality", - "emoticons template paste textcolor colorpicker textpattern", - "fullscreen autoresize" - ], - toolbar: "undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close", - convert_urls: false, - valid_elements: "*[*]", - // Shouldn't be needed due to the valid_elements setting, but TinyMCE would strip script.src without it. - extended_valid_elements: "script[type|defer|src|language]", - statusbar: false, - skin: "orchardlightgray", - inline: true, - fixed_toolbar_container: "#layout-editor-" + $scope.$id + " .layout-toolbar-container", - init_instance_callback: function (editor) { - if (editor.id == firstContentEditorId) - tinymce.execCommand("mceFocus", false, editor.id); - } - }); - } - else { - tinymce.remove("#layout-editor-" + $scope.$id + " .layout-content-markup"); - $element.find(".layout-toolbar-container").hide(); - $scope.element.inlineEditingIsActive = false; - } - }; - $(document).on("cut copy paste", function (e) { // If the pseudo clipboard was already invoked (which happens on the first clipboard // operation after page load even if native clipboard support exists) then sit this @@ -533,23 +494,12 @@ angular element.find(".layout-toolbar-container").click(function (e) { e.stopPropagation(); }); - // Intercept mousedown on editor while in inline editing mode to - // prevent current editor from losing focus. - element.mousedown(function (e) { - if (scope.element.inlineEditingIsActive) { - e.preventDefault(); - e.stopPropagation(); - } - }) // Unfocus and unselect everything on click outside of canvas. $(window).click(function (e) { - // Except when in inline editing mode. - if (!scope.element.inlineEditingIsActive) { - scope.$apply(function () { - scope.element.activeElement = null; - scope.element.focusedElement = null; - }); - } + scope.$apply(function () { + scope.element.activeElement = null; + scope.element.focusedElement = null; + }); }); } }; @@ -730,13 +680,6 @@ angular templateUrl: environment.templateUrl("Html"), replace: true, link: function (scope, element) { - // Mouse down events must not be intercepted by drag and drop while inline editing is active, - // otherwise clicks in inline editors will have no effect. - element.find(".layout-content-markup").mousedown(function (e) { - if (scope.element.editor.inlineEditingIsActive) { - e.stopPropagation(); - } - }); } }; } @@ -1051,4 +994,4 @@ angular }; } ]); -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Module.js","Clipboard.js","ScopeConfigurator.js","Editor.js","Canvas.js","Child.js","Column.js","Content.js","Html.js","Grid.js","Row.js","Popup.js","Toolbox.js","ToolboxGroup.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC3LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC/CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC9MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"LayoutEditor.js","sourcesContent":["angular.module(\"LayoutEditor\", [\"ngSanitize\", \"ngResource\", \"ui.sortable\"]);","var LayoutEditor;\r\n(function(LayoutEditor) {\r\n\r\n    var Clipboard = function () {\r\n        var self = this;\r\n        this._clipboardData = {};\r\n        this._isDisabled = false;\r\n        this._wasInvoked = false;\r\n\r\n        this.setData = function(contentType, data) {\r\n            self._clipboardData[contentType] = data;\r\n            self._wasInvoked = true;\r\n        };\r\n        this.getData = function (contentType) {\r\n            return self._clipboardData[contentType];\r\n            self._wasInvoked = true;\r\n        };\r\n        this.disable = function() {\r\n            self._isDisabled = true;\r\n            self._wasInvoked = false;\r\n            self._clipboardData = {};\r\n        };\r\n        this.isDisabled = function () {\r\n            return self._isDisabled;\r\n        }\r\n        this.wasInvoked = function () {\r\n            return self._wasInvoked;\r\n        }\r\n    }\r\n\r\n    LayoutEditor.Clipboard = new Clipboard();\r\n\r\n    angular\r\n        .module(\"LayoutEditor\")\r\n        .factory(\"clipboard\", [\r\n            function() {\r\n                return {\r\n                    setData: LayoutEditor.Clipboard.setData,\r\n                    getData: LayoutEditor.Clipboard.getData,\r\n                    disable: LayoutEditor.Clipboard.disable,\r\n                    isDisabled: LayoutEditor.Clipboard.isDisabled,\r\n                    wasInvoked: LayoutEditor.Clipboard.wasInvoked\r\n                };\r\n            }\r\n        ]);\r\n})(LayoutEditor || (LayoutEditor = {}));","angular\r\n    .module(\"LayoutEditor\")\r\n    .factory(\"scopeConfigurator\", [\"$timeout\", \"clipboard\",\r\n        function ($timeout, clipboard) {\r\n            return {\r\n\r\n                configureForElement: function ($scope, $element) {\r\n                \r\n                    $element.find(\".layout-panel\").click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n\r\n                    $element.parent().keydown(function (e) {\r\n                        var handled = false;\r\n                        var resetFocus = false;\r\n                        var element = $scope.element;\r\n                    \r\n                        if (element.editor.isDragging || element.editor.inlineEditingIsActive)\r\n                            return;\r\n\r\n                        // If native clipboard support exists, the pseudo-clipboard will have been disabled.\r\n                        if (!clipboard.isDisabled()) {\r\n                            var focusedElement = element.editor.focusedElement;\r\n                            if (!!focusedElement) {\r\n                                // Pseudo clipboard handling for browsers not allowing real clipboard operations.\r\n                                if (e.ctrlKey) {\r\n                                    switch (e.which) {\r\n                                    case 67: // C\r\n                                        focusedElement.copy(clipboard);\r\n                                        break;\r\n                                    case 88: // X\r\n                                        focusedElement.cut(clipboard);\r\n                                        break;\r\n                                    case 86: // V\r\n                                        focusedElement.paste(clipboard);\r\n                                        break;\r\n                                    }\r\n                                }\r\n                            }\r\n                        }\r\n\r\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 46) { // Del\r\n                            $scope.delete(element);\r\n                            handled = true;\r\n                        } else if (!e.ctrlKey && !e.shiftKey && !e.altKey && (e.which == 32 || e.which == 27)) { // Space or Esc\r\n                            $element.find(\".layout-panel-action-properties\").first().click();\r\n                            handled = true;\r\n                        }\r\n\r\n                        if (element.type == \"Content\") { // This is a content element.\r\n                            if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 13) { // Enter\r\n                                $element.find(\".layout-panel-action-edit\").first().click();\r\n                                handled = true;\r\n                            }\r\n                        }\r\n\r\n                        if (!!element.children) { // This is a container.\r\n                            if (!e.ctrlKey && !e.shiftKey && e.altKey && e.which == 40) { // Alt+Down\r\n                                if (element.children.length > 0)\r\n                                    element.children[0].setIsFocused();\r\n                                handled = true;\r\n                            }\r\n\r\n                            if (element.type == \"Column\") { // This is a column.\r\n                                var connectAdjacent = !e.ctrlKey;\r\n                                if (e.which == 37) { // Left\r\n                                    if (e.altKey)\r\n                                        element.expandLeft(connectAdjacent);\r\n                                    if (e.shiftKey)\r\n                                        element.contractRight(connectAdjacent);\r\n                                    handled = true;\r\n                                } else if (e.which == 39) { // Right\r\n                                    if (e.altKey)\r\n                                        element.contractLeft(connectAdjacent);\r\n                                    if (e.shiftKey)\r\n                                        element.expandRight(connectAdjacent);\r\n                                    handled = true;\r\n                                }\r\n                            }\r\n                        }\r\n\r\n                        if (!!element.parent) { // This is a child.\r\n                            if (e.altKey && e.which == 38) { // Alt+Up\r\n                                element.parent.setIsFocused();\r\n                                handled = true;\r\n                            }\r\n\r\n                            if (element.parent.type == \"Row\") { // Parent is a horizontal container.\r\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Left\r\n                                    element.parent.moveFocusPrevChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Right\r\n                                    element.parent.moveFocusNextChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Ctrl+Left\r\n                                    element.moveUp();\r\n                                    resetFocus = true;\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Ctrl+Right\r\n                                    element.moveDown();\r\n                                    handled = true;\r\n                                }\r\n                            }\r\n                            else { // Parent is a vertical container.\r\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Up\r\n                                    element.parent.moveFocusPrevChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Down\r\n                                    element.parent.moveFocusNextChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Ctrl+Up\r\n                                    element.moveUp();\r\n                                    resetFocus = true;\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Ctrl+Down\r\n                                    element.moveDown();\r\n                                    handled = true;\r\n                                }\r\n                            }\r\n                        }\r\n\r\n                        if (handled) {\r\n                            e.preventDefault();\r\n                        }\r\n\r\n                        e.stopPropagation();\r\n\r\n                        $scope.$apply(); // Event is not triggered by Angular directive but raw event handler on element.\r\n\r\n                        // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\r\n                        if (resetFocus) {\r\n                            window.setTimeout(function () {\r\n                                $scope.$apply(function () {\r\n                                    element.editor.focusedElement.setIsFocused();\r\n                                });\r\n                            }, 100);\r\n                        }\r\n                    });\r\n\r\n                    $scope.element.setIsFocusedEventHandlers.push(function () {\r\n                        $element.parent().focus();\r\n                    });\r\n\r\n                    $scope.delete = function (element) {\r\n                        element.delete();\r\n                    }\r\n                },\r\n\r\n                configureForContainer: function ($scope, $element) {\r\n                    var element = $scope.element;\r\n\r\n                    //$scope.isReceiving = false; // True when container is receiving an external element via drag/drop.\r\n                    $scope.getShowChildrenPlaceholder = function () {\r\n                        return $scope.element.children.length === 0 && !$scope.element.getIsDropTarget();\r\n                    };\r\n\r\n                    $scope.sortableOptions = {\r\n                        cursor: \"move\",\r\n                        delay: 150,\r\n                        disabled: element.getIsSealed(),\r\n                        distance: 5,\r\n                        //handle: element.children.length < 2 ? \".imaginary-class\" : false, // For some reason doesn't get re-evaluated after adding more children.\r\n                        start: function (e, ui) {\r\n                            $scope.$apply(function () {\r\n                                element.setIsDropTarget(true);\r\n                                element.editor.isDragging = true;\r\n                            });\r\n                            // Make the drop target placeholder as high as the item being dragged.\r\n                            ui.placeholder.height(ui.item.height() - 4);\r\n                            ui.placeholder.css(\"min-height\", 0);\r\n                        },\r\n                        stop: function (e, ui) {\r\n                            $scope.$apply(function () {\r\n                                element.editor.isDragging = false;\r\n                                element.setIsDropTarget(false);\r\n                            });\r\n                        },\r\n                        over: function (e, ui) {\r\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\r\n                                if (!!ui.sender[0].dropTargetTimeout) {\r\n                                    $timeout.cancel(ui.sender[0].dropTargetTimeout);\r\n                                    ui.sender[0].dropTargetTimeout = null;\r\n                                }\r\n                                $timeout(function () {\r\n                                    if (element.type == \"Row\") {\r\n                                        // If there was a previous drop target and it was a row, roll back any pending column adds to it.\r\n                                        var previousDropTarget = element.editor.dropTargetElement;\r\n                                        if (!!previousDropTarget && previousDropTarget.type == \"Row\")\r\n                                            previousDropTarget.rollbackAddColumn();\r\n                                    }\r\n                                    element.setIsDropTarget(false);\r\n                                });\r\n                                ui.sender[0].dropTargetTimeout = $timeout(function () {\r\n                                    if (element.type == \"Row\") {\r\n                                        var receivedColumn = ui.item.sortable.model;\r\n                                        var receivedColumnWidth = Math.floor(12 / (element.children.length + 1));\r\n                                        receivedColumn.width = receivedColumnWidth;\r\n                                        receivedColumn.offset = 0;\r\n                                        element.beginAddColumn(receivedColumnWidth);\r\n                                        // Make the drop target placeholder the correct width and as high as the highest existing column in the row.\r\n                                        var maxHeight = _.max(_($element.find(\"> .layout-children > .layout-column:not(.ui-sortable-placeholder)\")).map(function (e) {\r\n                                            return $(e).height();\r\n                                        }));\r\n                                        for (i = 1; i <= 12; i++)\r\n                                            ui.placeholder.removeClass(\"col-xs-\" + i);\r\n                                        ui.placeholder.addClass(\"col-xs-\" + receivedColumn.width);\r\n                                        if (maxHeight > 0) {\r\n                                            ui.placeholder.height(maxHeight);\r\n                                            ui.placeholder.css(\"min-height\", 0);\r\n                                        }\r\n                                        else {\r\n                                            ui.placeholder.height(0);\r\n                                            ui.placeholder.css(\"min-height\", \"\");\r\n                                        }\r\n                                    }\r\n                                    element.setIsDropTarget(true);\r\n                                }, 150);\r\n                            }\r\n                        },\r\n                        receive: function (e, ui) {\r\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\r\n                                $scope.$apply(function () {\r\n                                    var receivedElement = ui.item.sortable.model;\r\n                                    if (!!receivedElement) {\r\n                                        if (element.type == \"Row\")\r\n                                            element.commitAddColumn();\r\n                                        // Should ideally call LayoutEditor.Container.addChild() instead, but since this handler\r\n                                        // is run *before* the ui-sortable directive's handler, if we try to add the child to the\r\n                                        // array that handler will get an exception when trying to do the same.\r\n                                        // Because of this, we need to invoke \"setParent\" so that specific container types can perform element speficic initialization.\r\n                                        receivedElement.setEditor(element.editor);\r\n                                        receivedElement.setParent(element);\r\n\r\n                                        if (!!receivedElement.hasEditor) {\r\n                                            $scope.$root.editElement(receivedElement).then(function (args) {\r\n                                                if (!args.cancel) {\r\n                                                    receivedElement.data = args.element.data;\r\n                                                    receivedElement.applyElementEditorModel(args.elementEditorModel);\r\n\r\n                                                    if (!!receivedElement.setHtml)\r\n                                                        receivedElement.setHtml(args.element.html);\r\n                                                }\r\n                                                $timeout(function () {\r\n                                                    if (!!args.cancel)\r\n                                                        receivedElement.delete();\r\n                                                    else\r\n                                                        receivedElement.setIsFocused();\r\n                                                    //$scope.isReceiving = false;\r\n                                                    element.setIsDropTarget(false);\r\n\r\n                                                });\r\n                                                return;\r\n                                            });\r\n                                        }\r\n                                    }\r\n                                    $timeout(function () {\r\n                                        //$scope.isReceiving = false;\r\n                                        element.setIsDropTarget(false);\r\n                                        if (!!receivedElement)\r\n                                            receivedElement.setIsFocused();\r\n                                    });\r\n                                });\r\n                            }\r\n                        }\r\n                    };\r\n\r\n                    $scope.click = function (child, e) {\r\n                        if (!child.editor.isDragging)\r\n                            child.setIsFocused();\r\n                        e.stopPropagation();\r\n                    };\r\n\r\n                    $scope.getClasses = function (child) {\r\n                        var result = [\"layout-element\"];\r\n\r\n                        if (!!child.children) {\r\n                            result.push(\"layout-container\");\r\n                            if (child.getIsSealed())\r\n                                result.push(\"layout-container-sealed\");\r\n                        }\r\n\r\n                        result.push(\"layout-\" + child.type.toLowerCase());\r\n\r\n                        if (!!child.dropTargetClass)\r\n                            result.push(child.dropTargetClass);\r\n\r\n                        // TODO: Move these to either the Column directive or the Column model class.\r\n                        if (child.type == \"Row\") {\r\n                            result.push(\"row\");\r\n                            if (!child.canAddColumn())\r\n                                result.push(\"layout-row-full\");\r\n                        }\r\n                        if (child.type == \"Column\") {\r\n                            result.push(\"col-xs-\" + child.width);\r\n                            result.push(\"col-xs-offset-\" + child.offset);\r\n                        }\r\n                        if (child.type == \"Content\")\r\n                            result.push(\"layout-content-\" + child.contentTypeClass);\r\n\r\n                        if (child.getIsActive())\r\n                            result.push(\"layout-element-active\");\r\n                        if (child.getIsFocused())\r\n                            result.push(\"layout-element-focused\");\r\n                        if (child.getIsSelected())\r\n                            result.push(\"layout-element-selected\");\r\n                        if (child.getIsDropTarget())\r\n                            result.push(\"layout-element-droptarget\");\r\n                        if (child.isTemplated)\r\n                            result.push(\"layout-element-templated\");\r\n\r\n                        return result;\r\n                    };\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutEditor\", [\"environment\",\r\n        function (environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: {},\r\n                controller: [\"$scope\", \"$element\", \"$attrs\", \"$compile\", \"clipboard\",\r\n                    function ($scope, $element, $attrs, $compile, clipboard) {\r\n                        if (!!$attrs.model)\r\n                            $scope.element = eval($attrs.model);\r\n                        else\r\n                            throw new Error(\"The 'model' attribute must evaluate to a LayoutEditor.Editor object.\");\r\n\r\n                        $scope.click = function (canvas, e) {\r\n                            if (!canvas.editor.isDragging)\r\n                                canvas.setIsFocused();\r\n                            e.stopPropagation();\r\n                        };\r\n\r\n                        $scope.getClasses = function (canvas) {\r\n                            var result = [\"layout-element\", \"layout-container\", \"layout-canvas\"];\r\n\r\n                            if (canvas.getIsActive())\r\n                                result.push(\"layout-element-active\");\r\n                            if (canvas.getIsFocused())\r\n                                result.push(\"layout-element-focused\");\r\n                            if (canvas.getIsSelected())\r\n                                result.push(\"layout-element-selected\");\r\n                            if (canvas.getIsDropTarget())\r\n                                result.push(\"layout-element-droptarget\");\r\n                            if (canvas.isTemplated)\r\n                                result.push(\"layout-element-templated\");\r\n\r\n                            return result;\r\n                        };\r\n\r\n                        // An unfortunate side-effect of the next hack on line 54 is that the created elements aren't added to the DOM yet, so we can't use it to get to the parent \".layout-desiger\" element.\r\n                        // Work around: access that element directly (which efectively turns multiple layout editors on a single page impossible). \r\n                        // //var layoutDesignerHost = $element.closest(\".layout-designer\").data(\"layout-designer-host\");\r\n                        var layoutDesignerHost = $(\".layout-designer\").data(\"layout-designer-host\");\r\n\r\n                        $scope.$root.layoutDesignerHost = layoutDesignerHost;\r\n\r\n                        layoutDesignerHost.element.on(\"replacecanvas\", function (e, args) {\r\n                            var editor = $scope.element;\r\n                            var canvasData = {\r\n                                data: args.canvas.data,\r\n                                htmlId: args.canvas.htmlId,\r\n                                htmlClass: args.canvas.htmlClass,\r\n                                htmlStyle: args.canvas.htmlStyle,\r\n                                isTemplated: args.canvas.isTemplated,\r\n                                children: args.canvas.children\r\n                            };\r\n\r\n                            // HACK: Instead of simply updating the $scope.element with a new instance, we need to replace the entire orc-layout-editor markup\r\n                            // in order for angular to rebind starting with the Canvas element. Otherwise, for some reason, it will rebind starting with the first child of Canvas.\r\n                            // You can see this happening when setting a breakpoint in ScopeConfigurator where containers are initialized with drag & drop: on page load, the first element\r\n                            // is a Canvas (good), but after having selected another template, the first element is (typically) a Grid (bad).\r\n                            // Simply recompiling the orc-layout-editor directive will cause the entire thing to be generated, which works just fine as well (even though not is nice as simply leveraging model binding).\r\n                            layoutDesignerHost.editor = window.layoutEditor = new LayoutEditor.Editor(editor.config, canvasData);\r\n                            var template = \"<orc-layout-editor\" + \" model='window.layoutEditor' />\";\r\n                            var html = $compile(template)($scope);\r\n                            $(\".layout-editor-holder\").html(html);\r\n                        });\r\n\r\n                        $scope.$root.editElement = function (element) {\r\n                            var host = $scope.$root.layoutDesignerHost;\r\n                            return host.editElement(element);\r\n                        };\r\n\r\n                        $scope.$root.addElement = function (contentType) {\r\n                            var host = $scope.$root.layoutDesignerHost;\r\n                            return host.addElement(contentType);\r\n                        };\r\n\r\n                        $scope.toggleInlineEditing = function () {\r\n                            if (!$scope.element.inlineEditingIsActive) {\r\n                                $scope.element.inlineEditingIsActive = true;\r\n                                $element.find(\".layout-toolbar-container\").show();\r\n                                var selector = \"#layout-editor-\" + $scope.$id + \" .layout-html .layout-content-markup[data-templated=false]\";\r\n                                var firstContentEditorId = $(selector).first().attr(\"id\");\r\n                                tinymce.init({\r\n                                    selector: selector,\r\n                                    theme: \"modern\",\r\n                                    schema: \"html5\",\r\n                                    plugins: [\r\n                                        \"advlist autolink lists link image charmap print preview hr anchor pagebreak\",\r\n                                        \"searchreplace wordcount visualblocks visualchars code fullscreen\",\r\n                                        \"insertdatetime media nonbreaking table contextmenu directionality\",\r\n                                        \"emoticons template paste textcolor colorpicker textpattern\",\r\n                                        \"fullscreen autoresize\"\r\n                                    ],\r\n                                    toolbar: \"undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close\",\r\n                                    convert_urls: false,\r\n                                    valid_elements: \"*[*]\",\r\n                                    // Shouldn't be needed due to the valid_elements setting, but TinyMCE would strip script.src without it.\r\n                                    extended_valid_elements: \"script[type|defer|src|language]\",\r\n                                    statusbar: false,\r\n                                    skin: \"orchardlightgray\",\r\n                                    inline: true,\r\n                                    fixed_toolbar_container: \"#layout-editor-\" + $scope.$id + \" .layout-toolbar-container\",\r\n                                    init_instance_callback: function (editor) {\r\n                                        if (editor.id == firstContentEditorId)\r\n                                            tinymce.execCommand(\"mceFocus\", false, editor.id);\r\n                                    }\r\n                                });\r\n                            }\r\n                            else {\r\n                                tinymce.remove(\"#layout-editor-\" + $scope.$id + \" .layout-content-markup\");\r\n                                $element.find(\".layout-toolbar-container\").hide();\r\n                                $scope.element.inlineEditingIsActive = false;\r\n                            }\r\n                        };\r\n\r\n                        $(document).on(\"cut copy paste\", function (e) {\r\n                            // If the pseudo clipboard was already invoked (which happens on the first clipboard\r\n                            // operation after page load even if native clipboard support exists) then sit this\r\n                            // one operation out, but make sure whatever is on the pseudo clipboard gets migrated\r\n                            // to the native clipboard for subsequent operations.\r\n                            if (clipboard.wasInvoked()) {\r\n                                e.originalEvent.clipboardData.setData(\"text/plain\", clipboard.getData(\"text/plain\"));\r\n                                e.originalEvent.clipboardData.setData(\"text/json\", clipboard.getData(\"text/json\"));\r\n                                e.preventDefault();\r\n                            }\r\n                            else {\r\n                                var focusedElement = $scope.element.focusedElement;\r\n                                if (!!focusedElement) {\r\n                                    $scope.$apply(function () {\r\n                                        switch (e.type) {\r\n                                            case \"copy\":\r\n                                                focusedElement.copy(e.originalEvent.clipboardData);\r\n                                                break;\r\n                                            case \"cut\":\r\n                                                focusedElement.cut(e.originalEvent.clipboardData);\r\n                                                break;\r\n                                            case \"paste\":\r\n                                                focusedElement.paste(e.originalEvent.clipboardData);\r\n                                                break;\r\n                                        }\r\n                                    });\r\n\r\n                                    // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\r\n                                    window.setTimeout(function () {\r\n                                        $scope.$apply(function () {\r\n                                            if (!!$scope.element.focusedElement)\r\n                                                $scope.element.focusedElement.setIsFocused();\r\n                                        });\r\n                                    }, 100);\r\n\r\n                                    e.preventDefault();\r\n                                }\r\n                            }\r\n\r\n                            // Native clipboard support obviously exists, so disable the peudo clipboard from now on.\r\n                            clipboard.disable();\r\n                        });\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Editor\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                    // No clicks should propagate from the TinyMCE toolbars.\r\n                    element.find(\".layout-toolbar-container\").click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n                    // Intercept mousedown on editor while in inline editing mode to \r\n                    // prevent current editor from losing focus.\r\n                    element.mousedown(function (e) {\r\n                        if (scope.element.inlineEditingIsActive) {\r\n                            e.preventDefault();\r\n                            e.stopPropagation();\r\n                        }\r\n                    })\r\n                    // Unfocus and unselect everything on click outside of canvas.\r\n                    $(window).click(function (e) {\r\n                        // Except when in inline editing mode.\r\n                        if (!scope.element.inlineEditingIsActive) {\r\n                            scope.$apply(function () {\r\n                                scope.element.activeElement = null;\r\n                                scope.element.focusedElement = null;\r\n                            });\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutCanvas\", [\"scopeConfigurator\", \"environment\",\r\n        function (scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\", \"$attrs\",\r\n                    function ($scope, $element, $attrs) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"y\";\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Canvas\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutChild\", [\"$compile\",\r\n        function ($compile) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                link: function (scope, element) {\r\n                    var template = \"<orc-layout-\" + scope.element.type.toLowerCase() + \" element='element' />\";\r\n                    var html = $compile(template)(scope);\r\n                    $(element).replaceWith(html);\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutColumn\", [\"$compile\", \"scopeConfigurator\", \"environment\",\r\n        function ($compile, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"y\";\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Column\"),\r\n                replace: true,\r\n                link: function (scope, element, attrs) {\r\n                    element.find(\".layout-column-resize-bar\").draggable({\r\n                        axis: \"x\",\r\n                        helper: \"clone\",\r\n                        revert: true,\r\n                        start: function (e, ui) {\r\n                            scope.$apply(function () {\r\n                                scope.element.editor.isResizing = true;\r\n                            });\r\n                        },\r\n                        drag: function (e, ui) {\r\n                            var columnElement = element.parent();\r\n                            var columnSize = columnElement.width() / scope.element.width;\r\n                            var connectAdjacent = !e.ctrlKey;\r\n                            if ($(e.target).hasClass(\"layout-column-resize-bar-left\")) {\r\n                                var delta = ui.offset.left - columnElement.offset().left;\r\n                                if (delta < -columnSize && scope.element.canExpandLeft(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.expandLeft(connectAdjacent);\r\n                                    });\r\n                                }\r\n                                else if (delta > columnSize && scope.element.canContractLeft(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.contractLeft(connectAdjacent);\r\n                                    });\r\n                                }\r\n                            }\r\n                            else if ($(e.target).hasClass(\"layout-column-resize-bar-right\")) {\r\n                                var delta = ui.offset.left - columnElement.width() - columnElement.offset().left;\r\n                                if (delta > columnSize && scope.element.canExpandRight(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.expandRight(connectAdjacent);\r\n                                    });\r\n                                }\r\n                                else if (delta < -columnSize && scope.element.canContractRight(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.contractRight(connectAdjacent);\r\n                                    });\r\n                                }\r\n                            }\r\n\r\n                        },\r\n                        stop: function (e, ui) {\r\n                            scope.$apply(function () {\r\n                              scope.element.editor.isResizing = false;\r\n                            });\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutContent\", [\"$sce\", \"scopeConfigurator\", \"environment\",\r\n        function ($sce, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        $scope.edit = function () {\r\n                            $scope.$root.editElement($scope.element).then(function (args) {\r\n                                $scope.$apply(function () {\r\n                                    if (args.cancel)\r\n                                        return;\r\n\r\n                                    $scope.element.data = args.element.data;\r\n                                    $scope.element.setHtml(args.element.html);\r\n                                });\r\n                            });\r\n                        };\r\n\r\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\r\n                        $scope.element.setHtml = function (html) {\r\n                            $scope.element.html = html;\r\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\r\n                        };\r\n\r\n                        $scope.element.setHtml($scope.element.html);\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Content\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutHtml\", [\"$sce\", \"scopeConfigurator\", \"environment\",\r\n        function ($sce, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        $scope.edit = function () {\r\n                            $scope.$root.editElement($scope.element).then(function (args) {\r\n                                $scope.$apply(function () {\r\n                                    if (args.cancel)\r\n                                        return;\r\n\r\n                                    $scope.element.data = args.element.data;\r\n                                    $scope.element.setHtml(args.element.html);\r\n                                });\r\n                            });\r\n                        };\r\n                        $scope.updateContent = function (e) {\r\n                            $scope.element.setHtml(e.target.innerHTML);\r\n                        };\r\n\r\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\r\n                        $scope.element.setHtml = function (html) {\r\n                            $scope.element.html = html;\r\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\r\n                        };\r\n\r\n                        $scope.element.setHtml($scope.element.html);\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Html\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                    // Mouse down events must not be intercepted by drag and drop while inline editing is active,\r\n                    // otherwise clicks in inline editors will have no effect.\r\n                    element.find(\".layout-content-markup\").mousedown(function (e) {\r\n                        if (scope.element.editor.inlineEditingIsActive) {\r\n                            e.stopPropagation();\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutGrid\", [\"$compile\", \"scopeConfigurator\", \"environment\",\r\n        function ($compile, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"y\";\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Grid\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutRow\", [\"$compile\", \"scopeConfigurator\", \"environment\",\r\n        function ($compile, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"x\";\r\n                        $scope.sortableOptions[\"ui-floating\"] = true;\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Row\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutPopup\", [\r\n        function () {\r\n            return {\r\n                restrict: \"A\",\r\n                link: function (scope, element, attrs) {\r\n                    var popup = $(element);\r\n                    var trigger = popup.closest(\".layout-popup-trigger\");\r\n                    var parentElement = popup.closest(\".layout-element\");\r\n                    trigger.click(function () {\r\n                        popup.toggle();\r\n                        if (popup.is(\":visible\")) {\r\n                            popup.position({\r\n                                my: attrs.orcLayoutPopupMy || \"left top\",\r\n                                at: attrs.orcLayoutPopupAt || \"left bottom+4px\",\r\n                                of: trigger\r\n                            });\r\n                            popup.find(\"input\").first().focus();\r\n                        }\r\n                    });\r\n                    popup.click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n                    parentElement.click(function (e) {\r\n                        popup.hide();\r\n                    });\r\n                    popup.keydown(function (e) {\r\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 27) // Esc\r\n                            popup.hide();\r\n                        e.stopPropagation();\r\n                    });\r\n                    popup.on(\"cut copy paste\", function (e) {\r\n                        // Allow clipboard operations in popup without invoking clipboard event handlers on parent element.\r\n                        e.stopPropagation();\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutToolbox\", [\"$compile\", \"environment\",\r\n        function ($compile, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n\r\n                        $scope.resetElements = function () {\r\n\r\n                            $scope.gridElements = [\r\n                                LayoutEditor.Grid.from({\r\n                                    toolboxIcon: \"\\uf00a\",\r\n                                    toolboxLabel: \"Grid\",\r\n                                    toolboxDescription: \"Empty grid.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.rowElements = [\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (1 column)\",\r\n                                    toolboxDescription: \"Row with 1 column.\",\r\n                                    children: LayoutEditor.Column.times(1)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (2 columns)\",\r\n                                    toolboxDescription: \"Row with 2 columns.\",\r\n                                    children: LayoutEditor.Column.times(2)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (3 columns)\",\r\n                                    toolboxDescription: \"Row with 3 columns.\",\r\n                                    children: LayoutEditor.Column.times(3)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (4 columns)\",\r\n                                    toolboxDescription: \"Row with 4 columns.\",\r\n                                    children: LayoutEditor.Column.times(4)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (6 columns)\",\r\n                                    toolboxDescription: \"Row with 6 columns.\",\r\n                                    children: LayoutEditor.Column.times(6)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (12 columns)\",\r\n                                    toolboxDescription: \"Row with 12 columns.\",\r\n                                    children: LayoutEditor.Column.times(12)\r\n                                }), LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (empty)\",\r\n                                    toolboxDescription: \"Empty row.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.columnElements = [\r\n                                LayoutEditor.Column.from({\r\n                                    toolboxIcon: \"\\uf0db\",\r\n                                    toolboxLabel: \"Column\",\r\n                                    toolboxDescription: \"Empty column.\",\r\n                                    width: 1,\r\n                                    offset: 0,\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.canvasElements = [\r\n                                LayoutEditor.Canvas.from({\r\n                                    toolboxIcon: \"\\uf044\",\r\n                                    toolboxLabel: \"Canvas\",\r\n                                    toolboxDescription: \"Empty canvas.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.contentElementCategories = _($scope.element.config.categories).map(function (category) {\r\n                                return {\r\n                                    name: category.name,\r\n                                    elements: _(category.contentTypes).map(function (contentType) {\r\n                                        var type = contentType.type;\r\n                                        var factory = LayoutEditor.factories[type] || LayoutEditor.factories[\"Content\"];\r\n                                        var item = {\r\n                                            isTemplated: false,\r\n                                            contentType: contentType.id,\r\n                                            contentTypeLabel: contentType.label,\r\n                                            contentTypeClass: contentType.typeClass,\r\n                                            data: null,\r\n                                            hasEditor: contentType.hasEditor,\r\n                                            html: contentType.html\r\n                                        };\r\n                                        var element = factory(item);\r\n                                        element.toolboxIcon = contentType.icon || \"\\uf1c9\";\r\n                                        element.toolboxLabel = contentType.label;\r\n                                        element.toolboxDescription = contentType.description;\r\n                                        return element;\r\n                                    })\r\n                                };\r\n                            });\r\n\r\n                        };\r\n\r\n                        $scope.resetElements();\r\n\r\n                        $scope.getSortableOptions = function (type) {\r\n                            var editorId = $element.closest(\".layout-editor\").attr(\"id\");\r\n                            var parentClasses;\r\n                            var placeholderClasses;\r\n                            var floating = false;\r\n\r\n                            switch (type) {\r\n                                case \"Grid\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-grid ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Row\":\r\n                                    parentClasses = [\".layout-grid\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-row row ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Column\":\r\n                                    parentClasses = [\".layout-row:not(.layout-row-full)\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-column ui-sortable-placeholder\";\r\n                                    floating = true; // To ensure a smooth horizontal-list reordering. https://github.com/angular-ui/ui-sortable#floating\r\n                                    break;\r\n                                case \"Content\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-content ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Canvas\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-grid ui-sortable-placeholder\";\r\n                                    break;\r\n                            }\r\n\r\n                            return {\r\n                                cursor: \"move\",\r\n                                connectWith: _(parentClasses).map(function (e) { return \"#\" + editorId + \" \" + e + \":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children\"; }).join(\", \"),\r\n                                placeholder: placeholderClasses,\r\n                                \"ui-floating\": floating,\r\n                                create: function (e, ui) {\r\n                                    e.target.isToolbox = true; // Will indicate to connected sortables that dropped items were sent from toolbox.\r\n                                },\r\n                                start: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = true;\r\n                                    });\r\n                                },\r\n                                stop: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = false;\r\n                                        $scope.resetElements();\r\n                                    });\r\n                                },\r\n                                over: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.canvas.setIsDropTarget(false);\r\n                                    });\r\n                                },\r\n                            }\r\n                        };\r\n\r\n                        var layoutIsCollapsedCookieName = \"layoutToolboxCategory_Layout_IsCollapsed\";\r\n                        $scope.layoutIsCollapsed = $.cookie(layoutIsCollapsedCookieName) === \"true\";\r\n\r\n                        $scope.toggleLayoutIsCollapsed = function (e) {\r\n                            $scope.layoutIsCollapsed = !$scope.layoutIsCollapsed;\r\n                            $.cookie(layoutIsCollapsedCookieName, $scope.layoutIsCollapsed, { expires: 365 }); // Remember collapsed state for a year.\r\n                            e.preventDefault();\r\n                            e.stopPropagation();\r\n                        };\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Toolbox\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                    var toolbox = element.find(\".layout-toolbox\");\r\n                    $(window).on(\"resize scroll\", function (e) {\r\n                        var canvas = element.parent().find(\".layout-canvas\");\r\n                        // If the canvas is taller than the toolbox, make the toolbox sticky-positioned within the editor\r\n                        // to help the user avoid excessive vertical scrolling.\r\n                        var canvasIsTaller = !!canvas && canvas.height() > toolbox.height();\r\n                        var windowPos = $(window).scrollTop();\r\n                        if (canvasIsTaller && windowPos > element.offset().top + element.height() - toolbox.height()) {\r\n                            toolbox.addClass(\"sticky-bottom\");\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                        }\r\n                        else if (canvasIsTaller && windowPos > element.offset().top) {\r\n                            toolbox.addClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                        else {\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutToolboxGroup\", [\"$compile\", \"environment\",\r\n        function ($compile, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { category: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        var isCollapsedCookieName = \"layoutToolboxCategory_\" + $scope.category.name + \"_IsCollapsed\";\r\n                        $scope.isCollapsed = $.cookie(isCollapsedCookieName) === \"true\";\r\n                        $scope.toggleIsCollapsed = function (e) {\r\n                            $scope.isCollapsed = !$scope.isCollapsed;\r\n                            $.cookie(isCollapsedCookieName, $scope.isCollapsed, { expires: 365 }); // Remember collapsed state for a year.\r\n                            e.preventDefault();\r\n                            e.stopPropagation();\r\n                        };\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"ToolboxGroup\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);"],"sourceRoot":"/source/"} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Module.js","Clipboard.js","ScopeConfigurator.js","Editor.js","Canvas.js","Child.js","Column.js","Content.js","Html.js","Grid.js","Row.js","Popup.js","Toolbox.js","ToolboxGroup.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC9MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"LayoutEditor.js","sourcesContent":["angular.module(\"LayoutEditor\", [\"ngSanitize\", \"ngResource\", \"ui.sortable\"]);","var LayoutEditor;\r\n(function(LayoutEditor) {\r\n\r\n    var Clipboard = function () {\r\n        var self = this;\r\n        this._clipboardData = {};\r\n        this._isDisabled = false;\r\n        this._wasInvoked = false;\r\n\r\n        this.setData = function(contentType, data) {\r\n            self._clipboardData[contentType] = data;\r\n            self._wasInvoked = true;\r\n        };\r\n        this.getData = function (contentType) {\r\n            return self._clipboardData[contentType];\r\n            self._wasInvoked = true;\r\n        };\r\n        this.disable = function() {\r\n            self._isDisabled = true;\r\n            self._wasInvoked = false;\r\n            self._clipboardData = {};\r\n        };\r\n        this.isDisabled = function () {\r\n            return self._isDisabled;\r\n        }\r\n        this.wasInvoked = function () {\r\n            return self._wasInvoked;\r\n        }\r\n    }\r\n\r\n    LayoutEditor.Clipboard = new Clipboard();\r\n\r\n    angular\r\n        .module(\"LayoutEditor\")\r\n        .factory(\"clipboard\", [\r\n            function() {\r\n                return {\r\n                    setData: LayoutEditor.Clipboard.setData,\r\n                    getData: LayoutEditor.Clipboard.getData,\r\n                    disable: LayoutEditor.Clipboard.disable,\r\n                    isDisabled: LayoutEditor.Clipboard.isDisabled,\r\n                    wasInvoked: LayoutEditor.Clipboard.wasInvoked\r\n                };\r\n            }\r\n        ]);\r\n})(LayoutEditor || (LayoutEditor = {}));","angular\r\n    .module(\"LayoutEditor\")\r\n    .factory(\"scopeConfigurator\", [\"$timeout\", \"clipboard\",\r\n        function ($timeout, clipboard) {\r\n            return {\r\n\r\n                configureForElement: function ($scope, $element) {\r\n                \r\n                    $element.find(\".layout-panel\").click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n\r\n                    $element.parent().keydown(function (e) {\r\n                        var handled = false;\r\n                        var resetFocus = false;\r\n                        var element = $scope.element;\r\n                    \r\n                        if (element.editor.isDragging)\r\n                            return;\r\n\r\n                        // If native clipboard support exists, the pseudo-clipboard will have been disabled.\r\n                        if (!clipboard.isDisabled()) {\r\n                            var focusedElement = element.editor.focusedElement;\r\n                            if (!!focusedElement) {\r\n                                // Pseudo clipboard handling for browsers not allowing real clipboard operations.\r\n                                if (e.ctrlKey) {\r\n                                    switch (e.which) {\r\n                                    case 67: // C\r\n                                        focusedElement.copy(clipboard);\r\n                                        break;\r\n                                    case 88: // X\r\n                                        focusedElement.cut(clipboard);\r\n                                        break;\r\n                                    case 86: // V\r\n                                        focusedElement.paste(clipboard);\r\n                                        break;\r\n                                    }\r\n                                }\r\n                            }\r\n                        }\r\n\r\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 46) { // Del\r\n                            $scope.delete(element);\r\n                            handled = true;\r\n                        } else if (!e.ctrlKey && !e.shiftKey && !e.altKey && (e.which == 32 || e.which == 27)) { // Space or Esc\r\n                            $element.find(\".layout-panel-action-properties\").first().click();\r\n                            handled = true;\r\n                        }\r\n\r\n                        if (element.type == \"Content\") { // This is a content element.\r\n                            if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 13) { // Enter\r\n                                $element.find(\".layout-panel-action-edit\").first().click();\r\n                                handled = true;\r\n                            }\r\n                        }\r\n\r\n                        if (!!element.children) { // This is a container.\r\n                            if (!e.ctrlKey && !e.shiftKey && e.altKey && e.which == 40) { // Alt+Down\r\n                                if (element.children.length > 0)\r\n                                    element.children[0].setIsFocused();\r\n                                handled = true;\r\n                            }\r\n\r\n                            if (element.type == \"Column\") { // This is a column.\r\n                                var connectAdjacent = !e.ctrlKey;\r\n                                if (e.which == 37) { // Left\r\n                                    if (e.altKey)\r\n                                        element.expandLeft(connectAdjacent);\r\n                                    if (e.shiftKey)\r\n                                        element.contractRight(connectAdjacent);\r\n                                    handled = true;\r\n                                } else if (e.which == 39) { // Right\r\n                                    if (e.altKey)\r\n                                        element.contractLeft(connectAdjacent);\r\n                                    if (e.shiftKey)\r\n                                        element.expandRight(connectAdjacent);\r\n                                    handled = true;\r\n                                }\r\n                            }\r\n                        }\r\n\r\n                        if (!!element.parent) { // This is a child.\r\n                            if (e.altKey && e.which == 38) { // Alt+Up\r\n                                element.parent.setIsFocused();\r\n                                handled = true;\r\n                            }\r\n\r\n                            if (element.parent.type == \"Row\") { // Parent is a horizontal container.\r\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Left\r\n                                    element.parent.moveFocusPrevChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Right\r\n                                    element.parent.moveFocusNextChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Ctrl+Left\r\n                                    element.moveUp();\r\n                                    resetFocus = true;\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Ctrl+Right\r\n                                    element.moveDown();\r\n                                    handled = true;\r\n                                }\r\n                            }\r\n                            else { // Parent is a vertical container.\r\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Up\r\n                                    element.parent.moveFocusPrevChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Down\r\n                                    element.parent.moveFocusNextChild(element);\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Ctrl+Up\r\n                                    element.moveUp();\r\n                                    resetFocus = true;\r\n                                    handled = true;\r\n                                }\r\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Ctrl+Down\r\n                                    element.moveDown();\r\n                                    handled = true;\r\n                                }\r\n                            }\r\n                        }\r\n\r\n                        if (handled) {\r\n                            e.preventDefault();\r\n                        }\r\n\r\n                        e.stopPropagation();\r\n\r\n                        $scope.$apply(); // Event is not triggered by Angular directive but raw event handler on element.\r\n\r\n                        // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\r\n                        if (resetFocus) {\r\n                            window.setTimeout(function () {\r\n                                $scope.$apply(function () {\r\n                                    element.editor.focusedElement.setIsFocused();\r\n                                });\r\n                            }, 100);\r\n                        }\r\n                    });\r\n\r\n                    $scope.element.setIsFocusedEventHandlers.push(function () {\r\n                        $element.parent().focus();\r\n                    });\r\n\r\n                    $scope.delete = function (element) {\r\n                        element.delete();\r\n                    }\r\n                },\r\n\r\n                configureForContainer: function ($scope, $element) {\r\n                    var element = $scope.element;\r\n\r\n                    //$scope.isReceiving = false; // True when container is receiving an external element via drag/drop.\r\n                    $scope.getShowChildrenPlaceholder = function () {\r\n                        return $scope.element.children.length === 0 && !$scope.element.getIsDropTarget();\r\n                    };\r\n\r\n                    $scope.sortableOptions = {\r\n                        cursor: \"move\",\r\n                        delay: 150,\r\n                        disabled: element.getIsSealed(),\r\n                        distance: 5,\r\n                        //handle: element.children.length < 2 ? \".imaginary-class\" : false, // For some reason doesn't get re-evaluated after adding more children.\r\n                        start: function (e, ui) {\r\n                            $scope.$apply(function () {\r\n                                element.setIsDropTarget(true);\r\n                                element.editor.isDragging = true;\r\n                            });\r\n                            // Make the drop target placeholder as high as the item being dragged.\r\n                            ui.placeholder.height(ui.item.height() - 4);\r\n                            ui.placeholder.css(\"min-height\", 0);\r\n                        },\r\n                        stop: function (e, ui) {\r\n                            $scope.$apply(function () {\r\n                                element.editor.isDragging = false;\r\n                                element.setIsDropTarget(false);\r\n                            });\r\n                        },\r\n                        over: function (e, ui) {\r\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\r\n                                if (!!ui.sender[0].dropTargetTimeout) {\r\n                                    $timeout.cancel(ui.sender[0].dropTargetTimeout);\r\n                                    ui.sender[0].dropTargetTimeout = null;\r\n                                }\r\n                                $timeout(function () {\r\n                                    if (element.type == \"Row\") {\r\n                                        // If there was a previous drop target and it was a row, roll back any pending column adds to it.\r\n                                        var previousDropTarget = element.editor.dropTargetElement;\r\n                                        if (!!previousDropTarget && previousDropTarget.type == \"Row\")\r\n                                            previousDropTarget.rollbackAddColumn();\r\n                                    }\r\n                                    element.setIsDropTarget(false);\r\n                                });\r\n                                ui.sender[0].dropTargetTimeout = $timeout(function () {\r\n                                    if (element.type == \"Row\") {\r\n                                        var receivedColumn = ui.item.sortable.model;\r\n                                        var receivedColumnWidth = Math.floor(12 / (element.children.length + 1));\r\n                                        receivedColumn.width = receivedColumnWidth;\r\n                                        receivedColumn.offset = 0;\r\n                                        element.beginAddColumn(receivedColumnWidth);\r\n                                        // Make the drop target placeholder the correct width and as high as the highest existing column in the row.\r\n                                        var maxHeight = _.max(_($element.find(\"> .layout-children > .layout-column:not(.ui-sortable-placeholder)\")).map(function (e) {\r\n                                            return $(e).height();\r\n                                        }));\r\n                                        for (i = 1; i <= 12; i++)\r\n                                            ui.placeholder.removeClass(\"col-xs-\" + i);\r\n                                        ui.placeholder.addClass(\"col-xs-\" + receivedColumn.width);\r\n                                        if (maxHeight > 0) {\r\n                                            ui.placeholder.height(maxHeight);\r\n                                            ui.placeholder.css(\"min-height\", 0);\r\n                                        }\r\n                                        else {\r\n                                            ui.placeholder.height(0);\r\n                                            ui.placeholder.css(\"min-height\", \"\");\r\n                                        }\r\n                                    }\r\n                                    element.setIsDropTarget(true);\r\n                                }, 150);\r\n                            }\r\n                        },\r\n                        receive: function (e, ui) {\r\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\r\n                                $scope.$apply(function () {\r\n                                    var receivedElement = ui.item.sortable.model;\r\n                                    if (!!receivedElement) {\r\n                                        if (element.type == \"Row\")\r\n                                            element.commitAddColumn();\r\n                                        // Should ideally call LayoutEditor.Container.addChild() instead, but since this handler\r\n                                        // is run *before* the ui-sortable directive's handler, if we try to add the child to the\r\n                                        // array that handler will get an exception when trying to do the same.\r\n                                        // Because of this, we need to invoke \"setParent\" so that specific container types can perform element speficic initialization.\r\n                                        receivedElement.setEditor(element.editor);\r\n                                        receivedElement.setParent(element);\r\n\r\n                                        if (!!receivedElement.hasEditor) {\r\n                                            $scope.$root.editElement(receivedElement).then(function (args) {\r\n                                                if (!args.cancel) {\r\n                                                    receivedElement.data = args.element.data;\r\n                                                    receivedElement.applyElementEditorModel(args.elementEditorModel);\r\n\r\n                                                    if (!!receivedElement.setHtml)\r\n                                                        receivedElement.setHtml(args.element.html);\r\n                                                }\r\n                                                $timeout(function () {\r\n                                                    if (!!args.cancel)\r\n                                                        receivedElement.delete();\r\n                                                    else\r\n                                                        receivedElement.setIsFocused();\r\n                                                    //$scope.isReceiving = false;\r\n                                                    element.setIsDropTarget(false);\r\n\r\n                                                });\r\n                                                return;\r\n                                            });\r\n                                        }\r\n                                    }\r\n                                    $timeout(function () {\r\n                                        //$scope.isReceiving = false;\r\n                                        element.setIsDropTarget(false);\r\n                                        if (!!receivedElement)\r\n                                            receivedElement.setIsFocused();\r\n                                    });\r\n                                });\r\n                            }\r\n                        }\r\n                    };\r\n\r\n                    $scope.click = function (child, e) {\r\n                        if (!child.editor.isDragging)\r\n                            child.setIsFocused();\r\n                        e.stopPropagation();\r\n                    };\r\n\r\n                    $scope.getClasses = function (child) {\r\n                        var result = [\"layout-element\"];\r\n\r\n                        if (!!child.children) {\r\n                            result.push(\"layout-container\");\r\n                            if (child.getIsSealed())\r\n                                result.push(\"layout-container-sealed\");\r\n                        }\r\n\r\n                        result.push(\"layout-\" + child.type.toLowerCase());\r\n\r\n                        if (!!child.dropTargetClass)\r\n                            result.push(child.dropTargetClass);\r\n\r\n                        // TODO: Move these to either the Column directive or the Column model class.\r\n                        if (child.type == \"Row\") {\r\n                            result.push(\"row\");\r\n                            if (!child.canAddColumn())\r\n                                result.push(\"layout-row-full\");\r\n                        }\r\n                        if (child.type == \"Column\") {\r\n                            result.push(\"col-xs-\" + child.width);\r\n                            result.push(\"col-xs-offset-\" + child.offset);\r\n                        }\r\n                        if (child.type == \"Content\")\r\n                            result.push(\"layout-content-\" + child.contentTypeClass);\r\n\r\n                        if (child.getIsActive())\r\n                            result.push(\"layout-element-active\");\r\n                        if (child.getIsFocused())\r\n                            result.push(\"layout-element-focused\");\r\n                        if (child.getIsSelected())\r\n                            result.push(\"layout-element-selected\");\r\n                        if (child.getIsDropTarget())\r\n                            result.push(\"layout-element-droptarget\");\r\n                        if (child.isTemplated)\r\n                            result.push(\"layout-element-templated\");\r\n\r\n                        return result;\r\n                    };\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutEditor\", [\"environment\",\r\n        function (environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: {},\r\n                controller: [\"$scope\", \"$element\", \"$attrs\", \"$compile\", \"clipboard\",\r\n                    function ($scope, $element, $attrs, $compile, clipboard) {\r\n                        if (!!$attrs.model)\r\n                            $scope.element = eval($attrs.model);\r\n                        else\r\n                            throw new Error(\"The 'model' attribute must evaluate to a LayoutEditor.Editor object.\");\r\n\r\n                        $scope.click = function (canvas, e) {\r\n                            if (!canvas.editor.isDragging)\r\n                                canvas.setIsFocused();\r\n                            e.stopPropagation();\r\n                        };\r\n\r\n                        $scope.getClasses = function (canvas) {\r\n                            var result = [\"layout-element\", \"layout-container\", \"layout-canvas\"];\r\n\r\n                            if (canvas.getIsActive())\r\n                                result.push(\"layout-element-active\");\r\n                            if (canvas.getIsFocused())\r\n                                result.push(\"layout-element-focused\");\r\n                            if (canvas.getIsSelected())\r\n                                result.push(\"layout-element-selected\");\r\n                            if (canvas.getIsDropTarget())\r\n                                result.push(\"layout-element-droptarget\");\r\n                            if (canvas.isTemplated)\r\n                                result.push(\"layout-element-templated\");\r\n\r\n                            return result;\r\n                        };\r\n\r\n                        // An unfortunate side-effect of the next hack on line 54 is that the created elements aren't added to the DOM yet, so we can't use it to get to the parent \".layout-desiger\" element.\r\n                        // Work around: access that element directly (which efectively turns multiple layout editors on a single page impossible). \r\n                        // //var layoutDesignerHost = $element.closest(\".layout-designer\").data(\"layout-designer-host\");\r\n                        var layoutDesignerHost = $(\".layout-designer\").data(\"layout-designer-host\");\r\n\r\n                        $scope.$root.layoutDesignerHost = layoutDesignerHost;\r\n\r\n                        layoutDesignerHost.element.on(\"replacecanvas\", function (e, args) {\r\n                            var editor = $scope.element;\r\n                            var canvasData = {\r\n                                data: args.canvas.data,\r\n                                htmlId: args.canvas.htmlId,\r\n                                htmlClass: args.canvas.htmlClass,\r\n                                htmlStyle: args.canvas.htmlStyle,\r\n                                isTemplated: args.canvas.isTemplated,\r\n                                children: args.canvas.children\r\n                            };\r\n\r\n                            // HACK: Instead of simply updating the $scope.element with a new instance, we need to replace the entire orc-layout-editor markup\r\n                            // in order for angular to rebind starting with the Canvas element. Otherwise, for some reason, it will rebind starting with the first child of Canvas.\r\n                            // You can see this happening when setting a breakpoint in ScopeConfigurator where containers are initialized with drag & drop: on page load, the first element\r\n                            // is a Canvas (good), but after having selected another template, the first element is (typically) a Grid (bad).\r\n                            // Simply recompiling the orc-layout-editor directive will cause the entire thing to be generated, which works just fine as well (even though not is nice as simply leveraging model binding).\r\n                            layoutDesignerHost.editor = window.layoutEditor = new LayoutEditor.Editor(editor.config, canvasData);\r\n                            var template = \"<orc-layout-editor\" + \" model='window.layoutEditor' />\";\r\n                            var html = $compile(template)($scope);\r\n                            $(\".layout-editor-holder\").html(html);\r\n                        });\r\n\r\n                        $scope.$root.editElement = function (element) {\r\n                            var host = $scope.$root.layoutDesignerHost;\r\n                            return host.editElement(element);\r\n                        };\r\n\r\n                        $scope.$root.addElement = function (contentType) {\r\n                            var host = $scope.$root.layoutDesignerHost;\r\n                            return host.addElement(contentType);\r\n                        };\r\n\r\n                        $(document).on(\"cut copy paste\", function (e) {\r\n                            // If the pseudo clipboard was already invoked (which happens on the first clipboard\r\n                            // operation after page load even if native clipboard support exists) then sit this\r\n                            // one operation out, but make sure whatever is on the pseudo clipboard gets migrated\r\n                            // to the native clipboard for subsequent operations.\r\n                            if (clipboard.wasInvoked()) {\r\n                                e.originalEvent.clipboardData.setData(\"text/plain\", clipboard.getData(\"text/plain\"));\r\n                                e.originalEvent.clipboardData.setData(\"text/json\", clipboard.getData(\"text/json\"));\r\n                                e.preventDefault();\r\n                            }\r\n                            else {\r\n                                var focusedElement = $scope.element.focusedElement;\r\n                                if (!!focusedElement) {\r\n                                    $scope.$apply(function () {\r\n                                        switch (e.type) {\r\n                                            case \"copy\":\r\n                                                focusedElement.copy(e.originalEvent.clipboardData);\r\n                                                break;\r\n                                            case \"cut\":\r\n                                                focusedElement.cut(e.originalEvent.clipboardData);\r\n                                                break;\r\n                                            case \"paste\":\r\n                                                focusedElement.paste(e.originalEvent.clipboardData);\r\n                                                break;\r\n                                        }\r\n                                    });\r\n\r\n                                    // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\r\n                                    window.setTimeout(function () {\r\n                                        $scope.$apply(function () {\r\n                                            if (!!$scope.element.focusedElement)\r\n                                                $scope.element.focusedElement.setIsFocused();\r\n                                        });\r\n                                    }, 100);\r\n\r\n                                    e.preventDefault();\r\n                                }\r\n                            }\r\n\r\n                            // Native clipboard support obviously exists, so disable the peudo clipboard from now on.\r\n                            clipboard.disable();\r\n                        });\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Editor\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                    // No clicks should propagate from the TinyMCE toolbars.\r\n                    element.find(\".layout-toolbar-container\").click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n                    // Unfocus and unselect everything on click outside of canvas.\r\n                    $(window).click(function (e) {\r\n                        scope.$apply(function () {\r\n                            scope.element.activeElement = null;\r\n                            scope.element.focusedElement = null;\r\n                        });\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutCanvas\", [\"scopeConfigurator\", \"environment\",\r\n        function (scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\", \"$attrs\",\r\n                    function ($scope, $element, $attrs) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"y\";\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Canvas\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutChild\", [\"$compile\",\r\n        function ($compile) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                link: function (scope, element) {\r\n                    var template = \"<orc-layout-\" + scope.element.type.toLowerCase() + \" element='element' />\";\r\n                    var html = $compile(template)(scope);\r\n                    $(element).replaceWith(html);\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutColumn\", [\"$compile\", \"scopeConfigurator\", \"environment\",\r\n        function ($compile, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"y\";\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Column\"),\r\n                replace: true,\r\n                link: function (scope, element, attrs) {\r\n                    element.find(\".layout-column-resize-bar\").draggable({\r\n                        axis: \"x\",\r\n                        helper: \"clone\",\r\n                        revert: true,\r\n                        start: function (e, ui) {\r\n                            scope.$apply(function () {\r\n                                scope.element.editor.isResizing = true;\r\n                            });\r\n                        },\r\n                        drag: function (e, ui) {\r\n                            var columnElement = element.parent();\r\n                            var columnSize = columnElement.width() / scope.element.width;\r\n                            var connectAdjacent = !e.ctrlKey;\r\n                            if ($(e.target).hasClass(\"layout-column-resize-bar-left\")) {\r\n                                var delta = ui.offset.left - columnElement.offset().left;\r\n                                if (delta < -columnSize && scope.element.canExpandLeft(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.expandLeft(connectAdjacent);\r\n                                    });\r\n                                }\r\n                                else if (delta > columnSize && scope.element.canContractLeft(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.contractLeft(connectAdjacent);\r\n                                    });\r\n                                }\r\n                            }\r\n                            else if ($(e.target).hasClass(\"layout-column-resize-bar-right\")) {\r\n                                var delta = ui.offset.left - columnElement.width() - columnElement.offset().left;\r\n                                if (delta > columnSize && scope.element.canExpandRight(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.expandRight(connectAdjacent);\r\n                                    });\r\n                                }\r\n                                else if (delta < -columnSize && scope.element.canContractRight(connectAdjacent)) {\r\n                                    scope.$apply(function () {\r\n                                        scope.element.contractRight(connectAdjacent);\r\n                                    });\r\n                                }\r\n                            }\r\n\r\n                        },\r\n                        stop: function (e, ui) {\r\n                            scope.$apply(function () {\r\n                              scope.element.editor.isResizing = false;\r\n                            });\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutContent\", [\"$sce\", \"scopeConfigurator\", \"environment\",\r\n        function ($sce, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        $scope.edit = function () {\r\n                            $scope.$root.editElement($scope.element).then(function (args) {\r\n                                $scope.$apply(function () {\r\n                                    if (args.cancel)\r\n                                        return;\r\n\r\n                                    $scope.element.data = args.element.data;\r\n                                    $scope.element.setHtml(args.element.html);\r\n                                });\r\n                            });\r\n                        };\r\n\r\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\r\n                        $scope.element.setHtml = function (html) {\r\n                            $scope.element.html = html;\r\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\r\n                        };\r\n\r\n                        $scope.element.setHtml($scope.element.html);\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Content\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutHtml\", [\"$sce\", \"scopeConfigurator\", \"environment\",\r\n        function ($sce, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        $scope.edit = function () {\r\n                            $scope.$root.editElement($scope.element).then(function (args) {\r\n                                $scope.$apply(function () {\r\n                                    if (args.cancel)\r\n                                        return;\r\n\r\n                                    $scope.element.data = args.element.data;\r\n                                    $scope.element.setHtml(args.element.html);\r\n                                });\r\n                            });\r\n                        };\r\n                        $scope.updateContent = function (e) {\r\n                            $scope.element.setHtml(e.target.innerHTML);\r\n                        };\r\n\r\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\r\n                        $scope.element.setHtml = function (html) {\r\n                            $scope.element.html = html;\r\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\r\n                        };\r\n\r\n                        $scope.element.setHtml($scope.element.html);\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Html\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutGrid\", [\"$compile\", \"scopeConfigurator\", \"environment\",\r\n        function ($compile, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"y\";\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Grid\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutRow\", [\"$compile\", \"scopeConfigurator\", \"environment\",\r\n        function ($compile, scopeConfigurator, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { element: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        scopeConfigurator.configureForElement($scope, $element);\r\n                        scopeConfigurator.configureForContainer($scope, $element);\r\n                        $scope.sortableOptions[\"axis\"] = \"x\";\r\n                        $scope.sortableOptions[\"ui-floating\"] = true;\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Row\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutPopup\", [\r\n        function () {\r\n            return {\r\n                restrict: \"A\",\r\n                link: function (scope, element, attrs) {\r\n                    var popup = $(element);\r\n                    var trigger = popup.closest(\".layout-popup-trigger\");\r\n                    var parentElement = popup.closest(\".layout-element\");\r\n                    trigger.click(function () {\r\n                        popup.toggle();\r\n                        if (popup.is(\":visible\")) {\r\n                            popup.position({\r\n                                my: attrs.orcLayoutPopupMy || \"left top\",\r\n                                at: attrs.orcLayoutPopupAt || \"left bottom+4px\",\r\n                                of: trigger\r\n                            });\r\n                            popup.find(\"input\").first().focus();\r\n                        }\r\n                    });\r\n                    popup.click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n                    parentElement.click(function (e) {\r\n                        popup.hide();\r\n                    });\r\n                    popup.keydown(function (e) {\r\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 27) // Esc\r\n                            popup.hide();\r\n                        e.stopPropagation();\r\n                    });\r\n                    popup.on(\"cut copy paste\", function (e) {\r\n                        // Allow clipboard operations in popup without invoking clipboard event handlers on parent element.\r\n                        e.stopPropagation();\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutToolbox\", [\"$compile\", \"environment\",\r\n        function ($compile, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n\r\n                        $scope.resetElements = function () {\r\n\r\n                            $scope.gridElements = [\r\n                                LayoutEditor.Grid.from({\r\n                                    toolboxIcon: \"\\uf00a\",\r\n                                    toolboxLabel: \"Grid\",\r\n                                    toolboxDescription: \"Empty grid.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.rowElements = [\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (1 column)\",\r\n                                    toolboxDescription: \"Row with 1 column.\",\r\n                                    children: LayoutEditor.Column.times(1)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (2 columns)\",\r\n                                    toolboxDescription: \"Row with 2 columns.\",\r\n                                    children: LayoutEditor.Column.times(2)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (3 columns)\",\r\n                                    toolboxDescription: \"Row with 3 columns.\",\r\n                                    children: LayoutEditor.Column.times(3)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (4 columns)\",\r\n                                    toolboxDescription: \"Row with 4 columns.\",\r\n                                    children: LayoutEditor.Column.times(4)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (6 columns)\",\r\n                                    toolboxDescription: \"Row with 6 columns.\",\r\n                                    children: LayoutEditor.Column.times(6)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (12 columns)\",\r\n                                    toolboxDescription: \"Row with 12 columns.\",\r\n                                    children: LayoutEditor.Column.times(12)\r\n                                }), LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (empty)\",\r\n                                    toolboxDescription: \"Empty row.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.columnElements = [\r\n                                LayoutEditor.Column.from({\r\n                                    toolboxIcon: \"\\uf0db\",\r\n                                    toolboxLabel: \"Column\",\r\n                                    toolboxDescription: \"Empty column.\",\r\n                                    width: 1,\r\n                                    offset: 0,\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.canvasElements = [\r\n                                LayoutEditor.Canvas.from({\r\n                                    toolboxIcon: \"\\uf044\",\r\n                                    toolboxLabel: \"Canvas\",\r\n                                    toolboxDescription: \"Empty canvas.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.contentElementCategories = _($scope.element.config.categories).map(function (category) {\r\n                                return {\r\n                                    name: category.name,\r\n                                    elements: _(category.contentTypes).map(function (contentType) {\r\n                                        var type = contentType.type;\r\n                                        var factory = LayoutEditor.factories[type] || LayoutEditor.factories[\"Content\"];\r\n                                        var item = {\r\n                                            isTemplated: false,\r\n                                            contentType: contentType.id,\r\n                                            contentTypeLabel: contentType.label,\r\n                                            contentTypeClass: contentType.typeClass,\r\n                                            data: null,\r\n                                            hasEditor: contentType.hasEditor,\r\n                                            html: contentType.html\r\n                                        };\r\n                                        var element = factory(item);\r\n                                        element.toolboxIcon = contentType.icon || \"\\uf1c9\";\r\n                                        element.toolboxLabel = contentType.label;\r\n                                        element.toolboxDescription = contentType.description;\r\n                                        return element;\r\n                                    })\r\n                                };\r\n                            });\r\n\r\n                        };\r\n\r\n                        $scope.resetElements();\r\n\r\n                        $scope.getSortableOptions = function (type) {\r\n                            var editorId = $element.closest(\".layout-editor\").attr(\"id\");\r\n                            var parentClasses;\r\n                            var placeholderClasses;\r\n                            var floating = false;\r\n\r\n                            switch (type) {\r\n                                case \"Grid\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-grid ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Row\":\r\n                                    parentClasses = [\".layout-grid\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-row row ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Column\":\r\n                                    parentClasses = [\".layout-row:not(.layout-row-full)\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-column ui-sortable-placeholder\";\r\n                                    floating = true; // To ensure a smooth horizontal-list reordering. https://github.com/angular-ui/ui-sortable#floating\r\n                                    break;\r\n                                case \"Content\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-content ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Canvas\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-grid ui-sortable-placeholder\";\r\n                                    break;\r\n                            }\r\n\r\n                            return {\r\n                                cursor: \"move\",\r\n                                connectWith: _(parentClasses).map(function (e) { return \"#\" + editorId + \" \" + e + \":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children\"; }).join(\", \"),\r\n                                placeholder: placeholderClasses,\r\n                                \"ui-floating\": floating,\r\n                                create: function (e, ui) {\r\n                                    e.target.isToolbox = true; // Will indicate to connected sortables that dropped items were sent from toolbox.\r\n                                },\r\n                                start: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = true;\r\n                                    });\r\n                                },\r\n                                stop: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = false;\r\n                                        $scope.resetElements();\r\n                                    });\r\n                                },\r\n                                over: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.canvas.setIsDropTarget(false);\r\n                                    });\r\n                                },\r\n                            }\r\n                        };\r\n\r\n                        var layoutIsCollapsedCookieName = \"layoutToolboxCategory_Layout_IsCollapsed\";\r\n                        $scope.layoutIsCollapsed = $.cookie(layoutIsCollapsedCookieName) === \"true\";\r\n\r\n                        $scope.toggleLayoutIsCollapsed = function (e) {\r\n                            $scope.layoutIsCollapsed = !$scope.layoutIsCollapsed;\r\n                            $.cookie(layoutIsCollapsedCookieName, $scope.layoutIsCollapsed, { expires: 365 }); // Remember collapsed state for a year.\r\n                            e.preventDefault();\r\n                            e.stopPropagation();\r\n                        };\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Toolbox\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                    var toolbox = element.find(\".layout-toolbox\");\r\n                    $(window).on(\"resize scroll\", function (e) {\r\n                        var canvas = element.parent().find(\".layout-canvas\");\r\n                        // If the canvas is taller than the toolbox, make the toolbox sticky-positioned within the editor\r\n                        // to help the user avoid excessive vertical scrolling.\r\n                        var canvasIsTaller = !!canvas && canvas.height() > toolbox.height();\r\n                        var windowPos = $(window).scrollTop();\r\n                        if (canvasIsTaller && windowPos > element.offset().top + element.height() - toolbox.height()) {\r\n                            toolbox.addClass(\"sticky-bottom\");\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                        }\r\n                        else if (canvasIsTaller && windowPos > element.offset().top) {\r\n                            toolbox.addClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                        else {\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutToolboxGroup\", [\"$compile\", \"environment\",\r\n        function ($compile, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                scope: { category: \"=\" },\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n                        var isCollapsedCookieName = \"layoutToolboxCategory_\" + $scope.category.name + \"_IsCollapsed\";\r\n                        $scope.isCollapsed = $.cookie(isCollapsedCookieName) === \"true\";\r\n                        $scope.toggleIsCollapsed = function (e) {\r\n                            $scope.isCollapsed = !$scope.isCollapsed;\r\n                            $.cookie(isCollapsedCookieName, $scope.isCollapsed, { expires: 365 }); // Remember collapsed state for a year.\r\n                            e.preventDefault();\r\n                            e.stopPropagation();\r\n                        };\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"ToolboxGroup\"),\r\n                replace: true\r\n            };\r\n        }\r\n    ]);"],"sourceRoot":"/source/"} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js index d92fdf698..b31866784 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js @@ -1 +1 @@ -angular.module("LayoutEditor",["ngSanitize","ngResource","ui.sortable"]);var LayoutEditor;!function(e){var t=function(){var e=this;this._clipboardData={},this._isDisabled=!1,this._wasInvoked=!1,this.setData=function(t,o){e._clipboardData[t]=o,e._wasInvoked=!0},this.getData=function(t){return e._clipboardData[t]},this.disable=function(){e._isDisabled=!0,e._wasInvoked=!1,e._clipboardData={}},this.isDisabled=function(){return e._isDisabled},this.wasInvoked=function(){return e._wasInvoked}};e.Clipboard=new t,angular.module("LayoutEditor").factory("clipboard",[function(){return{setData:e.Clipboard.setData,getData:e.Clipboard.getData,disable:e.Clipboard.disable,isDisabled:e.Clipboard.isDisabled,wasInvoked:e.Clipboard.wasInvoked}}])}(LayoutEditor||(LayoutEditor={})),angular.module("LayoutEditor").factory("scopeConfigurator",["$timeout","clipboard",function(e,t){return{configureForElement:function(e,o){o.find(".layout-panel").click(function(e){e.stopPropagation()}),o.parent().keydown(function(n){var l=!1,a=!1,i=e.element;if(!i.editor.isDragging&&!i.editor.inlineEditingIsActive){if(!t.isDisabled()){var r=i.editor.focusedElement;if(r&&n.ctrlKey)switch(n.which){case 67:r.copy(t);break;case 88:r.cut(t);break;case 86:r.paste(t)}}if(n.ctrlKey||n.shiftKey||n.altKey||46!=n.which?n.ctrlKey||n.shiftKey||n.altKey||32!=n.which&&27!=n.which||(o.find(".layout-panel-action-properties").first().click(),l=!0):(e["delete"](i),l=!0),"Content"==i.type&&(n.ctrlKey||n.shiftKey||n.altKey||13!=n.which||(o.find(".layout-panel-action-edit").first().click(),l=!0)),i.children&&(n.ctrlKey||n.shiftKey||!n.altKey||40!=n.which||(i.children.length>0&&i.children[0].setIsFocused(),l=!0),"Column"==i.type)){var c=!n.ctrlKey;37==n.which?(n.altKey&&i.expandLeft(c),n.shiftKey&&i.contractRight(c),l=!0):39==n.which&&(n.altKey&&i.contractLeft(c),n.shiftKey&&i.expandRight(c),l=!0)}i.parent&&(n.altKey&&38==n.which&&(i.parent.setIsFocused(),l=!0),"Row"==i.parent.type?n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?n.ctrlKey||n.shiftKey||n.altKey||39!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||39!=n.which||(i.moveDown(),l=!0):(i.moveUp(),a=!0,l=!0):(i.parent.moveFocusNextChild(i),l=!0):(i.parent.moveFocusPrevChild(i),l=!0):n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?n.ctrlKey||n.shiftKey||n.altKey||40!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||40!=n.which||(i.moveDown(),l=!0):(i.moveUp(),a=!0,l=!0):(i.parent.moveFocusNextChild(i),l=!0):(i.parent.moveFocusPrevChild(i),l=!0)),l&&n.preventDefault(),n.stopPropagation(),e.$apply(),a&&window.setTimeout(function(){e.$apply(function(){i.editor.focusedElement.setIsFocused()})},100)}}),e.element.setIsFocusedEventHandlers.push(function(){o.parent().focus()}),e["delete"]=function(e){e["delete"]()}},configureForContainer:function(t,o){var n=t.element;t.getShowChildrenPlaceholder=function(){return 0===t.element.children.length&&!t.element.getIsDropTarget()},t.sortableOptions={cursor:"move",delay:150,disabled:n.getIsSealed(),distance:5,start:function(e,o){t.$apply(function(){n.setIsDropTarget(!0),n.editor.isDragging=!0}),o.placeholder.height(o.item.height()-4),o.placeholder.css("min-height",0)},stop:function(e,o){t.$apply(function(){n.editor.isDragging=!1,n.setIsDropTarget(!1)})},over:function(t,l){l.sender&&l.sender[0].isToolbox&&(l.sender[0].dropTargetTimeout&&(e.cancel(l.sender[0].dropTargetTimeout),l.sender[0].dropTargetTimeout=null),e(function(){if("Row"==n.type){var e=n.editor.dropTargetElement;e&&"Row"==e.type&&e.rollbackAddColumn()}n.setIsDropTarget(!1)}),l.sender[0].dropTargetTimeout=e(function(){if("Row"==n.type){var e=l.item.sortable.model,t=Math.floor(12/(n.children.length+1));e.width=t,e.offset=0,n.beginAddColumn(t);var a=_.max(_(o.find("> .layout-children > .layout-column:not(.ui-sortable-placeholder)")).map(function(e){return $(e).height()}));for(i=1;i<=12;i++)l.placeholder.removeClass("col-xs-"+i);l.placeholder.addClass("col-xs-"+e.width),a>0?(l.placeholder.height(a),l.placeholder.css("min-height",0)):(l.placeholder.height(0),l.placeholder.css("min-height",""))}n.setIsDropTarget(!0)},150))},receive:function(o,l){l.sender&&l.sender[0].isToolbox&&t.$apply(function(){var o=l.item.sortable.model;o&&("Row"==n.type&&n.commitAddColumn(),o.setEditor(n.editor),o.setParent(n),o.hasEditor&&t.$root.editElement(o).then(function(t){t.cancel||(o.data=t.element.data,o.applyElementEditorModel(t.elementEditorModel),o.setHtml&&o.setHtml(t.element.html)),e(function(){t.cancel?o["delete"]():o.setIsFocused(),n.setIsDropTarget(!1)})})),e(function(){n.setIsDropTarget(!1),o&&o.setIsFocused()})})}},t.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},t.getClasses=function(e){var t=["layout-element"];return e.children&&(t.push("layout-container"),e.getIsSealed()&&t.push("layout-container-sealed")),t.push("layout-"+e.type.toLowerCase()),e.dropTargetClass&&t.push(e.dropTargetClass),"Row"==e.type&&(t.push("row"),e.canAddColumn()||t.push("layout-row-full")),"Column"==e.type&&(t.push("col-xs-"+e.width),t.push("col-xs-offset-"+e.offset)),"Content"==e.type&&t.push("layout-content-"+e.contentTypeClass),e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t}}}}]),angular.module("LayoutEditor").directive("orcLayoutEditor",["environment",function(environment){return{restrict:"E",scope:{},controller:["$scope","$element","$attrs","$compile","clipboard",function($scope,$element,$attrs,$compile,clipboard){if(!$attrs.model)throw new Error("The 'model' attribute must evaluate to a LayoutEditor.Editor object.");$scope.element=eval($attrs.model),$scope.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},$scope.getClasses=function(e){var t=["layout-element","layout-container","layout-canvas"];return e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t};var layoutDesignerHost=$(".layout-designer").data("layout-designer-host");$scope.$root.layoutDesignerHost=layoutDesignerHost,layoutDesignerHost.element.on("replacecanvas",function(e,t){var o=$scope.element,n={data:t.canvas.data,htmlId:t.canvas.htmlId,htmlClass:t.canvas.htmlClass,htmlStyle:t.canvas.htmlStyle,isTemplated:t.canvas.isTemplated,children:t.canvas.children};layoutDesignerHost.editor=window.layoutEditor=new LayoutEditor.Editor(o.config,n);var l="",a=$compile(l)($scope);$(".layout-editor-holder").html(a)}),$scope.$root.editElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.editElement(e)},$scope.$root.addElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.addElement(e)},$scope.toggleInlineEditing=function(){if($scope.element.inlineEditingIsActive)tinymce.remove("#layout-editor-"+$scope.$id+" .layout-content-markup"),$element.find(".layout-toolbar-container").hide(),$scope.element.inlineEditingIsActive=!1;else{$scope.element.inlineEditingIsActive=!0,$element.find(".layout-toolbar-container").show();var e="#layout-editor-"+$scope.$id+" .layout-html .layout-content-markup[data-templated=false]",t=$(e).first().attr("id");tinymce.init({selector:e,theme:"modern",schema:"html5",plugins:["advlist autolink lists link image charmap print preview hr anchor pagebreak","searchreplace wordcount visualblocks visualchars code fullscreen","insertdatetime media nonbreaking table contextmenu directionality","emoticons template paste textcolor colorpicker textpattern","fullscreen autoresize"],toolbar:"undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close",convert_urls:!1,valid_elements:"*[*]",extended_valid_elements:"script[type|defer|src|language]",statusbar:!1,skin:"orchardlightgray",inline:!0,fixed_toolbar_container:"#layout-editor-"+$scope.$id+" .layout-toolbar-container",init_instance_callback:function(e){e.id==t&&tinymce.execCommand("mceFocus",!1,e.id)}})}},$(document).on("cut copy paste",function(e){if(clipboard.wasInvoked())e.originalEvent.clipboardData.setData("text/plain",clipboard.getData("text/plain")),e.originalEvent.clipboardData.setData("text/json",clipboard.getData("text/json")),e.preventDefault();else{var t=$scope.element.focusedElement;t&&($scope.$apply(function(){switch(e.type){case"copy":t.copy(e.originalEvent.clipboardData);break;case"cut":t.cut(e.originalEvent.clipboardData);break;case"paste":t.paste(e.originalEvent.clipboardData)}}),window.setTimeout(function(){$scope.$apply(function(){$scope.element.focusedElement&&$scope.element.focusedElement.setIsFocused()})},100),e.preventDefault())}clipboard.disable()})}],templateUrl:environment.templateUrl("Editor"),replace:!0,link:function(e,t){t.find(".layout-toolbar-container").click(function(e){e.stopPropagation()}),t.mousedown(function(t){e.element.inlineEditingIsActive&&(t.preventDefault(),t.stopPropagation())}),$(window).click(function(t){e.element.inlineEditingIsActive||e.$apply(function(){e.element.activeElement=null,e.element.focusedElement=null})})}}}]),angular.module("LayoutEditor").directive("orcLayoutCanvas",["scopeConfigurator","environment",function(e,t){return{restrict:"E",scope:{element:"="},controller:["$scope","$element","$attrs",function(t,o,n){e.configureForElement(t,o),e.configureForContainer(t,o),t.sortableOptions.axis="y"}],templateUrl:t.templateUrl("Canvas"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutChild",["$compile",function(e){return{restrict:"E",scope:{element:"="},link:function(t,o){var n="",l=e(n)(t);$(o).replaceWith(l)}}}]),angular.module("LayoutEditor").directive("orcLayoutColumn",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Column"),replace:!0,link:function(e,t,o){t.find(".layout-column-resize-bar").draggable({axis:"x",helper:"clone",revert:!0,start:function(t,o){e.$apply(function(){e.element.editor.isResizing=!0})},drag:function(o,n){var l=t.parent(),a=l.width()/e.element.width,i=!o.ctrlKey;if($(o.target).hasClass("layout-column-resize-bar-left")){var r=n.offset.left-l.offset().left;-a>r&&e.element.canExpandLeft(i)?e.$apply(function(){e.element.expandLeft(i)}):r>a&&e.element.canContractLeft(i)&&e.$apply(function(){e.element.contractLeft(i)})}else if($(o.target).hasClass("layout-column-resize-bar-right")){var r=n.offset.left-l.width()-l.offset().left;r>a&&e.element.canExpandRight(i)?e.$apply(function(){e.element.expandRight(i)}):-a>r&&e.element.canContractRight(i)&&e.$apply(function(){e.element.contractRight(i)})}},stop:function(t,o){e.$apply(function(){e.element.editor.isResizing=!1})}})}}}]),angular.module("LayoutEditor").directive("orcLayoutContent",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Content"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutHtml",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.updateContent=function(e){o.element.setHtml(e.target.innerHTML)},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Html"),replace:!0,link:function(e,t){t.find(".layout-content-markup").mousedown(function(t){e.element.editor.inlineEditingIsActive&&t.stopPropagation()})}}}]),angular.module("LayoutEditor").directive("orcLayoutGrid",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Grid"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutRow",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="x",e.sortableOptions["ui-floating"]=!0}],templateUrl:o.templateUrl("Row"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutPopup",[function(){return{restrict:"A",link:function(e,t,o){var n=$(t),l=n.closest(".layout-popup-trigger"),a=n.closest(".layout-element");l.click(function(){n.toggle(),n.is(":visible")&&(n.position({my:o.orcLayoutPopupMy||"left top",at:o.orcLayoutPopupAt||"left bottom+4px",of:l}),n.find("input").first().focus())}),n.click(function(e){e.stopPropagation()}),a.click(function(e){n.hide()}),n.keydown(function(e){e.ctrlKey||e.shiftKey||e.altKey||27!=e.which||n.hide(),e.stopPropagation()}),n.on("cut copy paste",function(e){e.stopPropagation()})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolbox",["$compile","environment",function(e,t){return{restrict:"E",controller:["$scope","$element",function(e,t){e.resetElements=function(){e.gridElements=[LayoutEditor.Grid.from({toolboxIcon:"",toolboxLabel:"Grid",toolboxDescription:"Empty grid.",children:[]})],e.rowElements=[LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (1 column)",toolboxDescription:"Row with 1 column.",children:LayoutEditor.Column.times(1)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (2 columns)",toolboxDescription:"Row with 2 columns.",children:LayoutEditor.Column.times(2)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (3 columns)",toolboxDescription:"Row with 3 columns.",children:LayoutEditor.Column.times(3)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (4 columns)",toolboxDescription:"Row with 4 columns.",children:LayoutEditor.Column.times(4)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (6 columns)",toolboxDescription:"Row with 6 columns.",children:LayoutEditor.Column.times(6)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (12 columns)",toolboxDescription:"Row with 12 columns.",children:LayoutEditor.Column.times(12)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (empty)",toolboxDescription:"Empty row.",children:[]})],e.columnElements=[LayoutEditor.Column.from({toolboxIcon:"",toolboxLabel:"Column",toolboxDescription:"Empty column.",width:1,offset:0,children:[]})],e.canvasElements=[LayoutEditor.Canvas.from({toolboxIcon:"",toolboxLabel:"Canvas",toolboxDescription:"Empty canvas.",children:[]})],e.contentElementCategories=_(e.element.config.categories).map(function(e){return{name:e.name,elements:_(e.contentTypes).map(function(e){var t=e.type,o=LayoutEditor.factories[t]||LayoutEditor.factories.Content,n={isTemplated:!1,contentType:e.id,contentTypeLabel:e.label,contentTypeClass:e.typeClass,data:null,hasEditor:e.hasEditor,html:e.html},l=o(n);return l.toolboxIcon=e.icon||"",l.toolboxLabel=e.label,l.toolboxDescription=e.description,l})}})},e.resetElements(),e.getSortableOptions=function(o){var n,l,a=t.closest(".layout-editor").attr("id"),i=!1;switch(o){case"Grid":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-container layout-grid ui-sortable-placeholder";break;case"Row":n=[".layout-grid"],l="layout-element layout-container layout-row row ui-sortable-placeholder";break;case"Column":n=[".layout-row:not(.layout-row-full)"],l="layout-element layout-container layout-column ui-sortable-placeholder",i=!0;break;case"Content":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-content ui-sortable-placeholder";break;case"Canvas":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-container layout-grid ui-sortable-placeholder"}return{cursor:"move",connectWith:_(n).map(function(e){return"#"+a+" "+e+":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children"}).join(", "),placeholder:l,"ui-floating":i,create:function(e,t){e.target.isToolbox=!0},start:function(t,o){e.$apply(function(){e.element.isDragging=!0})},stop:function(t,o){e.$apply(function(){e.element.isDragging=!1,e.resetElements()})},over:function(t,o){e.$apply(function(){e.element.canvas.setIsDropTarget(!1)})}}};var o="layoutToolboxCategory_Layout_IsCollapsed";e.layoutIsCollapsed="true"===$.cookie(o),e.toggleLayoutIsCollapsed=function(t){e.layoutIsCollapsed=!e.layoutIsCollapsed,$.cookie(o,e.layoutIsCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("Toolbox"),replace:!0,link:function(e,t){var o=t.find(".layout-toolbox");$(window).on("resize scroll",function(e){var n=t.parent().find(".layout-canvas"),l=!!n&&n.height()>o.height(),a=$(window).scrollTop();l&&a>t.offset().top+t.height()-o.height()?(o.addClass("sticky-bottom"),o.removeClass("sticky-top")):l&&a>t.offset().top?(o.addClass("sticky-top"),o.removeClass("sticky-bottom")):(o.removeClass("sticky-top"),o.removeClass("sticky-bottom"))})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolboxGroup",["$compile","environment",function(e,t){return{restrict:"E",scope:{category:"="},controller:["$scope","$element",function(e,t){var o="layoutToolboxCategory_"+e.category.name+"_IsCollapsed";e.isCollapsed="true"===$.cookie(o),e.toggleIsCollapsed=function(t){e.isCollapsed=!e.isCollapsed,$.cookie(o,e.isCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("ToolboxGroup"),replace:!0}}]); \ No newline at end of file +angular.module("LayoutEditor",["ngSanitize","ngResource","ui.sortable"]);var LayoutEditor;!function(e){var t=function(){var e=this;this._clipboardData={},this._isDisabled=!1,this._wasInvoked=!1,this.setData=function(t,o){e._clipboardData[t]=o,e._wasInvoked=!0},this.getData=function(t){return e._clipboardData[t]},this.disable=function(){e._isDisabled=!0,e._wasInvoked=!1,e._clipboardData={}},this.isDisabled=function(){return e._isDisabled},this.wasInvoked=function(){return e._wasInvoked}};e.Clipboard=new t,angular.module("LayoutEditor").factory("clipboard",[function(){return{setData:e.Clipboard.setData,getData:e.Clipboard.getData,disable:e.Clipboard.disable,isDisabled:e.Clipboard.isDisabled,wasInvoked:e.Clipboard.wasInvoked}}])}(LayoutEditor||(LayoutEditor={})),angular.module("LayoutEditor").factory("scopeConfigurator",["$timeout","clipboard",function(e,t){return{configureForElement:function(e,o){o.find(".layout-panel").click(function(e){e.stopPropagation()}),o.parent().keydown(function(n){var l=!1,a=!1,r=e.element;if(!r.editor.isDragging){if(!t.isDisabled()){var i=r.editor.focusedElement;if(i&&n.ctrlKey)switch(n.which){case 67:i.copy(t);break;case 88:i.cut(t);break;case 86:i.paste(t)}}if(n.ctrlKey||n.shiftKey||n.altKey||46!=n.which?n.ctrlKey||n.shiftKey||n.altKey||32!=n.which&&27!=n.which||(o.find(".layout-panel-action-properties").first().click(),l=!0):(e["delete"](r),l=!0),"Content"==r.type&&(n.ctrlKey||n.shiftKey||n.altKey||13!=n.which||(o.find(".layout-panel-action-edit").first().click(),l=!0)),r.children&&(n.ctrlKey||n.shiftKey||!n.altKey||40!=n.which||(r.children.length>0&&r.children[0].setIsFocused(),l=!0),"Column"==r.type)){var c=!n.ctrlKey;37==n.which?(n.altKey&&r.expandLeft(c),n.shiftKey&&r.contractRight(c),l=!0):39==n.which&&(n.altKey&&r.contractLeft(c),n.shiftKey&&r.expandRight(c),l=!0)}r.parent&&(n.altKey&&38==n.which&&(r.parent.setIsFocused(),l=!0),"Row"==r.parent.type?n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?n.ctrlKey||n.shiftKey||n.altKey||39!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||39!=n.which||(r.moveDown(),l=!0):(r.moveUp(),a=!0,l=!0):(r.parent.moveFocusNextChild(r),l=!0):(r.parent.moveFocusPrevChild(r),l=!0):n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?n.ctrlKey||n.shiftKey||n.altKey||40!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||40!=n.which||(r.moveDown(),l=!0):(r.moveUp(),a=!0,l=!0):(r.parent.moveFocusNextChild(r),l=!0):(r.parent.moveFocusPrevChild(r),l=!0)),l&&n.preventDefault(),n.stopPropagation(),e.$apply(),a&&window.setTimeout(function(){e.$apply(function(){r.editor.focusedElement.setIsFocused()})},100)}}),e.element.setIsFocusedEventHandlers.push(function(){o.parent().focus()}),e["delete"]=function(e){e["delete"]()}},configureForContainer:function(t,o){var n=t.element;t.getShowChildrenPlaceholder=function(){return 0===t.element.children.length&&!t.element.getIsDropTarget()},t.sortableOptions={cursor:"move",delay:150,disabled:n.getIsSealed(),distance:5,start:function(e,o){t.$apply(function(){n.setIsDropTarget(!0),n.editor.isDragging=!0}),o.placeholder.height(o.item.height()-4),o.placeholder.css("min-height",0)},stop:function(e,o){t.$apply(function(){n.editor.isDragging=!1,n.setIsDropTarget(!1)})},over:function(t,l){l.sender&&l.sender[0].isToolbox&&(l.sender[0].dropTargetTimeout&&(e.cancel(l.sender[0].dropTargetTimeout),l.sender[0].dropTargetTimeout=null),e(function(){if("Row"==n.type){var e=n.editor.dropTargetElement;e&&"Row"==e.type&&e.rollbackAddColumn()}n.setIsDropTarget(!1)}),l.sender[0].dropTargetTimeout=e(function(){if("Row"==n.type){var e=l.item.sortable.model,t=Math.floor(12/(n.children.length+1));e.width=t,e.offset=0,n.beginAddColumn(t);var a=_.max(_(o.find("> .layout-children > .layout-column:not(.ui-sortable-placeholder)")).map(function(e){return $(e).height()}));for(i=1;i<=12;i++)l.placeholder.removeClass("col-xs-"+i);l.placeholder.addClass("col-xs-"+e.width),a>0?(l.placeholder.height(a),l.placeholder.css("min-height",0)):(l.placeholder.height(0),l.placeholder.css("min-height",""))}n.setIsDropTarget(!0)},150))},receive:function(o,l){l.sender&&l.sender[0].isToolbox&&t.$apply(function(){var o=l.item.sortable.model;o&&("Row"==n.type&&n.commitAddColumn(),o.setEditor(n.editor),o.setParent(n),o.hasEditor&&t.$root.editElement(o).then(function(t){t.cancel||(o.data=t.element.data,o.applyElementEditorModel(t.elementEditorModel),o.setHtml&&o.setHtml(t.element.html)),e(function(){t.cancel?o["delete"]():o.setIsFocused(),n.setIsDropTarget(!1)})})),e(function(){n.setIsDropTarget(!1),o&&o.setIsFocused()})})}},t.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},t.getClasses=function(e){var t=["layout-element"];return e.children&&(t.push("layout-container"),e.getIsSealed()&&t.push("layout-container-sealed")),t.push("layout-"+e.type.toLowerCase()),e.dropTargetClass&&t.push(e.dropTargetClass),"Row"==e.type&&(t.push("row"),e.canAddColumn()||t.push("layout-row-full")),"Column"==e.type&&(t.push("col-xs-"+e.width),t.push("col-xs-offset-"+e.offset)),"Content"==e.type&&t.push("layout-content-"+e.contentTypeClass),e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t}}}}]),angular.module("LayoutEditor").directive("orcLayoutEditor",["environment",function(environment){return{restrict:"E",scope:{},controller:["$scope","$element","$attrs","$compile","clipboard",function($scope,$element,$attrs,$compile,clipboard){if(!$attrs.model)throw new Error("The 'model' attribute must evaluate to a LayoutEditor.Editor object.");$scope.element=eval($attrs.model),$scope.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},$scope.getClasses=function(e){var t=["layout-element","layout-container","layout-canvas"];return e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t};var layoutDesignerHost=$(".layout-designer").data("layout-designer-host");$scope.$root.layoutDesignerHost=layoutDesignerHost,layoutDesignerHost.element.on("replacecanvas",function(e,t){var o=$scope.element,n={data:t.canvas.data,htmlId:t.canvas.htmlId,htmlClass:t.canvas.htmlClass,htmlStyle:t.canvas.htmlStyle,isTemplated:t.canvas.isTemplated,children:t.canvas.children};layoutDesignerHost.editor=window.layoutEditor=new LayoutEditor.Editor(o.config,n);var l="",a=$compile(l)($scope);$(".layout-editor-holder").html(a)}),$scope.$root.editElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.editElement(e)},$scope.$root.addElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.addElement(e)},$(document).on("cut copy paste",function(e){if(clipboard.wasInvoked())e.originalEvent.clipboardData.setData("text/plain",clipboard.getData("text/plain")),e.originalEvent.clipboardData.setData("text/json",clipboard.getData("text/json")),e.preventDefault();else{var t=$scope.element.focusedElement;t&&($scope.$apply(function(){switch(e.type){case"copy":t.copy(e.originalEvent.clipboardData);break;case"cut":t.cut(e.originalEvent.clipboardData);break;case"paste":t.paste(e.originalEvent.clipboardData)}}),window.setTimeout(function(){$scope.$apply(function(){$scope.element.focusedElement&&$scope.element.focusedElement.setIsFocused()})},100),e.preventDefault())}clipboard.disable()})}],templateUrl:environment.templateUrl("Editor"),replace:!0,link:function(e,t){t.find(".layout-toolbar-container").click(function(e){e.stopPropagation()}),$(window).click(function(t){e.$apply(function(){e.element.activeElement=null,e.element.focusedElement=null})})}}}]),angular.module("LayoutEditor").directive("orcLayoutCanvas",["scopeConfigurator","environment",function(e,t){return{restrict:"E",scope:{element:"="},controller:["$scope","$element","$attrs",function(t,o,n){e.configureForElement(t,o),e.configureForContainer(t,o),t.sortableOptions.axis="y"}],templateUrl:t.templateUrl("Canvas"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutChild",["$compile",function(e){return{restrict:"E",scope:{element:"="},link:function(t,o){var n="",l=e(n)(t);$(o).replaceWith(l)}}}]),angular.module("LayoutEditor").directive("orcLayoutColumn",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Column"),replace:!0,link:function(e,t,o){t.find(".layout-column-resize-bar").draggable({axis:"x",helper:"clone",revert:!0,start:function(t,o){e.$apply(function(){e.element.editor.isResizing=!0})},drag:function(o,n){var l=t.parent(),a=l.width()/e.element.width,r=!o.ctrlKey;if($(o.target).hasClass("layout-column-resize-bar-left")){var i=n.offset.left-l.offset().left;-a>i&&e.element.canExpandLeft(r)?e.$apply(function(){e.element.expandLeft(r)}):i>a&&e.element.canContractLeft(r)&&e.$apply(function(){e.element.contractLeft(r)})}else if($(o.target).hasClass("layout-column-resize-bar-right")){var i=n.offset.left-l.width()-l.offset().left;i>a&&e.element.canExpandRight(r)?e.$apply(function(){e.element.expandRight(r)}):-a>i&&e.element.canContractRight(r)&&e.$apply(function(){e.element.contractRight(r)})}},stop:function(t,o){e.$apply(function(){e.element.editor.isResizing=!1})}})}}}]),angular.module("LayoutEditor").directive("orcLayoutContent",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Content"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutHtml",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.updateContent=function(e){o.element.setHtml(e.target.innerHTML)},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Html"),replace:!0,link:function(e,t){}}}]),angular.module("LayoutEditor").directive("orcLayoutGrid",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Grid"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutRow",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="x",e.sortableOptions["ui-floating"]=!0}],templateUrl:o.templateUrl("Row"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutPopup",[function(){return{restrict:"A",link:function(e,t,o){var n=$(t),l=n.closest(".layout-popup-trigger"),a=n.closest(".layout-element");l.click(function(){n.toggle(),n.is(":visible")&&(n.position({my:o.orcLayoutPopupMy||"left top",at:o.orcLayoutPopupAt||"left bottom+4px",of:l}),n.find("input").first().focus())}),n.click(function(e){e.stopPropagation()}),a.click(function(e){n.hide()}),n.keydown(function(e){e.ctrlKey||e.shiftKey||e.altKey||27!=e.which||n.hide(),e.stopPropagation()}),n.on("cut copy paste",function(e){e.stopPropagation()})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolbox",["$compile","environment",function(e,t){return{restrict:"E",controller:["$scope","$element",function(e,t){e.resetElements=function(){e.gridElements=[LayoutEditor.Grid.from({toolboxIcon:"",toolboxLabel:"Grid",toolboxDescription:"Empty grid.",children:[]})],e.rowElements=[LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (1 column)",toolboxDescription:"Row with 1 column.",children:LayoutEditor.Column.times(1)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (2 columns)",toolboxDescription:"Row with 2 columns.",children:LayoutEditor.Column.times(2)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (3 columns)",toolboxDescription:"Row with 3 columns.",children:LayoutEditor.Column.times(3)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (4 columns)",toolboxDescription:"Row with 4 columns.",children:LayoutEditor.Column.times(4)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (6 columns)",toolboxDescription:"Row with 6 columns.",children:LayoutEditor.Column.times(6)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (12 columns)",toolboxDescription:"Row with 12 columns.",children:LayoutEditor.Column.times(12)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (empty)",toolboxDescription:"Empty row.",children:[]})],e.columnElements=[LayoutEditor.Column.from({toolboxIcon:"",toolboxLabel:"Column",toolboxDescription:"Empty column.",width:1,offset:0,children:[]})],e.canvasElements=[LayoutEditor.Canvas.from({toolboxIcon:"",toolboxLabel:"Canvas",toolboxDescription:"Empty canvas.",children:[]})],e.contentElementCategories=_(e.element.config.categories).map(function(e){return{name:e.name,elements:_(e.contentTypes).map(function(e){var t=e.type,o=LayoutEditor.factories[t]||LayoutEditor.factories.Content,n={isTemplated:!1,contentType:e.id,contentTypeLabel:e.label,contentTypeClass:e.typeClass,data:null,hasEditor:e.hasEditor,html:e.html},l=o(n);return l.toolboxIcon=e.icon||"",l.toolboxLabel=e.label,l.toolboxDescription=e.description,l})}})},e.resetElements(),e.getSortableOptions=function(o){var n,l,a=t.closest(".layout-editor").attr("id"),r=!1;switch(o){case"Grid":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-container layout-grid ui-sortable-placeholder";break;case"Row":n=[".layout-grid"],l="layout-element layout-container layout-row row ui-sortable-placeholder";break;case"Column":n=[".layout-row:not(.layout-row-full)"],l="layout-element layout-container layout-column ui-sortable-placeholder",r=!0;break;case"Content":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-content ui-sortable-placeholder";break;case"Canvas":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-container layout-grid ui-sortable-placeholder"}return{cursor:"move",connectWith:_(n).map(function(e){return"#"+a+" "+e+":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children"}).join(", "),placeholder:l,"ui-floating":r,create:function(e,t){e.target.isToolbox=!0},start:function(t,o){e.$apply(function(){e.element.isDragging=!0})},stop:function(t,o){e.$apply(function(){e.element.isDragging=!1,e.resetElements()})},over:function(t,o){e.$apply(function(){e.element.canvas.setIsDropTarget(!1)})}}};var o="layoutToolboxCategory_Layout_IsCollapsed";e.layoutIsCollapsed="true"===$.cookie(o),e.toggleLayoutIsCollapsed=function(t){e.layoutIsCollapsed=!e.layoutIsCollapsed,$.cookie(o,e.layoutIsCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("Toolbox"),replace:!0,link:function(e,t){var o=t.find(".layout-toolbox");$(window).on("resize scroll",function(e){var n=t.parent().find(".layout-canvas"),l=!!n&&n.height()>o.height(),a=$(window).scrollTop();l&&a>t.offset().top+t.height()-o.height()?(o.addClass("sticky-bottom"),o.removeClass("sticky-top")):l&&a>t.offset().top?(o.addClass("sticky-top"),o.removeClass("sticky-bottom")):(o.removeClass("sticky-top"),o.removeClass("sticky-bottom"))})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolboxGroup",["$compile","environment",function(e,t){return{restrict:"E",scope:{category:"="},controller:["$scope","$element",function(e,t){var o="layoutToolboxCategory_"+e.category.name+"_IsCollapsed";e.isCollapsed="true"===$.cookie(o),e.toggleIsCollapsed=function(t){e.isCollapsed=!e.isCollapsed,$.cookie(o,e.isCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("ToolboxGroup"),replace:!0}}]); \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js index c79c8401e..4e7f4a183 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js @@ -52,7 +52,6 @@ var LayoutEditor; this.focusedElement = null; this.dropTargetElement = null; this.isDragging = false; - this.inlineEditingIsActive = false; this.isResizing = false; this.resetToolboxElements = function () { @@ -133,7 +132,7 @@ var LayoutEditor; this.setIsActive = function (value) { if (!this.editor) return; - if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing) + if (this.editor.isDragging || this.editor.isResizing) return; if (value) @@ -153,7 +152,7 @@ var LayoutEditor; return; if (!this.children && this.isTemplated) return; - if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing) + if (this.editor.isDragging || this.editor.isResizing) return; this.editor.focusedElement = this; @@ -1050,4 +1049,4 @@ var LayoutEditor; LayoutEditor.registerFactory("Html", function(value) { return LayoutEditor.Html.from(value); }); })(jQuery, LayoutEditor || (LayoutEditor = {})); -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Helpers.js","Editor.js","Element.js","Container.js","Canvas.js","Grid.js","Row.js","Column.js","Content.js","Html.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC9IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC5JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"Models.js","sourcesContent":["var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    Array.prototype.move = function (from, to) {\r\n        this.splice(to, 0, this.splice(from, 1)[0]);\r\n    };\r\n\r\n    LayoutEditor.childrenFrom = function(values) {\r\n        return _(values).map(function(value) {\r\n            return LayoutEditor.elementFrom(value);\r\n        });\r\n    };\r\n\r\n    var registerFactory = LayoutEditor.registerFactory = function(type, factory) {\r\n        var factories = LayoutEditor.factories = LayoutEditor.factories || {};\r\n        factories[type] = factory;\r\n    };\r\n\r\n    registerFactory(\"Canvas\", function (value) { return LayoutEditor.Canvas.from(value); });\r\n    registerFactory(\"Grid\", function(value) { return LayoutEditor.Grid.from(value); });\r\n    registerFactory(\"Row\", function(value) { return LayoutEditor.Row.from(value); });\r\n    registerFactory(\"Column\", function(value) { return LayoutEditor.Column.from(value); });\r\n    registerFactory(\"Content\", function(value) { return LayoutEditor.Content.from(value); });\r\n\r\n    LayoutEditor.elementFrom = function (value) {\r\n        var factory = LayoutEditor.factories[value.type];\r\n\r\n        if (!factory)\r\n            throw new Error(\"No element with type \\\"\" + value.type + \"\\\" was found.\");\r\n\r\n        var element = factory(value);\r\n        return element;\r\n    };\r\n\r\n    LayoutEditor.setModel = function (elementSelector, model) {\r\n        $(elementSelector).scope().element = model;\r\n    };\r\n\r\n    LayoutEditor.getModel = function (elementSelector) {\r\n        return $(elementSelector).scope().element;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Editor = function (config, canvasData) {\r\n        this.config = config;\r\n        this.canvas = LayoutEditor.Canvas.from(canvasData);\r\n        this.initialState = JSON.stringify(this.canvas.toObject());\r\n        this.activeElement = null;\r\n        this.focusedElement = null;\r\n        this.dropTargetElement = null;\r\n        this.isDragging = false;\r\n        this.inlineEditingIsActive = false;\r\n        this.isResizing = false;\r\n\r\n        this.resetToolboxElements = function () {\r\n            this.toolboxElements = [\r\n                LayoutEditor.Row.from({\r\n                    children: []\r\n                })\r\n            ];\r\n        };\r\n\r\n        this.isDirty = function() {\r\n            var currentState = JSON.stringify(this.canvas.toObject());\r\n            return this.initialState != currentState;\r\n        };\r\n\r\n        this.resetToolboxElements();\r\n        this.canvas.setEditor(this);\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));\r\n","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Element = function (type, data, htmlId, htmlClass, htmlStyle, isTemplated, rule) {\r\n        if (!type)\r\n            throw new Error(\"Parameter 'type' is required.\");\r\n\r\n        this.type = type;\r\n        this.data = data;\r\n        this.htmlId = htmlId;\r\n        this.htmlClass = htmlClass;\r\n        this.htmlStyle = htmlStyle;\r\n        this.isTemplated = isTemplated;\r\n        this.rule = rule;\r\n\r\n        this.editor = null;\r\n        this.parent = null;\r\n        this.setIsFocusedEventHandlers = [];\r\n\r\n        this.setEditor = function (editor) {\r\n            this.editor = editor;\r\n            if (!!this.children && _.isArray(this.children)) {\r\n                _(this.children).each(function (child) {\r\n                    child.setEditor(editor);\r\n                });\r\n            }\r\n        };\r\n\r\n        this.setParent = function(parentElement) {\r\n            this.parent = parentElement;\r\n            this.parent.onChildAdded(this);\r\n\r\n            var currentAncestor = parentElement;\r\n            while (!!currentAncestor) {\r\n                currentAncestor.onDescendantAdded(this, parentElement);\r\n                currentAncestor = currentAncestor.parent;\r\n            }\r\n        };\r\n\r\n        this.setIsTemplated = function (value) {\r\n            this.isTemplated = value;\r\n            if (!!this.children && _.isArray(this.children)) {\r\n                _(this.children).each(function (child) {\r\n                    child.setIsTemplated(value);\r\n                });\r\n            }\r\n        };\r\n\r\n        this.applyElementEditorModel = function() { /* Virtual */ };\r\n\r\n        this.getIsActive = function () {\r\n            if (!this.editor)\r\n                return false;\r\n            return this.editor.activeElement === this && !this.getIsFocused();\r\n        };\r\n\r\n        this.setIsActive = function (value) {\r\n            if (!this.editor)\r\n                return;\r\n            if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing)\r\n                return;\r\n\r\n            if (value)\r\n                this.editor.activeElement = this;\r\n            else\r\n                this.editor.activeElement = this.parent;\r\n        };\r\n\r\n        this.getIsFocused = function () {\r\n            if (!this.editor)\r\n                return false;\r\n            return this.editor.focusedElement === this;\r\n        };\r\n\r\n        this.setIsFocused = function () {\r\n            if (!this.editor)\r\n            \treturn;\r\n            if (!this.children && this.isTemplated)\r\n            \treturn;\r\n            if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing)\r\n                return;\r\n\r\n            this.editor.focusedElement = this;\r\n            _(this.setIsFocusedEventHandlers).each(function (item) {\r\n                try {\r\n                    item();\r\n                }\r\n                catch (ex) {\r\n                    // Ignore.\r\n                }\r\n            });\r\n        };\r\n\r\n        this.getIsSelected = function () {\r\n            if (this.getIsFocused())\r\n                return true;\r\n\r\n            if (!!this.children && _.isArray(this.children)) {\r\n                return _(this.children).any(function(child) {\r\n                    return child.getIsSelected();\r\n                });\r\n            }\r\n\r\n            return false;\r\n        };\r\n\r\n        this.getIsDropTarget = function () {\r\n            if (!this.editor)\r\n                return false;\r\n            return this.editor.dropTargetElement === this;\r\n        }\r\n\r\n        this.setIsDropTarget = function (value) {\r\n            if (!this.editor)\r\n                return;\r\n            if (value)\r\n                this.editor.dropTargetElement = this;\r\n            else\r\n                this.editor.dropTargetElement = null;\r\n        };\r\n\r\n        this.canDelete = function () {\r\n            if (this.isTemplated || !this.parent)\r\n                return false;\r\n            return true;\r\n        };\r\n\r\n        this.delete = function () {\r\n            if (!this.canDelete())\r\n                return;\r\n            this.parent.deleteChild(this);\r\n        };\r\n\r\n        this.canMoveUp = function () {\r\n            if (this.isTemplated || !this.parent)\r\n                return false;\r\n            return this.parent.canMoveChildUp(this);\r\n        };\r\n\r\n        this.moveUp = function () {\r\n            if (!this.canMoveUp())\r\n                return;\r\n            this.parent.moveChildUp(this);\r\n        };\r\n\r\n        this.canMoveDown = function () {\r\n            if (this.isTemplated || !this.parent)\r\n                return false;\r\n            return this.parent.canMoveChildDown(this);\r\n        };\r\n\r\n        this.moveDown = function () {\r\n            if (!this.canMoveDown())\r\n                return;\r\n            this.parent.moveChildDown(this);\r\n        };\r\n\r\n        this.elementToObject = function () {\r\n            return {\r\n                type: this.type,\r\n                data: this.data,\r\n                htmlId: this.htmlId,\r\n                htmlClass: this.htmlClass,\r\n                htmlStyle: this.htmlStyle,\r\n                isTemplated: this.isTemplated,\r\n                rule: this.rule\r\n            };\r\n        };\r\n\r\n        this.getEditorObject = function() {\r\n            return {};\r\n        };\r\n\r\n        this.copy = function (clipboardData) {\r\n            var text = this.getInnerText();\r\n            clipboardData.setData(\"text/plain\", text);\r\n\r\n            var data = this.toObject();\r\n            var json = JSON.stringify(data, null, \"\\t\");\r\n            clipboardData.setData(\"text/json\", json);\r\n        };\r\n\r\n        this.cut = function (clipboardData) {\r\n            if (this.canDelete()) {\r\n                this.copy(clipboardData);\r\n                this.delete();\r\n            }\r\n        };\r\n\r\n        this.paste = function (clipboardData) {\r\n            if (!!this.parent)\r\n                this.parent.paste(clipboardData);\r\n        };\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Container = function (allowedChildTypes, children) {\r\n\r\n        this.allowedChildTypes = allowedChildTypes;\r\n        this.children = children;\r\n        this.isContainer = true;\r\n\r\n        var self = this;\r\n\r\n        this.onChildAdded = function (element) { /* Virtual */ };\r\n        this.onDescendantAdded = function (element, parentElement) { /* Virtual */ };\r\n\r\n        this.setChildren = function (children) {\r\n            this.children = children;\r\n            _(this.children).each(function (child) {\r\n                child.setParent(self);\r\n            });\r\n        };\r\n\r\n        this.setChildren(children);\r\n\r\n        this.getIsSealed = function () {\r\n            return _(this.children).any(function (child) {\r\n                return child.isTemplated;\r\n            });\r\n        };\r\n\r\n        var _baseSetIsFocused = this.setIsFocused;\r\n        this.setIsFocused = function () {\r\n            if (this.getIsSealed())\r\n                return;\r\n            _baseSetIsFocused.call(this);\r\n        };\r\n\r\n        this.addChild = function (child) {\r\n            if (!_(this.children).contains(child) && (_(this.allowedChildTypes).contains(child.type) || child.isContainable))\r\n                this.children.push(child);\r\n            child.setEditor(this.editor);\r\n            child.setIsTemplated(false);\r\n            child.setParent(this);\r\n        };\r\n\r\n        this.deleteChild = function (child) {\r\n            var index = _(this.children).indexOf(child);\r\n            if (index >= 0) {\r\n                this.children.splice(index, 1);\r\n                if (child.getIsActive())\r\n                    this.editor.activeElement = null;\r\n                if (child.getIsFocused()) {\r\n                    // If the deleted child was focused, try to set new focus to the most appropriate sibling or parent.\r\n                    if (this.children.length > index)\r\n                        this.children[index].setIsFocused();\r\n                    else if (index > 0)\r\n                        this.children[index - 1].setIsFocused();\r\n                    else\r\n                        this.setIsFocused();\r\n                }\r\n            }\r\n        };\r\n\r\n        this.moveFocusPrevChild = function (child) {\r\n            if (this.children.length < 2)\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            if (index > 0)\r\n                this.children[index - 1].setIsFocused();\r\n        };\r\n\r\n        this.moveFocusNextChild = function (child) {\r\n            if (this.children.length < 2)\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            if (index < this.children.length - 1)\r\n                this.children[index + 1].setIsFocused();\r\n        };\r\n\r\n        this.insertChild = function (child, afterChild) {\r\n            if (!_(this.children).contains(child)) {\r\n                var index = Math.max(_(this.children).indexOf(afterChild), 0);\r\n                this.children.splice(index + 1, 0, child);\r\n                child.setEditor(this.editor);\r\n                child.parent = this;\r\n            }\r\n        };\r\n\r\n        this.moveChildUp = function (child) {\r\n            if (!this.canMoveChildUp(child))\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            this.children.move(index, index - 1);\r\n        };\r\n\r\n        this.moveChildDown = function (child) {\r\n            if (!this.canMoveChildDown(child))\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            this.children.move(index, index + 1);\r\n        };\r\n\r\n        this.canMoveChildUp = function (child) {\r\n            var index = _(this.children).indexOf(child);\r\n            return index > 0;\r\n        };\r\n\r\n        this.canMoveChildDown = function (child) {\r\n            var index = _(this.children).indexOf(child);\r\n            return index < this.children.length - 1;\r\n        };\r\n\r\n        this.childrenToObject = function () {\r\n            return _(this.children).map(function (child) {\r\n                return child.toObject();\r\n            });\r\n        };\r\n\r\n        this.getInnerText = function () {\r\n            return _(this.children).reduce(function (memo, child) {\r\n                return memo + \"\\n\" + child.getInnerText();\r\n            }, \"\");\r\n        }\r\n\r\n        this.paste = function (clipboardData) {\r\n            var json = clipboardData.getData(\"text/json\");\r\n            if (!!json) {\r\n                var data = JSON.parse(json);\r\n                var child = LayoutEditor.elementFrom(data);\r\n                this.pasteChild(child);\r\n            }\r\n        };\r\n\r\n        this.pasteChild = function (child) {\r\n            if (_(this.allowedChildTypes).contains(child.type) || child.isContainable) {\r\n                this.addChild(child);\r\n                child.setIsFocused();\r\n            }\r\n            else if (!!this.parent)\r\n                this.parent.pasteChild(child);\r\n        }\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Canvas = function (data, htmlId, htmlClass, htmlStyle, isTemplated, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Canvas\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Canvas\", \"Grid\", \"Content\"], children);\r\n\r\n        this.isContainable = true;\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Canvas.from = function (value) {\r\n        var result = new LayoutEditor.Canvas(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));\r\n","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Grid = function (data, htmlId, htmlClass, htmlStyle, isTemplated, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Grid\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Row\"], children);\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Grid.from = function (value) {\r\n        var result = new LayoutEditor.Grid(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Row = function (data, htmlId, htmlClass, htmlStyle, isTemplated, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Row\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Column\"], children);\r\n\r\n        var _self = this;\r\n\r\n        function _getTotalColumnsWidth() {\r\n            return _(_self.children).reduce(function (memo, child) {\r\n                return memo + child.offset + child.width;\r\n            }, 0);\r\n        }\r\n\r\n        // Implements a simple algorithm to distribute space (either positive or negative)\r\n        // between the child columns of the row. Negative space is distributed when making\r\n        // room for a new column (e.c. clipboard paste or dropping from the toolbox) while\r\n        // positive space is distributed when filling the grap of a removed column.\r\n        function _distributeSpace(space) {\r\n            if (space == 0)\r\n                return true;\r\n             \r\n            var undistributedSpace = space;\r\n\r\n            if (undistributedSpace < 0) {\r\n                var vacantSpace = 12 - _getTotalColumnsWidth();\r\n                undistributedSpace += vacantSpace;\r\n                if (undistributedSpace > 0)\r\n                    undistributedSpace = 0;\r\n            }\r\n\r\n            // If space is negative, try to decrease offsets first.\r\n            while (undistributedSpace < 0 && _(_self.children).any(function (column) { return column.offset > 0; })) { // While there is still offset left to remove.\r\n                for (i = 0; i < _self.children.length && undistributedSpace < 0; i++) {\r\n                    var column = _self.children[i];\r\n                    if (column.offset > 0) {\r\n                        column.offset--;\r\n                        undistributedSpace++;\r\n                    }\r\n                }\r\n            }\r\n\r\n            function hasWidth(column) {\r\n                if (undistributedSpace > 0)\r\n                    return column.width < 12;\r\n                else if (undistributedSpace < 0)\r\n                    return column.width > 1;\r\n                return false;\r\n            }\r\n\r\n            // Try to distribute remaining space (could be negative or positive) using widths.\r\n            while (undistributedSpace != 0) {\r\n                // Any more column width available for distribution?\r\n                if (!_(_self.children).any(hasWidth))\r\n                    break;\r\n                for (i = 0; i < _self.children.length && undistributedSpace != 0; i++) {\r\n                    var column = _self.children[i % _self.children.length];\r\n                    if (hasWidth(column)) {\r\n                        var delta = undistributedSpace / Math.abs(undistributedSpace);\r\n                        column.width += delta;\r\n                        undistributedSpace -= delta;\r\n                    }\r\n                }                \r\n            }\r\n\r\n            return undistributedSpace == 0;\r\n        }\r\n\r\n        var _isAddingColumn = false;\r\n\r\n        this.canAddColumn = function () {\r\n            return this.children.length < 12;\r\n        };\r\n\r\n        this.beginAddColumn = function (newColumnWidth) {\r\n            if (!!_isAddingColumn)\r\n                throw new Error(\"Column add operation is already in progress.\")\r\n            _(this.children).each(function (column) {\r\n                column.beginChange();\r\n            });\r\n            if (_distributeSpace(-newColumnWidth)) {\r\n                _isAddingColumn = true;\r\n                return true;\r\n            }\r\n            _(this.children).each(function (column) {\r\n                column.rollbackChange();\r\n            });\r\n            return false;\r\n        };\r\n\r\n        this.commitAddColumn = function () {\r\n            if (!_isAddingColumn)\r\n                throw new Error(\"No column add operation in progress.\")\r\n            _(this.children).each(function (column) {\r\n                column.commitChange();\r\n            });\r\n            _isAddingColumn = false;\r\n        };\r\n\r\n        this.rollbackAddColumn = function () {\r\n            if (!_isAddingColumn)\r\n                throw new Error(\"No column add operation in progress.\")\r\n            _(this.children).each(function (column) {\r\n                column.rollbackChange();\r\n            });\r\n            _isAddingColumn = false;\r\n        };\r\n\r\n        var _baseDeleteChild = this.deleteChild;\r\n        this.deleteChild = function (column) { \r\n            var width = column.width;\r\n            _baseDeleteChild.call(this, column);\r\n            _distributeSpace(width);\r\n        };\r\n\r\n        this.canContractColumnRight = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0)\r\n                return column.width > 1;\r\n            return false;\r\n        };\r\n\r\n        this.contractColumnRight = function (column, connectAdjacent) {\r\n            if (!this.canContractColumnRight(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (column.width > 1) {\r\n                    column.width--;\r\n                    if (this.children.length > index + 1) {\r\n                        var nextColumn = this.children[index + 1];\r\n                        if (connectAdjacent && nextColumn.offset == 0)\r\n                            nextColumn.width++;\r\n                        else\r\n                            nextColumn.offset++;\r\n                    }\r\n                }\r\n            }\r\n        };\r\n\r\n        this.canExpandColumnRight = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (column.width >= 12)\r\n                    return false;\r\n                if (this.children.length > index + 1) {\r\n                    var nextColumn = this.children[index + 1];\r\n                    if (connectAdjacent && nextColumn.offset == 0)\r\n                        return nextColumn.width > 1;\r\n                    else\r\n                        return nextColumn.offset > 0;\r\n                }\r\n                return _getTotalColumnsWidth() < 12;\r\n            }\r\n            return false;\r\n        };\r\n\r\n        this.expandColumnRight = function (column, connectAdjacent) {\r\n            if (!this.canExpandColumnRight(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (this.children.length > index + 1) {\r\n                    var nextColumn = this.children[index + 1];\r\n                    if (connectAdjacent && nextColumn.offset == 0)\r\n                        nextColumn.width--;\r\n                    else\r\n                        nextColumn.offset--;\r\n                }\r\n                column.width++;\r\n            }\r\n        };\r\n\r\n        this.canExpandColumnLeft = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (column.width >= 12)\r\n                    return false;\r\n                if (index > 0) {\r\n                    var prevColumn = this.children[index - 1];\r\n                    if (connectAdjacent && column.offset == 0)\r\n                        return prevColumn.width > 1;\r\n                }\r\n                return column.offset > 0;\r\n            }\r\n            return false;\r\n        };\r\n\r\n        this.expandColumnLeft = function (column, connectAdjacent) {\r\n            if (!this.canExpandColumnLeft(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (index > 0) {\r\n                    var prevColumn = this.children[index - 1];\r\n                    if (connectAdjacent && column.offset == 0)\r\n                        prevColumn.width--;\r\n                    else\r\n                        column.offset--;\r\n                }\r\n                else\r\n                    column.offset--;\r\n                column.width++;\r\n            }\r\n        };\r\n\r\n        this.canContractColumnLeft = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0)\r\n                return column.width > 1;\r\n            return false;\r\n        };\r\n\r\n        this.contractColumnLeft = function (column, connectAdjacent) {\r\n            if (!this.canContractColumnLeft(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (index > 0) {\r\n                    var prevColumn = this.children[index - 1];\r\n                    if (connectAdjacent && column.offset == 0)\r\n                        prevColumn.width++;\r\n                    else\r\n                        column.offset++;\r\n                }\r\n                else\r\n                    column.offset++;\r\n                column.width--;\r\n            }\r\n        };\r\n\r\n        this.evenColumns = function () {\r\n            if (this.children.length == 0)\r\n                return;\r\n\r\n            var evenWidth = Math.floor(12 / this.children.length);\r\n            _(this.children).each(function (column) {\r\n                column.width = evenWidth;\r\n                column.offset = 0;\r\n            });\r\n\r\n            var rest = 12 % this.children.length;\r\n            if (rest > 0)\r\n                _distributeSpace(rest);\r\n        };\r\n\r\n        var _basePasteChild = this.pasteChild;\r\n        this.pasteChild = function (child) {\r\n            if (child.type == \"Column\") {\r\n                if (this.beginAddColumn(child.width)) {\r\n                    this.commitAddColumn();\r\n                    _basePasteChild.call(this, child)\r\n                }\r\n            }\r\n            else if (!!this.parent)\r\n                this.parent.pasteChild(child);\r\n        }\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Row.from = function (value) {\r\n        var result = new LayoutEditor.Row(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n    LayoutEditor.Column = function (data, htmlId, htmlClass, htmlStyle, isTemplated, width, offset, collapsible, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Column\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Grid\", \"Content\"], children);\r\n\r\n        this.width = width;\r\n        this.offset = offset;\r\n        this.collapsible = collapsible;\r\n\r\n        var _hasPendingChange = false;\r\n        var _origWidth = 0;\r\n        var _origOffset = 0;\r\n\r\n        this.beginChange = function () {\r\n            if (!!_hasPendingChange)\r\n                throw new Error(\"Column already has a pending change.\");\r\n            _hasPendingChange = true;\r\n            _origWidth = this.width;\r\n            _origOffset = this.offset;\r\n        };\r\n\r\n        this.commitChange = function () {\r\n            if (!_hasPendingChange)\r\n                throw new Error(\"Column has no pending change.\");\r\n            _origWidth = 0;\r\n            _origOffset = 0;\r\n            _hasPendingChange = false;\r\n        };\r\n\r\n        this.rollbackChange = function () {\r\n            if (!_hasPendingChange)\r\n                throw new Error(\"Column has no pending change.\");\r\n            this.width = _origWidth;\r\n            this.offset = _origOffset;\r\n            _origWidth = 0;\r\n            _origOffset = 0;\r\n            _hasPendingChange = false;\r\n        };\r\n\r\n        this.canSplit = function () {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.width > 1;\r\n        };\r\n\r\n        this.split = function () {\r\n            if (!this.canSplit())\r\n                return;\r\n\r\n            var newColumnWidth = Math.floor(this.width / 2);\r\n            var newColumn = LayoutEditor.Column.from({\r\n                data: null,\r\n                htmlId: null,\r\n                htmlClass: null,\r\n                htmlStyle: null,\r\n                width: newColumnWidth,\r\n                offset: 0,\r\n                children: []\r\n            });\r\n\r\n            this.width = this.width - newColumnWidth;\r\n            this.parent.insertChild(newColumn, this);\r\n            newColumn.setIsFocused();\r\n        };\r\n\r\n        this.canContractRight = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canContractColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.contractRight = function (connectAdjacent) {\r\n            if (!this.canContractRight(connectAdjacent))\r\n                return;\r\n            this.parent.contractColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.canExpandRight = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canExpandColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.expandRight = function (connectAdjacent) {\r\n            if (!this.canExpandRight(connectAdjacent))\r\n                return;\r\n            this.parent.expandColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.canExpandLeft = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canExpandColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.expandLeft = function (connectAdjacent) {\r\n            if (!this.canExpandLeft(connectAdjacent))\r\n                return;\r\n            this.parent.expandColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.canContractLeft = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canContractColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.contractLeft = function (connectAdjacent) {\r\n            if (!this.canContractLeft(connectAdjacent))\r\n                return;\r\n            this.parent.contractColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.width = this.width;\r\n            result.offset = this.offset;\r\n            result.collapsible = this.collapsible;\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Column.from = function (value) {\r\n        var result = new LayoutEditor.Column(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.width,\r\n            value.offset,\r\n            value.collapsible,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n        return result;\r\n    };\r\n\r\n    LayoutEditor.Column.times = function (value) {\r\n        return _.times(value, function (n) {\r\n            return LayoutEditor.Column.from({\r\n                data: null,\r\n                htmlId: null,\r\n                htmlClass: null,\r\n                isTemplated: false,\r\n                width: 12 / value,\r\n                offset: 0,\r\n                collapsible: null,\r\n                children: []\r\n            });\r\n        });\r\n    };\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Content = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor, rule) {\r\n        LayoutEditor.Element.call(this, \"Content\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n\r\n        this.contentType = contentType;\r\n        this.contentTypeLabel = contentTypeLabel;\r\n        this.contentTypeClass = contentTypeClass;\r\n        this.html = html;\r\n        this.hasEditor = hasEditor;\r\n\r\n        this.getInnerText = function () {\r\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\r\n        };\r\n\r\n        // This function will be overwritten by the Content directive.\r\n        this.setHtml = function (html) {\r\n            this.html = html;\r\n            this.htmlUnsafe = html;\r\n        }\r\n\r\n        this.toObject = function () {\r\n            return {\r\n                \"type\": \"Content\"\r\n            };\r\n        };\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.contentType = this.contentType;\r\n            result.contentTypeLabel = this.contentTypeLabel;\r\n            result.contentTypeClass = this.contentTypeClass;\r\n            result.html = this.html;\r\n            result.hasEditor = hasEditor;\r\n            return result;\r\n        };\r\n\r\n        this.setHtml(html);\r\n    };\r\n\r\n    LayoutEditor.Content.from = function (value) {\r\n        var result = new LayoutEditor.Content(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.contentType,\r\n            value.contentTypeLabel,\r\n            value.contentTypeClass,\r\n            value.html,\r\n            value.hasEditor,\r\n            value.rule);\r\n\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function ($, LayoutEditor) {\r\n\r\n    LayoutEditor.Html = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor, rule) {\r\n        LayoutEditor.Element.call(this, \"Html\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n\r\n        this.contentType = contentType;\r\n        this.contentTypeLabel = contentTypeLabel;\r\n        this.contentTypeClass = contentTypeClass;\r\n        this.html = html;\r\n        this.hasEditor = hasEditor;\r\n        this.isContainable = true;\r\n\r\n        this.getInnerText = function () {\r\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\r\n        };\r\n\r\n        // This function will be overwritten by the Content directive.\r\n        this.setHtml = function (html) {\r\n            this.html = html;\r\n            this.htmlUnsafe = html;\r\n        }\r\n\r\n        this.toObject = function () {\r\n            return {\r\n                \"type\": \"Html\"\r\n            };\r\n        };\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.contentType = this.contentType;\r\n            result.contentTypeLabel = this.contentTypeLabel;\r\n            result.contentTypeClass = this.contentTypeClass;\r\n            result.html = this.html;\r\n            result.hasEditor = hasEditor;\r\n            return result;\r\n        };\r\n\r\n        var getEditorObject = this.getEditorObject;\r\n        this.getEditorObject = function () {\r\n            var dto = getEditorObject();\r\n            return $.extend(dto, {\r\n                Content: this.html\r\n            });\r\n        }\r\n\r\n        this.setHtml(html);\r\n    };\r\n\r\n    LayoutEditor.Html.from = function (value) {\r\n        var result = new LayoutEditor.Html(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.contentType,\r\n            value.contentTypeLabel,\r\n            value.contentTypeClass,\r\n            value.html,\r\n            value.hasEditor,\r\n            value.rule);\r\n\r\n        return result;\r\n    };\r\n\r\n    LayoutEditor.registerFactory(\"Html\", function(value) { return LayoutEditor.Html.from(value); });\r\n\r\n})(jQuery, LayoutEditor || (LayoutEditor = {}));"],"sourceRoot":"/source/"} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Helpers.js","Editor.js","Element.js","Container.js","Canvas.js","Grid.js","Row.js","Column.js","Content.js","Html.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC9IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC5JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"Models.js","sourcesContent":["var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    Array.prototype.move = function (from, to) {\r\n        this.splice(to, 0, this.splice(from, 1)[0]);\r\n    };\r\n\r\n    LayoutEditor.childrenFrom = function(values) {\r\n        return _(values).map(function(value) {\r\n            return LayoutEditor.elementFrom(value);\r\n        });\r\n    };\r\n\r\n    var registerFactory = LayoutEditor.registerFactory = function(type, factory) {\r\n        var factories = LayoutEditor.factories = LayoutEditor.factories || {};\r\n        factories[type] = factory;\r\n    };\r\n\r\n    registerFactory(\"Canvas\", function (value) { return LayoutEditor.Canvas.from(value); });\r\n    registerFactory(\"Grid\", function(value) { return LayoutEditor.Grid.from(value); });\r\n    registerFactory(\"Row\", function(value) { return LayoutEditor.Row.from(value); });\r\n    registerFactory(\"Column\", function(value) { return LayoutEditor.Column.from(value); });\r\n    registerFactory(\"Content\", function(value) { return LayoutEditor.Content.from(value); });\r\n\r\n    LayoutEditor.elementFrom = function (value) {\r\n        var factory = LayoutEditor.factories[value.type];\r\n\r\n        if (!factory)\r\n            throw new Error(\"No element with type \\\"\" + value.type + \"\\\" was found.\");\r\n\r\n        var element = factory(value);\r\n        return element;\r\n    };\r\n\r\n    LayoutEditor.setModel = function (elementSelector, model) {\r\n        $(elementSelector).scope().element = model;\r\n    };\r\n\r\n    LayoutEditor.getModel = function (elementSelector) {\r\n        return $(elementSelector).scope().element;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Editor = function (config, canvasData) {\r\n        this.config = config;\r\n        this.canvas = LayoutEditor.Canvas.from(canvasData);\r\n        this.initialState = JSON.stringify(this.canvas.toObject());\r\n        this.activeElement = null;\r\n        this.focusedElement = null;\r\n        this.dropTargetElement = null;\r\n        this.isDragging = false;\r\n        this.isResizing = false;\r\n\r\n        this.resetToolboxElements = function () {\r\n            this.toolboxElements = [\r\n                LayoutEditor.Row.from({\r\n                    children: []\r\n                })\r\n            ];\r\n        };\r\n\r\n        this.isDirty = function() {\r\n            var currentState = JSON.stringify(this.canvas.toObject());\r\n            return this.initialState != currentState;\r\n        };\r\n\r\n        this.resetToolboxElements();\r\n        this.canvas.setEditor(this);\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));\r\n","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Element = function (type, data, htmlId, htmlClass, htmlStyle, isTemplated, rule) {\r\n        if (!type)\r\n            throw new Error(\"Parameter 'type' is required.\");\r\n\r\n        this.type = type;\r\n        this.data = data;\r\n        this.htmlId = htmlId;\r\n        this.htmlClass = htmlClass;\r\n        this.htmlStyle = htmlStyle;\r\n        this.isTemplated = isTemplated;\r\n        this.rule = rule;\r\n\r\n        this.editor = null;\r\n        this.parent = null;\r\n        this.setIsFocusedEventHandlers = [];\r\n\r\n        this.setEditor = function (editor) {\r\n            this.editor = editor;\r\n            if (!!this.children && _.isArray(this.children)) {\r\n                _(this.children).each(function (child) {\r\n                    child.setEditor(editor);\r\n                });\r\n            }\r\n        };\r\n\r\n        this.setParent = function(parentElement) {\r\n            this.parent = parentElement;\r\n            this.parent.onChildAdded(this);\r\n\r\n            var currentAncestor = parentElement;\r\n            while (!!currentAncestor) {\r\n                currentAncestor.onDescendantAdded(this, parentElement);\r\n                currentAncestor = currentAncestor.parent;\r\n            }\r\n        };\r\n\r\n        this.setIsTemplated = function (value) {\r\n            this.isTemplated = value;\r\n            if (!!this.children && _.isArray(this.children)) {\r\n                _(this.children).each(function (child) {\r\n                    child.setIsTemplated(value);\r\n                });\r\n            }\r\n        };\r\n\r\n        this.applyElementEditorModel = function() { /* Virtual */ };\r\n\r\n        this.getIsActive = function () {\r\n            if (!this.editor)\r\n                return false;\r\n            return this.editor.activeElement === this && !this.getIsFocused();\r\n        };\r\n\r\n        this.setIsActive = function (value) {\r\n            if (!this.editor)\r\n                return;\r\n            if (this.editor.isDragging || this.editor.isResizing)\r\n                return;\r\n\r\n            if (value)\r\n                this.editor.activeElement = this;\r\n            else\r\n                this.editor.activeElement = this.parent;\r\n        };\r\n\r\n        this.getIsFocused = function () {\r\n            if (!this.editor)\r\n                return false;\r\n            return this.editor.focusedElement === this;\r\n        };\r\n\r\n        this.setIsFocused = function () {\r\n            if (!this.editor)\r\n            \treturn;\r\n            if (!this.children && this.isTemplated)\r\n            \treturn;\r\n            if (this.editor.isDragging || this.editor.isResizing)\r\n                return;\r\n\r\n            this.editor.focusedElement = this;\r\n            _(this.setIsFocusedEventHandlers).each(function (item) {\r\n                try {\r\n                    item();\r\n                }\r\n                catch (ex) {\r\n                    // Ignore.\r\n                }\r\n            });\r\n        };\r\n\r\n        this.getIsSelected = function () {\r\n            if (this.getIsFocused())\r\n                return true;\r\n\r\n            if (!!this.children && _.isArray(this.children)) {\r\n                return _(this.children).any(function(child) {\r\n                    return child.getIsSelected();\r\n                });\r\n            }\r\n\r\n            return false;\r\n        };\r\n\r\n        this.getIsDropTarget = function () {\r\n            if (!this.editor)\r\n                return false;\r\n            return this.editor.dropTargetElement === this;\r\n        }\r\n\r\n        this.setIsDropTarget = function (value) {\r\n            if (!this.editor)\r\n                return;\r\n            if (value)\r\n                this.editor.dropTargetElement = this;\r\n            else\r\n                this.editor.dropTargetElement = null;\r\n        };\r\n\r\n        this.canDelete = function () {\r\n            if (this.isTemplated || !this.parent)\r\n                return false;\r\n            return true;\r\n        };\r\n\r\n        this.delete = function () {\r\n            if (!this.canDelete())\r\n                return;\r\n            this.parent.deleteChild(this);\r\n        };\r\n\r\n        this.canMoveUp = function () {\r\n            if (this.isTemplated || !this.parent)\r\n                return false;\r\n            return this.parent.canMoveChildUp(this);\r\n        };\r\n\r\n        this.moveUp = function () {\r\n            if (!this.canMoveUp())\r\n                return;\r\n            this.parent.moveChildUp(this);\r\n        };\r\n\r\n        this.canMoveDown = function () {\r\n            if (this.isTemplated || !this.parent)\r\n                return false;\r\n            return this.parent.canMoveChildDown(this);\r\n        };\r\n\r\n        this.moveDown = function () {\r\n            if (!this.canMoveDown())\r\n                return;\r\n            this.parent.moveChildDown(this);\r\n        };\r\n\r\n        this.elementToObject = function () {\r\n            return {\r\n                type: this.type,\r\n                data: this.data,\r\n                htmlId: this.htmlId,\r\n                htmlClass: this.htmlClass,\r\n                htmlStyle: this.htmlStyle,\r\n                isTemplated: this.isTemplated,\r\n                rule: this.rule\r\n            };\r\n        };\r\n\r\n        this.getEditorObject = function() {\r\n            return {};\r\n        };\r\n\r\n        this.copy = function (clipboardData) {\r\n            var text = this.getInnerText();\r\n            clipboardData.setData(\"text/plain\", text);\r\n\r\n            var data = this.toObject();\r\n            var json = JSON.stringify(data, null, \"\\t\");\r\n            clipboardData.setData(\"text/json\", json);\r\n        };\r\n\r\n        this.cut = function (clipboardData) {\r\n            if (this.canDelete()) {\r\n                this.copy(clipboardData);\r\n                this.delete();\r\n            }\r\n        };\r\n\r\n        this.paste = function (clipboardData) {\r\n            if (!!this.parent)\r\n                this.parent.paste(clipboardData);\r\n        };\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Container = function (allowedChildTypes, children) {\r\n\r\n        this.allowedChildTypes = allowedChildTypes;\r\n        this.children = children;\r\n        this.isContainer = true;\r\n\r\n        var self = this;\r\n\r\n        this.onChildAdded = function (element) { /* Virtual */ };\r\n        this.onDescendantAdded = function (element, parentElement) { /* Virtual */ };\r\n\r\n        this.setChildren = function (children) {\r\n            this.children = children;\r\n            _(this.children).each(function (child) {\r\n                child.setParent(self);\r\n            });\r\n        };\r\n\r\n        this.setChildren(children);\r\n\r\n        this.getIsSealed = function () {\r\n            return _(this.children).any(function (child) {\r\n                return child.isTemplated;\r\n            });\r\n        };\r\n\r\n        var _baseSetIsFocused = this.setIsFocused;\r\n        this.setIsFocused = function () {\r\n            if (this.getIsSealed())\r\n                return;\r\n            _baseSetIsFocused.call(this);\r\n        };\r\n\r\n        this.addChild = function (child) {\r\n            if (!_(this.children).contains(child) && (_(this.allowedChildTypes).contains(child.type) || child.isContainable))\r\n                this.children.push(child);\r\n            child.setEditor(this.editor);\r\n            child.setIsTemplated(false);\r\n            child.setParent(this);\r\n        };\r\n\r\n        this.deleteChild = function (child) {\r\n            var index = _(this.children).indexOf(child);\r\n            if (index >= 0) {\r\n                this.children.splice(index, 1);\r\n                if (child.getIsActive())\r\n                    this.editor.activeElement = null;\r\n                if (child.getIsFocused()) {\r\n                    // If the deleted child was focused, try to set new focus to the most appropriate sibling or parent.\r\n                    if (this.children.length > index)\r\n                        this.children[index].setIsFocused();\r\n                    else if (index > 0)\r\n                        this.children[index - 1].setIsFocused();\r\n                    else\r\n                        this.setIsFocused();\r\n                }\r\n            }\r\n        };\r\n\r\n        this.moveFocusPrevChild = function (child) {\r\n            if (this.children.length < 2)\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            if (index > 0)\r\n                this.children[index - 1].setIsFocused();\r\n        };\r\n\r\n        this.moveFocusNextChild = function (child) {\r\n            if (this.children.length < 2)\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            if (index < this.children.length - 1)\r\n                this.children[index + 1].setIsFocused();\r\n        };\r\n\r\n        this.insertChild = function (child, afterChild) {\r\n            if (!_(this.children).contains(child)) {\r\n                var index = Math.max(_(this.children).indexOf(afterChild), 0);\r\n                this.children.splice(index + 1, 0, child);\r\n                child.setEditor(this.editor);\r\n                child.parent = this;\r\n            }\r\n        };\r\n\r\n        this.moveChildUp = function (child) {\r\n            if (!this.canMoveChildUp(child))\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            this.children.move(index, index - 1);\r\n        };\r\n\r\n        this.moveChildDown = function (child) {\r\n            if (!this.canMoveChildDown(child))\r\n                return;\r\n            var index = _(this.children).indexOf(child);\r\n            this.children.move(index, index + 1);\r\n        };\r\n\r\n        this.canMoveChildUp = function (child) {\r\n            var index = _(this.children).indexOf(child);\r\n            return index > 0;\r\n        };\r\n\r\n        this.canMoveChildDown = function (child) {\r\n            var index = _(this.children).indexOf(child);\r\n            return index < this.children.length - 1;\r\n        };\r\n\r\n        this.childrenToObject = function () {\r\n            return _(this.children).map(function (child) {\r\n                return child.toObject();\r\n            });\r\n        };\r\n\r\n        this.getInnerText = function () {\r\n            return _(this.children).reduce(function (memo, child) {\r\n                return memo + \"\\n\" + child.getInnerText();\r\n            }, \"\");\r\n        }\r\n\r\n        this.paste = function (clipboardData) {\r\n            var json = clipboardData.getData(\"text/json\");\r\n            if (!!json) {\r\n                var data = JSON.parse(json);\r\n                var child = LayoutEditor.elementFrom(data);\r\n                this.pasteChild(child);\r\n            }\r\n        };\r\n\r\n        this.pasteChild = function (child) {\r\n            if (_(this.allowedChildTypes).contains(child.type) || child.isContainable) {\r\n                this.addChild(child);\r\n                child.setIsFocused();\r\n            }\r\n            else if (!!this.parent)\r\n                this.parent.pasteChild(child);\r\n        }\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Canvas = function (data, htmlId, htmlClass, htmlStyle, isTemplated, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Canvas\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Canvas\", \"Grid\", \"Content\"], children);\r\n\r\n        this.isContainable = true;\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Canvas.from = function (value) {\r\n        var result = new LayoutEditor.Canvas(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));\r\n","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Grid = function (data, htmlId, htmlClass, htmlStyle, isTemplated, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Grid\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Row\"], children);\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Grid.from = function (value) {\r\n        var result = new LayoutEditor.Grid(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Row = function (data, htmlId, htmlClass, htmlStyle, isTemplated, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Row\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Column\"], children);\r\n\r\n        var _self = this;\r\n\r\n        function _getTotalColumnsWidth() {\r\n            return _(_self.children).reduce(function (memo, child) {\r\n                return memo + child.offset + child.width;\r\n            }, 0);\r\n        }\r\n\r\n        // Implements a simple algorithm to distribute space (either positive or negative)\r\n        // between the child columns of the row. Negative space is distributed when making\r\n        // room for a new column (e.c. clipboard paste or dropping from the toolbox) while\r\n        // positive space is distributed when filling the grap of a removed column.\r\n        function _distributeSpace(space) {\r\n            if (space == 0)\r\n                return true;\r\n             \r\n            var undistributedSpace = space;\r\n\r\n            if (undistributedSpace < 0) {\r\n                var vacantSpace = 12 - _getTotalColumnsWidth();\r\n                undistributedSpace += vacantSpace;\r\n                if (undistributedSpace > 0)\r\n                    undistributedSpace = 0;\r\n            }\r\n\r\n            // If space is negative, try to decrease offsets first.\r\n            while (undistributedSpace < 0 && _(_self.children).any(function (column) { return column.offset > 0; })) { // While there is still offset left to remove.\r\n                for (i = 0; i < _self.children.length && undistributedSpace < 0; i++) {\r\n                    var column = _self.children[i];\r\n                    if (column.offset > 0) {\r\n                        column.offset--;\r\n                        undistributedSpace++;\r\n                    }\r\n                }\r\n            }\r\n\r\n            function hasWidth(column) {\r\n                if (undistributedSpace > 0)\r\n                    return column.width < 12;\r\n                else if (undistributedSpace < 0)\r\n                    return column.width > 1;\r\n                return false;\r\n            }\r\n\r\n            // Try to distribute remaining space (could be negative or positive) using widths.\r\n            while (undistributedSpace != 0) {\r\n                // Any more column width available for distribution?\r\n                if (!_(_self.children).any(hasWidth))\r\n                    break;\r\n                for (i = 0; i < _self.children.length && undistributedSpace != 0; i++) {\r\n                    var column = _self.children[i % _self.children.length];\r\n                    if (hasWidth(column)) {\r\n                        var delta = undistributedSpace / Math.abs(undistributedSpace);\r\n                        column.width += delta;\r\n                        undistributedSpace -= delta;\r\n                    }\r\n                }                \r\n            }\r\n\r\n            return undistributedSpace == 0;\r\n        }\r\n\r\n        var _isAddingColumn = false;\r\n\r\n        this.canAddColumn = function () {\r\n            return this.children.length < 12;\r\n        };\r\n\r\n        this.beginAddColumn = function (newColumnWidth) {\r\n            if (!!_isAddingColumn)\r\n                throw new Error(\"Column add operation is already in progress.\")\r\n            _(this.children).each(function (column) {\r\n                column.beginChange();\r\n            });\r\n            if (_distributeSpace(-newColumnWidth)) {\r\n                _isAddingColumn = true;\r\n                return true;\r\n            }\r\n            _(this.children).each(function (column) {\r\n                column.rollbackChange();\r\n            });\r\n            return false;\r\n        };\r\n\r\n        this.commitAddColumn = function () {\r\n            if (!_isAddingColumn)\r\n                throw new Error(\"No column add operation in progress.\")\r\n            _(this.children).each(function (column) {\r\n                column.commitChange();\r\n            });\r\n            _isAddingColumn = false;\r\n        };\r\n\r\n        this.rollbackAddColumn = function () {\r\n            if (!_isAddingColumn)\r\n                throw new Error(\"No column add operation in progress.\")\r\n            _(this.children).each(function (column) {\r\n                column.rollbackChange();\r\n            });\r\n            _isAddingColumn = false;\r\n        };\r\n\r\n        var _baseDeleteChild = this.deleteChild;\r\n        this.deleteChild = function (column) { \r\n            var width = column.width;\r\n            _baseDeleteChild.call(this, column);\r\n            _distributeSpace(width);\r\n        };\r\n\r\n        this.canContractColumnRight = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0)\r\n                return column.width > 1;\r\n            return false;\r\n        };\r\n\r\n        this.contractColumnRight = function (column, connectAdjacent) {\r\n            if (!this.canContractColumnRight(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (column.width > 1) {\r\n                    column.width--;\r\n                    if (this.children.length > index + 1) {\r\n                        var nextColumn = this.children[index + 1];\r\n                        if (connectAdjacent && nextColumn.offset == 0)\r\n                            nextColumn.width++;\r\n                        else\r\n                            nextColumn.offset++;\r\n                    }\r\n                }\r\n            }\r\n        };\r\n\r\n        this.canExpandColumnRight = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (column.width >= 12)\r\n                    return false;\r\n                if (this.children.length > index + 1) {\r\n                    var nextColumn = this.children[index + 1];\r\n                    if (connectAdjacent && nextColumn.offset == 0)\r\n                        return nextColumn.width > 1;\r\n                    else\r\n                        return nextColumn.offset > 0;\r\n                }\r\n                return _getTotalColumnsWidth() < 12;\r\n            }\r\n            return false;\r\n        };\r\n\r\n        this.expandColumnRight = function (column, connectAdjacent) {\r\n            if (!this.canExpandColumnRight(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (this.children.length > index + 1) {\r\n                    var nextColumn = this.children[index + 1];\r\n                    if (connectAdjacent && nextColumn.offset == 0)\r\n                        nextColumn.width--;\r\n                    else\r\n                        nextColumn.offset--;\r\n                }\r\n                column.width++;\r\n            }\r\n        };\r\n\r\n        this.canExpandColumnLeft = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (column.width >= 12)\r\n                    return false;\r\n                if (index > 0) {\r\n                    var prevColumn = this.children[index - 1];\r\n                    if (connectAdjacent && column.offset == 0)\r\n                        return prevColumn.width > 1;\r\n                }\r\n                return column.offset > 0;\r\n            }\r\n            return false;\r\n        };\r\n\r\n        this.expandColumnLeft = function (column, connectAdjacent) {\r\n            if (!this.canExpandColumnLeft(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (index > 0) {\r\n                    var prevColumn = this.children[index - 1];\r\n                    if (connectAdjacent && column.offset == 0)\r\n                        prevColumn.width--;\r\n                    else\r\n                        column.offset--;\r\n                }\r\n                else\r\n                    column.offset--;\r\n                column.width++;\r\n            }\r\n        };\r\n\r\n        this.canContractColumnLeft = function (column, connectAdjacent) {\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0)\r\n                return column.width > 1;\r\n            return false;\r\n        };\r\n\r\n        this.contractColumnLeft = function (column, connectAdjacent) {\r\n            if (!this.canContractColumnLeft(column, connectAdjacent))\r\n                return;\r\n\r\n            var index = _(this.children).indexOf(column);\r\n            if (index >= 0) {\r\n                if (index > 0) {\r\n                    var prevColumn = this.children[index - 1];\r\n                    if (connectAdjacent && column.offset == 0)\r\n                        prevColumn.width++;\r\n                    else\r\n                        column.offset++;\r\n                }\r\n                else\r\n                    column.offset++;\r\n                column.width--;\r\n            }\r\n        };\r\n\r\n        this.evenColumns = function () {\r\n            if (this.children.length == 0)\r\n                return;\r\n\r\n            var evenWidth = Math.floor(12 / this.children.length);\r\n            _(this.children).each(function (column) {\r\n                column.width = evenWidth;\r\n                column.offset = 0;\r\n            });\r\n\r\n            var rest = 12 % this.children.length;\r\n            if (rest > 0)\r\n                _distributeSpace(rest);\r\n        };\r\n\r\n        var _basePasteChild = this.pasteChild;\r\n        this.pasteChild = function (child) {\r\n            if (child.type == \"Column\") {\r\n                if (this.beginAddColumn(child.width)) {\r\n                    this.commitAddColumn();\r\n                    _basePasteChild.call(this, child)\r\n                }\r\n            }\r\n            else if (!!this.parent)\r\n                this.parent.pasteChild(child);\r\n        }\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Row.from = function (value) {\r\n        var result = new LayoutEditor.Row(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n    LayoutEditor.Column = function (data, htmlId, htmlClass, htmlStyle, isTemplated, width, offset, collapsible, rule, children) {\r\n        LayoutEditor.Element.call(this, \"Column\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n        LayoutEditor.Container.call(this, [\"Grid\", \"Content\"], children);\r\n\r\n        this.width = width;\r\n        this.offset = offset;\r\n        this.collapsible = collapsible;\r\n\r\n        var _hasPendingChange = false;\r\n        var _origWidth = 0;\r\n        var _origOffset = 0;\r\n\r\n        this.beginChange = function () {\r\n            if (!!_hasPendingChange)\r\n                throw new Error(\"Column already has a pending change.\");\r\n            _hasPendingChange = true;\r\n            _origWidth = this.width;\r\n            _origOffset = this.offset;\r\n        };\r\n\r\n        this.commitChange = function () {\r\n            if (!_hasPendingChange)\r\n                throw new Error(\"Column has no pending change.\");\r\n            _origWidth = 0;\r\n            _origOffset = 0;\r\n            _hasPendingChange = false;\r\n        };\r\n\r\n        this.rollbackChange = function () {\r\n            if (!_hasPendingChange)\r\n                throw new Error(\"Column has no pending change.\");\r\n            this.width = _origWidth;\r\n            this.offset = _origOffset;\r\n            _origWidth = 0;\r\n            _origOffset = 0;\r\n            _hasPendingChange = false;\r\n        };\r\n\r\n        this.canSplit = function () {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.width > 1;\r\n        };\r\n\r\n        this.split = function () {\r\n            if (!this.canSplit())\r\n                return;\r\n\r\n            var newColumnWidth = Math.floor(this.width / 2);\r\n            var newColumn = LayoutEditor.Column.from({\r\n                data: null,\r\n                htmlId: null,\r\n                htmlClass: null,\r\n                htmlStyle: null,\r\n                width: newColumnWidth,\r\n                offset: 0,\r\n                children: []\r\n            });\r\n\r\n            this.width = this.width - newColumnWidth;\r\n            this.parent.insertChild(newColumn, this);\r\n            newColumn.setIsFocused();\r\n        };\r\n\r\n        this.canContractRight = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canContractColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.contractRight = function (connectAdjacent) {\r\n            if (!this.canContractRight(connectAdjacent))\r\n                return;\r\n            this.parent.contractColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.canExpandRight = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canExpandColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.expandRight = function (connectAdjacent) {\r\n            if (!this.canExpandRight(connectAdjacent))\r\n                return;\r\n            this.parent.expandColumnRight(this, connectAdjacent);\r\n        };\r\n\r\n        this.canExpandLeft = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canExpandColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.expandLeft = function (connectAdjacent) {\r\n            if (!this.canExpandLeft(connectAdjacent))\r\n                return;\r\n            this.parent.expandColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.canContractLeft = function (connectAdjacent) {\r\n            if (this.isTemplated)\r\n                return false;\r\n            return this.parent.canContractColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.contractLeft = function (connectAdjacent) {\r\n            if (!this.canContractLeft(connectAdjacent))\r\n                return;\r\n            this.parent.contractColumnLeft(this, connectAdjacent);\r\n        };\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.width = this.width;\r\n            result.offset = this.offset;\r\n            result.collapsible = this.collapsible;\r\n            result.children = this.childrenToObject();\r\n            return result;\r\n        };\r\n    };\r\n\r\n    LayoutEditor.Column.from = function (value) {\r\n        var result = new LayoutEditor.Column(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.width,\r\n            value.offset,\r\n            value.collapsible,\r\n            value.rule,\r\n            LayoutEditor.childrenFrom(value.children));\r\n        result.toolboxIcon = value.toolboxIcon;\r\n        result.toolboxLabel = value.toolboxLabel;\r\n        result.toolboxDescription = value.toolboxDescription;\r\n        return result;\r\n    };\r\n\r\n    LayoutEditor.Column.times = function (value) {\r\n        return _.times(value, function (n) {\r\n            return LayoutEditor.Column.from({\r\n                data: null,\r\n                htmlId: null,\r\n                htmlClass: null,\r\n                isTemplated: false,\r\n                width: 12 / value,\r\n                offset: 0,\r\n                collapsible: null,\r\n                children: []\r\n            });\r\n        });\r\n    };\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function (LayoutEditor) {\r\n\r\n    LayoutEditor.Content = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor, rule) {\r\n        LayoutEditor.Element.call(this, \"Content\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n\r\n        this.contentType = contentType;\r\n        this.contentTypeLabel = contentTypeLabel;\r\n        this.contentTypeClass = contentTypeClass;\r\n        this.html = html;\r\n        this.hasEditor = hasEditor;\r\n\r\n        this.getInnerText = function () {\r\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\r\n        };\r\n\r\n        // This function will be overwritten by the Content directive.\r\n        this.setHtml = function (html) {\r\n            this.html = html;\r\n            this.htmlUnsafe = html;\r\n        }\r\n\r\n        this.toObject = function () {\r\n            return {\r\n                \"type\": \"Content\"\r\n            };\r\n        };\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.contentType = this.contentType;\r\n            result.contentTypeLabel = this.contentTypeLabel;\r\n            result.contentTypeClass = this.contentTypeClass;\r\n            result.html = this.html;\r\n            result.hasEditor = hasEditor;\r\n            return result;\r\n        };\r\n\r\n        this.setHtml(html);\r\n    };\r\n\r\n    LayoutEditor.Content.from = function (value) {\r\n        var result = new LayoutEditor.Content(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.contentType,\r\n            value.contentTypeLabel,\r\n            value.contentTypeClass,\r\n            value.html,\r\n            value.hasEditor,\r\n            value.rule);\r\n\r\n        return result;\r\n    };\r\n\r\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\r\n(function ($, LayoutEditor) {\r\n\r\n    LayoutEditor.Html = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor, rule) {\r\n        LayoutEditor.Element.call(this, \"Html\", data, htmlId, htmlClass, htmlStyle, isTemplated, rule);\r\n\r\n        this.contentType = contentType;\r\n        this.contentTypeLabel = contentTypeLabel;\r\n        this.contentTypeClass = contentTypeClass;\r\n        this.html = html;\r\n        this.hasEditor = hasEditor;\r\n        this.isContainable = true;\r\n\r\n        this.getInnerText = function () {\r\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\r\n        };\r\n\r\n        // This function will be overwritten by the Content directive.\r\n        this.setHtml = function (html) {\r\n            this.html = html;\r\n            this.htmlUnsafe = html;\r\n        }\r\n\r\n        this.toObject = function () {\r\n            return {\r\n                \"type\": \"Html\"\r\n            };\r\n        };\r\n\r\n        this.toObject = function () {\r\n            var result = this.elementToObject();\r\n            result.contentType = this.contentType;\r\n            result.contentTypeLabel = this.contentTypeLabel;\r\n            result.contentTypeClass = this.contentTypeClass;\r\n            result.html = this.html;\r\n            result.hasEditor = hasEditor;\r\n            return result;\r\n        };\r\n\r\n        var getEditorObject = this.getEditorObject;\r\n        this.getEditorObject = function () {\r\n            var dto = getEditorObject();\r\n            return $.extend(dto, {\r\n                Content: this.html\r\n            });\r\n        }\r\n\r\n        this.setHtml(html);\r\n    };\r\n\r\n    LayoutEditor.Html.from = function (value) {\r\n        var result = new LayoutEditor.Html(\r\n            value.data,\r\n            value.htmlId,\r\n            value.htmlClass,\r\n            value.htmlStyle,\r\n            value.isTemplated,\r\n            value.contentType,\r\n            value.contentTypeLabel,\r\n            value.contentTypeClass,\r\n            value.html,\r\n            value.hasEditor,\r\n            value.rule);\r\n\r\n        return result;\r\n    };\r\n\r\n    LayoutEditor.registerFactory(\"Html\", function(value) { return LayoutEditor.Html.from(value); });\r\n\r\n})(jQuery, LayoutEditor || (LayoutEditor = {}));"],"sourceRoot":"/source/"} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js index de69e3feb..9f7d3923c 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js @@ -1 +1 @@ -var LayoutEditor;!function(t){Array.prototype.move=function(t,i){this.splice(i,0,this.splice(t,1)[0])},t.childrenFrom=function(i){return _(i).map(function(i){return t.elementFrom(i)})};var i=t.registerFactory=function(i,n){var e=t.factories=t.factories||{};e[i]=n};i("Canvas",function(i){return t.Canvas.from(i)}),i("Grid",function(i){return t.Grid.from(i)}),i("Row",function(i){return t.Row.from(i)}),i("Column",function(i){return t.Column.from(i)}),i("Content",function(i){return t.Content.from(i)}),t.elementFrom=function(i){var n=t.factories[i.type];if(!n)throw new Error('No element with type "'+i.type+'" was found.');var e=n(i);return e},t.setModel=function(t,i){$(t).scope().element=i},t.getModel=function(t){return $(t).scope().element}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Editor=function(i,n){this.config=i,this.canvas=t.Canvas.from(n),this.initialState=JSON.stringify(this.canvas.toObject()),this.activeElement=null,this.focusedElement=null,this.dropTargetElement=null,this.isDragging=!1,this.inlineEditingIsActive=!1,this.isResizing=!1,this.resetToolboxElements=function(){this.toolboxElements=[t.Row.from({children:[]})]},this.isDirty=function(){var t=JSON.stringify(this.canvas.toObject());return this.initialState!=t},this.resetToolboxElements(),this.canvas.setEditor(this)}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Element=function(t,i,n,e,o,s,h){if(!t)throw new Error("Parameter 'type' is required.");this.type=t,this.data=i,this.htmlId=n,this.htmlClass=e,this.htmlStyle=o,this.isTemplated=s,this.rule=h,this.editor=null,this.parent=null,this.setIsFocusedEventHandlers=[],this.setEditor=function(t){this.editor=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setEditor(t)})},this.setParent=function(t){this.parent=t,this.parent.onChildAdded(this);for(var i=t;i;)i.onDescendantAdded(this,t),i=i.parent},this.setIsTemplated=function(t){this.isTemplated=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setIsTemplated(t)})},this.applyElementEditorModel=function(){},this.getIsActive=function(){return this.editor?this.editor.activeElement===this&&!this.getIsFocused():!1},this.setIsActive=function(t){this.editor&&(this.editor.isDragging||this.editor.inlineEditingIsActive||this.editor.isResizing||(this.editor.activeElement=t?this:this.parent))},this.getIsFocused=function(){return this.editor?this.editor.focusedElement===this:!1},this.setIsFocused=function(){this.editor&&(this.children||!this.isTemplated)&&(this.editor.isDragging||this.editor.inlineEditingIsActive||this.editor.isResizing||(this.editor.focusedElement=this,_(this.setIsFocusedEventHandlers).each(function(t){try{t()}catch(i){}})))},this.getIsSelected=function(){return this.getIsFocused()?!0:this.children&&_.isArray(this.children)?_(this.children).any(function(t){return t.getIsSelected()}):!1},this.getIsDropTarget=function(){return this.editor?this.editor.dropTargetElement===this:!1},this.setIsDropTarget=function(t){this.editor&&(this.editor.dropTargetElement=t?this:null)},this.canDelete=function(){return this.isTemplated||!this.parent?!1:!0},this["delete"]=function(){this.canDelete()&&this.parent.deleteChild(this)},this.canMoveUp=function(){return this.isTemplated||!this.parent?!1:this.parent.canMoveChildUp(this)},this.moveUp=function(){this.canMoveUp()&&this.parent.moveChildUp(this)},this.canMoveDown=function(){return this.isTemplated||!this.parent?!1:this.parent.canMoveChildDown(this)},this.moveDown=function(){this.canMoveDown()&&this.parent.moveChildDown(this)},this.elementToObject=function(){return{type:this.type,data:this.data,htmlId:this.htmlId,htmlClass:this.htmlClass,htmlStyle:this.htmlStyle,isTemplated:this.isTemplated,rule:this.rule}},this.getEditorObject=function(){return{}},this.copy=function(t){var i=this.getInnerText();t.setData("text/plain",i);var n=this.toObject(),e=JSON.stringify(n,null," ");t.setData("text/json",e)},this.cut=function(t){this.canDelete()&&(this.copy(t),this["delete"]())},this.paste=function(t){this.parent&&this.parent.paste(t)}}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Container=function(i,n){this.allowedChildTypes=i,this.children=n,this.isContainer=!0;var e=this;this.onChildAdded=function(t){},this.onDescendantAdded=function(t,i){},this.setChildren=function(t){this.children=t,_(this.children).each(function(t){t.setParent(e)})},this.setChildren(n),this.getIsSealed=function(){return _(this.children).any(function(t){return t.isTemplated})};var o=this.setIsFocused;this.setIsFocused=function(){this.getIsSealed()||o.call(this)},this.addChild=function(t){_(this.children).contains(t)||!_(this.allowedChildTypes).contains(t.type)&&!t.isContainable||this.children.push(t),t.setEditor(this.editor),t.setIsTemplated(!1),t.setParent(this)},this.deleteChild=function(t){var i=_(this.children).indexOf(t);i>=0&&(this.children.splice(i,1),t.getIsActive()&&(this.editor.activeElement=null),t.getIsFocused()&&(this.children.length>i?this.children[i].setIsFocused():i>0?this.children[i-1].setIsFocused():this.setIsFocused()))},this.moveFocusPrevChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i>0&&this.children[i-1].setIsFocused()}},this.moveFocusNextChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i0},this.canMoveChildDown=function(t){var i=_(this.children).indexOf(t);return i0?t.width<12:0>e?t.width>1:!1}if(0==t)return!0;var e=t;if(0>e){var o=12-c();e+=o,e>0&&(e=0)}for(;0>e&&_(d.children).any(function(t){return t.offset>0});)for(i=0;ie;i++){var s=d.children[i];s.offset>0&&(s.offset--,e++)}for(;0!=e&&_(d.children).any(n);)for(i=0;i=0?t.width>1:!1},this.contractColumnRight=function(t,i){if(this.canContractColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0&&t.width>1&&(t.width--,this.children.length>n+1)){var e=this.children[n+1];i&&0==e.offset?e.width++:e.offset++}}},this.canExpandColumnRight=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(this.children.length>n+1){var e=this.children[n+1];return i&&0==e.offset?e.width>1:e.offset>0}return c()<12}return!1},this.expandColumnRight=function(t,i){if(this.canExpandColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(this.children.length>n+1){var e=this.children[n+1];i&&0==e.offset?e.width--:e.offset--}t.width++}}},this.canExpandColumnLeft=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(n>0){var e=this.children[n-1];if(i&&0==t.offset)return e.width>1}return t.offset>0}return!1},this.expandColumnLeft=function(t,i){if(this.canExpandColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width--:t.offset--}else t.offset--;t.width++}}},this.canContractColumnLeft=function(t,i){var n=_(this.children).indexOf(t);return n>=0?t.width>1:!1},this.contractColumnLeft=function(t,i){if(this.canContractColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width++:t.offset++}else t.offset++;t.width--}}},this.evenColumns=function(){if(0!=this.children.length){var t=Math.floor(12/this.children.length);_(this.children).each(function(i){i.width=t,i.offset=0});var i=12%this.children.length;i>0&&a(i)}};var m=this.pasteChild;this.pasteChild=function(t){"Column"==t.type?this.beginAddColumn(t.width)&&(this.commitAddColumn(),m.call(this,t)):this.parent&&this.parent.pasteChild(t)},this.toObject=function(){var t=this.elementToObject();return t.children=this.childrenToObject(),t}},t.Row.from=function(i){var n=new t.Row(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.rule,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Column=function(i,n,e,o,s,h,r,l,c,a){t.Element.call(this,"Column",i,n,e,o,s,c),t.Container.call(this,["Grid","Content"],a),this.width=h,this.offset=r,this.collapsible=l;var d=!1,u=0,f=0;this.beginChange=function(){if(d)throw new Error("Column already has a pending change.");d=!0,u=this.width,f=this.offset},this.commitChange=function(){if(!d)throw new Error("Column has no pending change.");u=0,f=0,d=!1},this.rollbackChange=function(){if(!d)throw new Error("Column has no pending change.");this.width=u,this.offset=f,u=0,f=0,d=!1},this.canSplit=function(){return this.isTemplated?!1:this.width>1},this.split=function(){if(this.canSplit()){var i=Math.floor(this.width/2),n=t.Column.from({data:null,htmlId:null,htmlClass:null,htmlStyle:null,width:i,offset:0,children:[]});this.width=this.width-i,this.parent.insertChild(n,this),n.setIsFocused()}},this.canContractRight=function(t){return this.isTemplated?!1:this.parent.canContractColumnRight(this,t)},this.contractRight=function(t){this.canContractRight(t)&&this.parent.contractColumnRight(this,t)},this.canExpandRight=function(t){return this.isTemplated?!1:this.parent.canExpandColumnRight(this,t)},this.expandRight=function(t){this.canExpandRight(t)&&this.parent.expandColumnRight(this,t)},this.canExpandLeft=function(t){return this.isTemplated?!1:this.parent.canExpandColumnLeft(this,t)},this.expandLeft=function(t){this.canExpandLeft(t)&&this.parent.expandColumnLeft(this,t)},this.canContractLeft=function(t){return this.isTemplated?!1:this.parent.canContractColumnLeft(this,t)},this.contractLeft=function(t){this.canContractLeft(t)&&this.parent.contractColumnLeft(this,t)},this.toObject=function(){var t=this.elementToObject();return t.width=this.width,t.offset=this.offset,t.collapsible=this.collapsible,t.children=this.childrenToObject(),t}},t.Column.from=function(i){var n=new t.Column(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.width,i.offset,i.collapsible,i.rule,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n},t.Column.times=function(i){return _.times(i,function(n){return t.Column.from({data:null,htmlId:null,htmlClass:null,isTemplated:!1,width:12/i,offset:0,collapsible:null,children:[]})})}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Content=function(i,n,e,o,s,h,r,l,c,a,d){t.Element.call(this,"Content",i,n,e,o,s,d),this.contentType=h,this.contentTypeLabel=r,this.contentTypeClass=l,this.html=c,this.hasEditor=a,this.getInnerText=function(){return $($.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Content"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=a,t},this.setHtml(c)},t.Content.from=function(i){var n=new t.Content(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.contentType,i.contentTypeLabel,i.contentTypeClass,i.html,i.hasEditor,i.rule);return n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t,i){i.Html=function(n,e,o,s,h,r,l,c,a,d,u){i.Element.call(this,"Html",n,e,o,s,h,u),this.contentType=r,this.contentTypeLabel=l,this.contentTypeClass=c,this.html=a,this.hasEditor=d,this.isContainable=!0,this.getInnerText=function(){return t(t.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Html"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=d,t};var f=this.getEditorObject;this.getEditorObject=function(){var i=f();return t.extend(i,{Content:this.html})},this.setHtml(a)},i.Html.from=function(t){var n=new i.Html(t.data,t.htmlId,t.htmlClass,t.htmlStyle,t.isTemplated,t.contentType,t.contentTypeLabel,t.contentTypeClass,t.html,t.hasEditor,t.rule);return n},i.registerFactory("Html",function(t){return i.Html.from(t)})}(jQuery,LayoutEditor||(LayoutEditor={})); \ No newline at end of file +var LayoutEditor;!function(t){Array.prototype.move=function(t,i){this.splice(i,0,this.splice(t,1)[0])},t.childrenFrom=function(i){return _(i).map(function(i){return t.elementFrom(i)})};var i=t.registerFactory=function(i,n){var e=t.factories=t.factories||{};e[i]=n};i("Canvas",function(i){return t.Canvas.from(i)}),i("Grid",function(i){return t.Grid.from(i)}),i("Row",function(i){return t.Row.from(i)}),i("Column",function(i){return t.Column.from(i)}),i("Content",function(i){return t.Content.from(i)}),t.elementFrom=function(i){var n=t.factories[i.type];if(!n)throw new Error('No element with type "'+i.type+'" was found.');var e=n(i);return e},t.setModel=function(t,i){$(t).scope().element=i},t.getModel=function(t){return $(t).scope().element}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Editor=function(i,n){this.config=i,this.canvas=t.Canvas.from(n),this.initialState=JSON.stringify(this.canvas.toObject()),this.activeElement=null,this.focusedElement=null,this.dropTargetElement=null,this.isDragging=!1,this.isResizing=!1,this.resetToolboxElements=function(){this.toolboxElements=[t.Row.from({children:[]})]},this.isDirty=function(){var t=JSON.stringify(this.canvas.toObject());return this.initialState!=t},this.resetToolboxElements(),this.canvas.setEditor(this)}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Element=function(t,i,n,e,o,s,h){if(!t)throw new Error("Parameter 'type' is required.");this.type=t,this.data=i,this.htmlId=n,this.htmlClass=e,this.htmlStyle=o,this.isTemplated=s,this.rule=h,this.editor=null,this.parent=null,this.setIsFocusedEventHandlers=[],this.setEditor=function(t){this.editor=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setEditor(t)})},this.setParent=function(t){this.parent=t,this.parent.onChildAdded(this);for(var i=t;i;)i.onDescendantAdded(this,t),i=i.parent},this.setIsTemplated=function(t){this.isTemplated=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setIsTemplated(t)})},this.applyElementEditorModel=function(){},this.getIsActive=function(){return this.editor?this.editor.activeElement===this&&!this.getIsFocused():!1},this.setIsActive=function(t){this.editor&&(this.editor.isDragging||this.editor.isResizing||(this.editor.activeElement=t?this:this.parent))},this.getIsFocused=function(){return this.editor?this.editor.focusedElement===this:!1},this.setIsFocused=function(){this.editor&&(this.children||!this.isTemplated)&&(this.editor.isDragging||this.editor.isResizing||(this.editor.focusedElement=this,_(this.setIsFocusedEventHandlers).each(function(t){try{t()}catch(i){}})))},this.getIsSelected=function(){return this.getIsFocused()?!0:this.children&&_.isArray(this.children)?_(this.children).any(function(t){return t.getIsSelected()}):!1},this.getIsDropTarget=function(){return this.editor?this.editor.dropTargetElement===this:!1},this.setIsDropTarget=function(t){this.editor&&(this.editor.dropTargetElement=t?this:null)},this.canDelete=function(){return this.isTemplated||!this.parent?!1:!0},this["delete"]=function(){this.canDelete()&&this.parent.deleteChild(this)},this.canMoveUp=function(){return this.isTemplated||!this.parent?!1:this.parent.canMoveChildUp(this)},this.moveUp=function(){this.canMoveUp()&&this.parent.moveChildUp(this)},this.canMoveDown=function(){return this.isTemplated||!this.parent?!1:this.parent.canMoveChildDown(this)},this.moveDown=function(){this.canMoveDown()&&this.parent.moveChildDown(this)},this.elementToObject=function(){return{type:this.type,data:this.data,htmlId:this.htmlId,htmlClass:this.htmlClass,htmlStyle:this.htmlStyle,isTemplated:this.isTemplated,rule:this.rule}},this.getEditorObject=function(){return{}},this.copy=function(t){var i=this.getInnerText();t.setData("text/plain",i);var n=this.toObject(),e=JSON.stringify(n,null," ");t.setData("text/json",e)},this.cut=function(t){this.canDelete()&&(this.copy(t),this["delete"]())},this.paste=function(t){this.parent&&this.parent.paste(t)}}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Container=function(i,n){this.allowedChildTypes=i,this.children=n,this.isContainer=!0;var e=this;this.onChildAdded=function(t){},this.onDescendantAdded=function(t,i){},this.setChildren=function(t){this.children=t,_(this.children).each(function(t){t.setParent(e)})},this.setChildren(n),this.getIsSealed=function(){return _(this.children).any(function(t){return t.isTemplated})};var o=this.setIsFocused;this.setIsFocused=function(){this.getIsSealed()||o.call(this)},this.addChild=function(t){_(this.children).contains(t)||!_(this.allowedChildTypes).contains(t.type)&&!t.isContainable||this.children.push(t),t.setEditor(this.editor),t.setIsTemplated(!1),t.setParent(this)},this.deleteChild=function(t){var i=_(this.children).indexOf(t);i>=0&&(this.children.splice(i,1),t.getIsActive()&&(this.editor.activeElement=null),t.getIsFocused()&&(this.children.length>i?this.children[i].setIsFocused():i>0?this.children[i-1].setIsFocused():this.setIsFocused()))},this.moveFocusPrevChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i>0&&this.children[i-1].setIsFocused()}},this.moveFocusNextChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i0},this.canMoveChildDown=function(t){var i=_(this.children).indexOf(t);return i0?t.width<12:0>e?t.width>1:!1}if(0==t)return!0;var e=t;if(0>e){var o=12-c();e+=o,e>0&&(e=0)}for(;0>e&&_(d.children).any(function(t){return t.offset>0});)for(i=0;ie;i++){var s=d.children[i];s.offset>0&&(s.offset--,e++)}for(;0!=e&&_(d.children).any(n);)for(i=0;i=0?t.width>1:!1},this.contractColumnRight=function(t,i){if(this.canContractColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0&&t.width>1&&(t.width--,this.children.length>n+1)){var e=this.children[n+1];i&&0==e.offset?e.width++:e.offset++}}},this.canExpandColumnRight=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(this.children.length>n+1){var e=this.children[n+1];return i&&0==e.offset?e.width>1:e.offset>0}return c()<12}return!1},this.expandColumnRight=function(t,i){if(this.canExpandColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(this.children.length>n+1){var e=this.children[n+1];i&&0==e.offset?e.width--:e.offset--}t.width++}}},this.canExpandColumnLeft=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(n>0){var e=this.children[n-1];if(i&&0==t.offset)return e.width>1}return t.offset>0}return!1},this.expandColumnLeft=function(t,i){if(this.canExpandColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width--:t.offset--}else t.offset--;t.width++}}},this.canContractColumnLeft=function(t,i){var n=_(this.children).indexOf(t);return n>=0?t.width>1:!1},this.contractColumnLeft=function(t,i){if(this.canContractColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width++:t.offset++}else t.offset++;t.width--}}},this.evenColumns=function(){if(0!=this.children.length){var t=Math.floor(12/this.children.length);_(this.children).each(function(i){i.width=t,i.offset=0});var i=12%this.children.length;i>0&&a(i)}};var m=this.pasteChild;this.pasteChild=function(t){"Column"==t.type?this.beginAddColumn(t.width)&&(this.commitAddColumn(),m.call(this,t)):this.parent&&this.parent.pasteChild(t)},this.toObject=function(){var t=this.elementToObject();return t.children=this.childrenToObject(),t}},t.Row.from=function(i){var n=new t.Row(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.rule,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Column=function(i,n,e,o,s,h,r,l,c,a){t.Element.call(this,"Column",i,n,e,o,s,c),t.Container.call(this,["Grid","Content"],a),this.width=h,this.offset=r,this.collapsible=l;var d=!1,u=0,f=0;this.beginChange=function(){if(d)throw new Error("Column already has a pending change.");d=!0,u=this.width,f=this.offset},this.commitChange=function(){if(!d)throw new Error("Column has no pending change.");u=0,f=0,d=!1},this.rollbackChange=function(){if(!d)throw new Error("Column has no pending change.");this.width=u,this.offset=f,u=0,f=0,d=!1},this.canSplit=function(){return this.isTemplated?!1:this.width>1},this.split=function(){if(this.canSplit()){var i=Math.floor(this.width/2),n=t.Column.from({data:null,htmlId:null,htmlClass:null,htmlStyle:null,width:i,offset:0,children:[]});this.width=this.width-i,this.parent.insertChild(n,this),n.setIsFocused()}},this.canContractRight=function(t){return this.isTemplated?!1:this.parent.canContractColumnRight(this,t)},this.contractRight=function(t){this.canContractRight(t)&&this.parent.contractColumnRight(this,t)},this.canExpandRight=function(t){return this.isTemplated?!1:this.parent.canExpandColumnRight(this,t)},this.expandRight=function(t){this.canExpandRight(t)&&this.parent.expandColumnRight(this,t)},this.canExpandLeft=function(t){return this.isTemplated?!1:this.parent.canExpandColumnLeft(this,t)},this.expandLeft=function(t){this.canExpandLeft(t)&&this.parent.expandColumnLeft(this,t)},this.canContractLeft=function(t){return this.isTemplated?!1:this.parent.canContractColumnLeft(this,t)},this.contractLeft=function(t){this.canContractLeft(t)&&this.parent.contractColumnLeft(this,t)},this.toObject=function(){var t=this.elementToObject();return t.width=this.width,t.offset=this.offset,t.collapsible=this.collapsible,t.children=this.childrenToObject(),t}},t.Column.from=function(i){var n=new t.Column(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.width,i.offset,i.collapsible,i.rule,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n},t.Column.times=function(i){return _.times(i,function(n){return t.Column.from({data:null,htmlId:null,htmlClass:null,isTemplated:!1,width:12/i,offset:0,collapsible:null,children:[]})})}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Content=function(i,n,e,o,s,h,r,l,c,a,d){t.Element.call(this,"Content",i,n,e,o,s,d),this.contentType=h,this.contentTypeLabel=r,this.contentTypeClass=l,this.html=c,this.hasEditor=a,this.getInnerText=function(){return $($.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Content"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=a,t},this.setHtml(c)},t.Content.from=function(i){var n=new t.Content(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.contentType,i.contentTypeLabel,i.contentTypeClass,i.html,i.hasEditor,i.rule);return n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t,i){i.Html=function(n,e,o,s,h,r,l,c,a,d,u){i.Element.call(this,"Html",n,e,o,s,h,u),this.contentType=r,this.contentTypeLabel=l,this.contentTypeClass=c,this.html=a,this.hasEditor=d,this.isContainable=!0,this.getInnerText=function(){return t(t.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Html"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=d,t};var f=this.getEditorObject;this.getEditorObject=function(){var i=f();return t.extend(i,{Content:this.html})},this.setHtml(a)},i.Html.from=function(t){var n=new i.Html(t.data,t.htmlId,t.htmlClass,t.htmlStyle,t.isTemplated,t.contentType,t.contentTypeLabel,t.contentTypeClass,t.html,t.hasEditor,t.rule);return n},i.registerFactory("Html",function(t){return i.Html.from(t)})}(jQuery,LayoutEditor||(LayoutEditor={})); \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementFactory.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementFactory.cs index 4793ce28b..e58930309 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementFactory.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementFactory.cs @@ -5,16 +5,19 @@ using Orchard.Localization; namespace Orchard.Layouts.Services { public class ElementFactory : IElementFactory { private readonly IElementEventHandler _elementEventHandler; + private readonly IWorkContextAccessor _workContextAccessor; - public ElementFactory(IElementEventHandler elementEventHandler) { + public ElementFactory(IElementEventHandler elementEventHandler, IWorkContextAccessor workContextAccessor) { _elementEventHandler = elementEventHandler; + _workContextAccessor = workContextAccessor; T = NullLocalizer.Instance; } public Localizer T { get; set; } public Element Activate(Type elementType, Action initialize = null) { - var element = (Element)Activator.CreateInstance(elementType); + var workContext = _workContextAccessor.GetContext(); + var element = (Element)workContext.Resolve(elementType); if (initialize != null) initialize(element); @@ -23,7 +26,8 @@ namespace Orchard.Layouts.Services { } public T Activate(Action initialize = null) where T : Element { - var element = (T)Activator.CreateInstance(typeof(T)); + var workContext = _workContextAccessor.GetContext(); + var element = workContext.Resolve(); if (initialize != null) initialize(element); diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementManager.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementManager.cs index fa535ef78..1ff94d067 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementManager.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Services/ElementManager.cs @@ -38,7 +38,7 @@ namespace Orchard.Layouts.Services { public IEnumerable DescribeElements(DescribeElementsContext context) { var contentType = context.Content != null ? context.Content.ContentItem.ContentType : default(string); var cacheKey = String.Format("LayoutElementTypes-{0}-{1}", contentType ?? "AnyType", context.CacheVaryParam); - return _cacheManager.Get(cacheKey, acquireContext => { + return _cacheManager.Get(cacheKey, true, acquireContext => { var harvesterContext = new HarvestElementsContext { Content = context.Content }; @@ -55,7 +55,7 @@ namespace Orchard.Layouts.Services { public IEnumerable GetCategories(DescribeElementsContext context) { var contentType = context.Content != null ? context.Content.ContentItem.ContentType : default(string); - return _cacheManager.Get(String.Format("ElementCategories-{0}-{1}", contentType ?? "AnyType", context.CacheVaryParam), acquireContext => { + return _cacheManager.Get(String.Format("ElementCategories-{0}-{1}", contentType ?? "AnyType", context.CacheVaryParam), true, acquireContext => { var elements = DescribeElements(context); var categoryDictionary = GetCategories(); var categoryDescriptorDictionary = new Dictionary(); diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Settings/LayoutSettingsHooks.cs b/src/Orchard.Web/Modules/Orchard.Layouts/Settings/LayoutSettingsHooks.cs index fb7804c2d..c39e4fe42 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Settings/LayoutSettingsHooks.cs +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Settings/LayoutSettingsHooks.cs @@ -22,12 +22,7 @@ namespace Orchard.Layouts.Settings { yield break; var model = definition.Settings.GetModel(); - - if (String.IsNullOrWhiteSpace(model.DefaultLayoutData)) { - var defaultData = _serializer.Serialize(_layoutManager.CreateDefaultLayout()); - model.DefaultLayoutData = defaultData; - } - + yield return DefinitionTemplate(model); } diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Styles/LayoutEditor.css b/src/Orchard.Web/Modules/Orchard.Layouts/Styles/LayoutEditor.css index 56edf88f7..04dd49724 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Styles/LayoutEditor.css +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Styles/LayoutEditor.css @@ -1114,4 +1114,4 @@ table[rules=all i] > tfoot > tr > td, table[rules=all i] > tfoot > tr > th { font-size: 0.9em; } -/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Editor.less","LayoutEditor.css","Element.less","Container.less","Canvas.less","Row.less","Column.less","Content.less","Reset.less","Toolbox.less","Popup.less"],"names":[],"mappings":"AAEA;EACI,sBAAA;EAAA,qBAAA;EAAA,cAAA;EACA,uCAAA;MAAA,uBAAA;UAAA,+BAAA;EACA,mBAAA;EACA,UAAA;CCDH;ADHD;EAOQ,sBAAA;EAAA,qBAAA;EAAA,cAAA;CCDP;ADND;EAUY,kBAAA;CCDX;ADMD;EACI,sBAAA;EAAA,qBAAA;EAAA,cAAA;EACA,gBAAA;EACA,gBAAA;EACA,6BAAA;MAAA,wBAAA;UAAA,qBAAA;CCJH;ADAD;EAOQ,qBAAA;MAAA,qBAAA;UAAA,aAAA;EACA,0BAAA;EACA,0BAAA;CCJP;ADLD;EAYY,cAAA;EACA,oBAAA;EACA,iBAAA;CCJX;ADVD;EAiBgB,uBAAA;CCJf;ADbD;EAsBY,wBAAA;EACA,qBAAA;EACA,oBAAA;CCNX;ADlBD;EA6BQ,oBAAA;CCRP;ADYD;EACI,cAAA;CCVH;ADaO;;EACI,aAAA;EACA,eAAA;CCVX;ADaO;EACI,YAAA;CCXX;ADCD;;EAcY,gBAAA;CCXX;ADHD;EAkBY,uBAAA;EACA,YAAA;EACA,WAAA;CCZX;ADcW;EACI,oBAAA;EACA,YAAA;CCZf;ADZD;EA6BY,gBAAA;CCdX;ADfD;EAkCQ,mBAAA;EACA,0BAAA;EACA,iBAAA;EACA,uBAAA;CChBP;ADrBD;EAyCQ,qBAAA;EACA,mBAAA;CCjBP;ADzBD;EA8CQ,oBAAA;CClBP;;AC5ED;EAEQ,mBAAA;EACA,cAAA;EACA,gBAAA;EACA,iBAAA;EACA,WAAA;CD8EP;AC3EO;EACI,eAAA;CD6EX;ACvFD;EAegB,cAAA;EACA,mBAAA;EACA,UAAA;EACA,YAAA;EACA,aAAA;EACA,eAAA;EACA,iBAAA;EACA,oBAAA;EACA,kBAAA;EACA,uBAAA;CD2Ef;ACnGD;EA2BoB,sBAAA;EACA,aAAA;EACA,mBAAA;CD2EnB;ACxGD;EAiCoB,gBAAA;CD0EnB;AC3GD;EAqCoB,cAAA;EACA,YAAA;EACA,gBAAA;EACA,mBAAA;CDyEnB;ACjHD;EA6CgB,WAAA;EACA,WAAA;CDuEf;ACrHD;EAqDQ,iBAAA;CDmEP;AChEG;;EAGQ,kBAAA;EACA,oBAAA;CDiEX;ACrEG;;EAOY,aAAA;CDkEf;ACzEG;EAcQ,uCAAA;CD8DX;AC5EG;EAiBY,eAAA;EACA,YAAA;EACA,2CAAA;CD8Df;ACjFG;EA2BQ,sBAAA;CDyDX;ACpFG;EA8BY,eAAA;EACA,0BAAA;EACA,eAAA;CDyDf;ACzFG;EAmCgB,sBAAA;CDyDnB;ACxDmB;EACI,0BAAA;CD0DvB;ACvDmB;EACI,gBAAA;EACA,gCAAA;CDyDvB;ACvDuB;EACI,0BAAA;CDyD3B;ACrDmB;EACI,eAAA;EACA,0BAAA;CDuDvB;ACrDuB;EACI,0BAAA;CDuD3B;AC7GG;EA+DQ,2CAAA;CDiDX;ACxKD;EA4HQ,cAAA;CD+CP;AC3KD;EAiIQ,uDAAA;CD6CP;AC9KD;EAoIY,eAAA;EACA,+BAAA;EACA,iBAAA;EACA,sCAAA;EACA,2CAAA;CD6CX;ACrLD;EA+IY,gBAAA;EACA,iBAAA;CDyCX;;AEzLD;EAMgB,cAAA;EACA,+BAAA;MAAA,2BAAA;UAAA,uBAAA;EACA,aAAA;EACA,iBAAA;EACA,4CAAA;EACA,mBAAA;EACA,cAAA;EACA,gCAAA;MAAA,sBAAA;UAAA,wBAAA;EACA,gBAAA;EACA,mBAAA;EACA,aAAA;EACA,mBAAA;CFuLf;AExMD;EAqBgB,cAAA;CFsLf;AE3MD;EAyBoB,iBAAA;CFqLnB;AE9MD;EA8BoB,cAAA;CFmLnB;AE/KW;EAEQ,sBAAA;EAAA,qBAAA;EAAA,cAAA;CFgLnB;AElLW;EAMQ,mBAAA;EACA,OAAA;EACA,SAAA;EACA,UAAA;EACA,QAAA;CF+KnB;AEzKG;EAGY,4CAAA;CFyKf;AE5KG;EAMgB,aAAA;CFyKnB;AE/KG;EAWY,cAAA;CFuKf;AElLG;EAcgB,cAAA;CFuKnB;;AGvOD;EAGY,cAAA;CHwOX;;AIzOO;EAEQ,2DAAA;CJ2Of;AI7OO;EAIY,4BAAA;CJ4OnB;AIhPO;EASQ,0BAAA;CJ0Of;AInPO;EAWY,0BAAA;CJ2OnB;;AKxPD;EAIgB,cAAA;EACA,mBAAA;EACA,YAAA;EACA,OAAA;EACA,YAAA;EACA,aAAA;EACA,mBAAA;CLwPf;AKlQD;EAcgB,WAAA;CLuPf;AKrQD;EAkBgB,YAAA;CLsPf;AKlPO;EACI,eAAA;CLoPX;;AM1QD;;EAEQ,gBAAA;CN6QP;AM/QD;;EAOgB,aAAA;EACA,mBAAA;EACA,oBAAA;EN4Qd,yFAAyF;EACzF,qCAAqC;EACrC,qCAAqC;EACrC,uDAAuD;EACvD,wDAAwD;EACxD;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BC;CACF;AMrTD;;ECAQ,UAAA;EACA,WAAA;EACA,wBAAA;CPyTP;AM3TD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECOQ,eAAA;CPsVP;AM7VD;;;;;;;;;;;;;;ECWQ,gBAAA;EACA,mBAAA;CPkWP;AM9WD;;;;ECgBQ,kBAAA;EACA,mBAAA;CPoWP;AMrXD;;ECqBQ,mBAAA;CPoWP;AMzXD;;;;;;;;ECyBQ,uBAAA;EACA,iBAAA;CP0WP;AMpYD;;;;;;;;;;EC8BQ,mBAAA;CPkXP;AMhZD;;;;ECkCQ,oBAAA;CPoXP;AMtZD;;;;;;;;ECsCQ,uBAAA;CP0XP;AMhaD;;EC0CQ,kBAAA;CP0XP;AMpaD;;EC8CQ,mBAAA;CP0XP;AMxaD;;ECkDQ,oBAAA;CP0XP;AM5aD;;ECsDQ,sBAAA;CP0XP;AMhbD;;;;EC0DQ,oBAAA;EACA,mBAAA;CP4XP;AMvbD;;ECgEQ,cAAA;CP2XP;AM3bD;;ECoEQ,mBAAA;EACA,oBAAA;CP2XP;AMhcD;;ECyEQ,mBAAA;EACA,oBAAA;EACA,eAAA;EACA,8BAAA;EACA,4BAAA;UAAA,oBAAA;CP2XP;AMxcD;;ECiFQ,6BAAA;CP2XP;AM5cD;;ECqFQ,6BAAA;CP2XP;AMhdD;;;;;;;;;;ECyFQ,sBAAA;CPmYP;AM5dD;;EC8FQ,eAAA;CPkYP;AMheD;;ECkGQ,eAAA;CPkYP;AMpeD;;;;ECsGQ,2BAAA;CPoYP;AM1eD;;;;;;;;EC2GQ,aAAA;CPyYP;AMpfD;;EC+GQ,cAAA;CPyYP;AMxfD;;ECmHQ,mBAAA;EACA,aAAA;CPyYP;AM7fD;;;;ECwHQ,kCAAA;CP2YP;AMngBD;;;;EC4HQ,2BAAA;CP6YP;AMzgBD;;;;;;ECgIQ,8BAAA;CPiZP;AMjhBD;;ECoIQ,uBAAA;CPiZP;AMrhBD;;ECwIQ,oBAAA;CPiZP;AMzhBD;;EC4IQ,qBAAA;CPiZP;AM7hBD;;ECgJQ,cAAA;EACA,iBAAA;CPiZP;AMliBD;;ECqJQ,oBAAA;CPiZP;AMtiBD;;ECyJQ,iBAAA;CPiZP;AM1iBD;;EC6JQ,oBAAA;CPiZP;AM9iBD;;;;;;;;;;;;;;;;;;;;;;ECiKQ,eAAA;CPqaP;AMtkBD;;ECqKQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AM7kBD;;EC4KQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMplBD;;ECmLQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AM3lBD;;EC0LQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMlmBD;;ECiMQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMzmBD;;ECwMQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMhnBD;;;;;;;;;;;;EC+MQ,eAAA;CP+aP;AM9nBD;;ECmNQ,mBAAA;CP+aP;AMloBD;;;;;;;;ECuNQ,gBAAA;EACA,mBAAA;CPqbP;AM7oBD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EC+NY,cAAA;EACA,iBAAA;CPgdX;AMhrBD;;ECoOQ,kBAAA;CPgdP;AMprBD;;;;;;ECwOQ,mBAAA;CPodP;AM5rBD;;EC4OQ,yBAAA;CPodP;AMhsBD;;;;ECgPQ,sBAAA;CPsdP;AMtsBD;;;;;;;;;;;;ECsPY,wBAAA;CP8dX;AMptBD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECkQgB,wBAAA;CPwff;AM1vBD;;ECsQQ,eAAA;CPwfP;AM9vBD;;EC0QQ,uBAAA;CPwfP;AMlwBD;;;;EC8QQ,4BAAA;CP0fP;AMxwBD;;;;ECkRQ,sBAAA;CP4fP;AM9wBD;;;;ECsRQ,4BAAA;CP8fP;AMpxBD;;;;EC0RQ,yBAAA;CPggBP;AM1xBD;;;;EC8RQ,4BAAA;CPkgBP;AMhyBD;;;;ECkSQ,mBAAA;CPogBP;AMtyBD;;;;;;;;ECsSQ,oBAAA;CP0gBP;AMhzBD;;;;;;;;;;;;;;;;EC2SY,qBAAA;CPuhBX;AMl0BD;;EC+SQ,uBAAA;EACA,oBAAA;EACA,0BAAA;EACA,qBAAA;CPuhBP;AMz0BD;;;;ECsTQ,aAAA;CPyhBP;AM/0BD;;EC0TQ,kBAAA;CPyhBP;AMn1BD;;;;;;;;EC8TQ,uBAAA;CP+hBP;AM71BD;;;;;;ECkUQ,wBAAA;CPmiBP;AMr2BD;;;;;;ECsUQ,mBAAA;CPuiBP;AM72BD;;;;;;;;EC0UQ,sBAAA;CP6iBP;AMv3BD;;;;;;;;;;;;;;ECwWQ,qBAAA;CP+hBP;AMv4BD;;EC4WQ,sBAAA;CP+hBP;AM34BD;;;;;;;;;;;;;;ECiXQ,uBAAA;CP0iBP;AM35BD;;;;ECqXQ,qBAAA;CP4iBP;AMj6BD;;ECyXQ,YAAA;EACA,oBAAA;EACA,kBAAA;EACA,mBAAA;CP4iBP;AMx6BD;;ECgYQ,iBAAA;EACA,kBAAA;EACA,8BAAA;EACA,+BAAA;CP4iBP;AM/6BD;;ECuYQ,kBAAA;EACA,mBAAA;CP4iBP;AMp7BD;;EAeoB,yBAAA;CNy6BnB;AMx7BD;;EAmBoB,4BAAA;CNy6BnB;AMp6BO;;;;EAEQ,eAAA;EACA,YAAA;EACA,gBAAA;EACA,aAAA;CNw6Bf;AMr8BD;;EAmCQ,eAAA;EACA,YAAA;EACA,gBAAA;EACA,aAAA;CNs6BP;;AQ78BD;EAEQ,mBAAA;EACA,kBAAA;EACA,aAAA;EACA,uBAAA;MAAA,qBAAA;UAAA,eAAA;CR+8BP;AQp9BD;EAQY,0BAAA;EACA,aAAA;EACA,kBAAA;EACA,aAAA;EACA,0BAAA;CR+8BX;AQ78BW;EACI,gBAAA;EACA,OAAA;EACA,iBAAA;EACA,iBAAA;CR+8Bf;AQ58BW;EACI,mBAAA;EACA,UAAA;CR88Bf;AQr+BD;EA2BgB,iBAAA;CR68Bf;AQx+BD;EA8BoB,eAAA;EACA,mBAAA;EACA,sBAAA;CR68BnB;AQ38BmB;EACI,sBAAA;EACA,YAAA;EACA,kBAAA;EACA,8CAAA;EACA,mBAAA;EACA,iBAAA;CR68BvB;AQz8Be;EAEQ,iBAAA;CR08BvB;AQ58Be;EAMQ,cAAA;CRy8BvB;AQ3/BD;EAuDoB,gBAAA;CRu8BnB;AQ9/BD;EA2DoB,0BAAA;EACA,uBAAA;EACA,kBAAA;EACA,gBAAA;CRs8BnB;AQpgCD;EAiEwB,sBAAA;EACA,YAAA;EACA,8CAAA;CRs8BvB;AQzgCD;EAuEwB,gBAAA;CRq8BvB;AQ5gCD;EA4EoB,gBAAA;CRm8BnB;;AS/gCD;EAEQ,cAAA;EACA,mBAAA;EACA,UAAA;EAGA,iDAAA;EACA,0BAAA;EACA,eAAA;EACA,0BAAA;EACA,iBAAA;EACA,YAAA;EACA,eAAA;EACA,iBAAA;EACA,gBAAA;EACA,oBAAA;EACA,oBAAA;EACA,iBAAA;CTihCP;AS/gCO;EACI,aAAA;CTihCX;AStiCD;EAyBY,sBAAA;EAAA,qBAAA;EAAA,cAAA;EACA,iBAAA;CTghCX;AS1iCD;EA6BgB,iBAAA;EACA,iDAAA;EACA,kBAAA;CTghCf;AS/iCD;EAoCY,iBAAA;CT8gCX;ASljCD;EAwCY,iBAAA;EACA,kBAAA;EACA,eAAA;CT6gCX;ASvjCD;EA8CY,0BAAA;EACA,gBAAA;CT4gCX;AS3jCD;;EAoDgB,YAAA;CT2gCf;AS/jCD;EAwDgB,eAAA;EACA,iBAAA;CT0gCf","file":"LayoutEditor.css","sourcesContent":["@import \"Variables.less\";\r\n\r\n.layout-editor-toolbar {\r\n    display: flex;\r\n    justify-content: space-between;\r\n    position: relative;\r\n    top: 10px;\r\n\r\n    .layout-editor-toolbar-group {\r\n        display: flex;\r\n\r\n        > li + li {\r\n            margin-left: 12px;\r\n        }\r\n    }\r\n}\r\n\r\n.layout-editor {\r\n    display: flex;\r\n    margin-top: 1em;\r\n    font-size: @font-size;\r\n    align-items: stretch;\r\n\r\n    > .layout-canvas-wrapper {\r\n        flex-grow: 1;\r\n        background-color: @gray-bg;\r\n        border: 1px solid @gray-border;\r\n\r\n        > .layout-toolbar-container {\r\n            display: none;\r\n            margin: @container-padding @container-padding 0;\r\n            min-height: 71px;\r\n\r\n            > .mce-panel {\r\n                width: 100% !important;\r\n            }\r\n        }\r\n\r\n        .layout-content > .layout-element-wrapper .layout-content-markup > .layout-placeholder {\r\n            border: 1px dashed #ccc;\r\n            padding: 0.2em 0.4em;\r\n            background: #e8e8e8;\r\n        }\r\n    }\r\n\r\n    .layout-snippet {\r\n        background: #e8e8e8;\r\n    }\r\n}\r\n\r\n.layout-editor-help-dialog {\r\n    display: none;\r\n\r\n    .help-row {\r\n        &:before, &:after {\r\n            content: \" \"; // 1\r\n            display: table; // 2\r\n        }\r\n\r\n        &:after {\r\n            clear: both;\r\n        }\r\n\r\n        > .help-column-full, > .help-column-half {\r\n            margin: 0.5em 0;\r\n        }\r\n\r\n        > .help-column-half {\r\n            box-sizing: border-box;\r\n            float: left;\r\n            width: 50%;\r\n\r\n            &:nth-child(2n) {\r\n                padding-right: 10px;\r\n                clear: left;\r\n            }\r\n        }\r\n\r\n        + .help-row {\r\n            margin-top: 1em;\r\n        }\r\n    }\r\n\r\n    code {\r\n        border-radius: 4px;\r\n        background-color: #f3f4f5;\r\n        padding: 2px 4px;\r\n        font-family: monospace;\r\n    }\r\n\r\n    p {\r\n        margin-bottom: 0.5em;\r\n        line-height: 1.6em;\r\n    }\r\n\r\n    table > tbody > tr > td:first-child {\r\n        padding-right: 10px;\r\n    }\r\n}\r\n",".layout-editor-toolbar {\n  display: flex;\n  justify-content: space-between;\n  position: relative;\n  top: 10px;\n}\n.layout-editor-toolbar .layout-editor-toolbar-group {\n  display: flex;\n}\n.layout-editor-toolbar .layout-editor-toolbar-group > li + li {\n  margin-left: 12px;\n}\n.layout-editor {\n  display: flex;\n  margin-top: 1em;\n  font-size: 14px;\n  align-items: stretch;\n}\n.layout-editor > .layout-canvas-wrapper {\n  flex-grow: 1;\n  background-color: #f3f4f5;\n  border: 1px solid #e4e5e6;\n}\n.layout-editor > .layout-canvas-wrapper > .layout-toolbar-container {\n  display: none;\n  margin: 12px 12px 0;\n  min-height: 71px;\n}\n.layout-editor > .layout-canvas-wrapper > .layout-toolbar-container > .mce-panel {\n  width: 100% !important;\n}\n.layout-editor > .layout-canvas-wrapper .layout-content > .layout-element-wrapper .layout-content-markup > .layout-placeholder {\n  border: 1px dashed #ccc;\n  padding: 0.2em 0.4em;\n  background: #e8e8e8;\n}\n.layout-editor .layout-snippet {\n  background: #e8e8e8;\n}\n.layout-editor-help-dialog {\n  display: none;\n}\n.layout-editor-help-dialog .help-row:before,\n.layout-editor-help-dialog .help-row:after {\n  content: \" \";\n  display: table;\n}\n.layout-editor-help-dialog .help-row:after {\n  clear: both;\n}\n.layout-editor-help-dialog .help-row > .help-column-full,\n.layout-editor-help-dialog .help-row > .help-column-half {\n  margin: 0.5em 0;\n}\n.layout-editor-help-dialog .help-row > .help-column-half {\n  box-sizing: border-box;\n  float: left;\n  width: 50%;\n}\n.layout-editor-help-dialog .help-row > .help-column-half:nth-child(2n) {\n  padding-right: 10px;\n  clear: left;\n}\n.layout-editor-help-dialog .help-row + .help-row {\n  margin-top: 1em;\n}\n.layout-editor-help-dialog code {\n  border-radius: 4px;\n  background-color: #f3f4f5;\n  padding: 2px 4px;\n  font-family: monospace;\n}\n.layout-editor-help-dialog p {\n  margin-bottom: 0.5em;\n  line-height: 1.6em;\n}\n.layout-editor-help-dialog table > tbody > tr > td:first-child {\n  padding-right: 10px;\n}\n\n.layout-editor .layout-element {\n  position: relative;\n  margin-top: 0;\n  margin-right: 0;\n  margin-bottom: 0;\n  padding: 0;\n}\n.layout-editor .layout-element:not(.layout-column) {\n  margin-left: 0;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel {\n  display: none;\n  position: absolute;\n  margin: 0;\n  z-index: 20;\n  height: 25px;\n  padding: 0 6px;\n  list-style: none;\n  white-space: nowrap;\n  line-height: 25px;\n  vertical-align: middle;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel > .layout-panel-item {\n  display: inline-block;\n  height: 25px;\n  padding: 1px 6px 0;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel > .layout-panel-label {\n  font-size: 13px;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel > .layout-panel-action {\n  display: none;\n  width: 28px;\n  cursor: pointer;\n  text-align: center;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel-main {\n  top: -27px;\n  left: -2px;\n}\n.layout-editor li.layout-element {\n  list-style: none;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active,\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused {\n  border-width: 2px;\n  border-style: solid;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active > .layout-element-wrapper,\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper {\n  margin: -2px;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active {\n  border-color: rgba(104, 104, 104, 0.1);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active > .layout-element-wrapper > .layout-panel-main {\n  display: block;\n  z-index: 30;\n  background-color: rgba(104, 104, 104, 0.1);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused {\n  border-color: #648721;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel {\n  display: block;\n  background-color: #648721;\n  color: #fefefe;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action {\n  display: inline-block;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action:hover {\n  background-color: #82b02b;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.disabled {\n  cursor: default;\n  color: rgba(254, 254, 254, 0.4);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.disabled:hover {\n  background-color: #648721;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.active {\n  color: #deff42;\n  background-color: #739b26;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.active:hover {\n  background-color: #82b02b;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-selected {\n  background-color: rgba(100, 135, 33, 0.08);\n}\n.layout-editor .ui-sortable-placeholder {\n  display: none;\n}\n.layout-editor .layout-element-droptarget {\n  box-shadow: inset 0 0 12px 6px rgba(100, 135, 33, 0.5);\n}\n.layout-editor .layout-element-droptarget .ui-sortable-placeholder {\n  display: block;\n  visibility: visible !important;\n  min-height: 78px;\n  border: 2px dashed #648721 !important;\n  background-color: rgba(100, 135, 33, 0.16);\n}\n.layout-editor .media-thumbnail img {\n  max-width: 100%;\n  max-height: 100%;\n}\n\n.layout-editor .layout-container > .layout-element-wrapper > .layout-container-children-placeholder {\n  display: none;\n  flex-direction: column;\n  margin: 11px;\n  min-height: 80px;\n  border: 1px dashed rgba(124, 124, 124, 0.4);\n  border-radius: 4px;\n  padding: 12px;\n  justify-content: center;\n  font-size: 13px;\n  font-style: italic;\n  opacity: 0.6;\n  text-align: center;\n}\n.layout-editor .layout-container > .layout-element-wrapper > .layout-children {\n  padding: 12px;\n}\n.layout-editor .layout-container > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-container) + .layout-element:not(.layout-container) {\n  margin-top: 12px;\n}\n.layout-editor .layout-container > .layout-element-wrapper > .layout-children > .ui-sortable-helper:first-child + .layout-element:not(.layout-container) {\n  margin-top: 0;\n}\n.layout-editor .layout-container > .layout-element-wrapper.layout-container-empty > .layout-container-children-placeholder {\n  display: flex;\n}\n.layout-editor .layout-container > .layout-element-wrapper.layout-container-empty > .layout-children {\n  position: absolute;\n  top: 0;\n  right: 0;\n  bottom: 0;\n  left: 0;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) {\n  border: 1px dashed rgba(124, 124, 124, 0.6);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) > .layout-element-wrapper {\n  margin: -1px;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-container + .layout-container:not(.layout-element-active) {\n  border-top: 0;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-container + .layout-container:not(.layout-element-active) > .layout-element-wrapper {\n  margin-top: 0;\n}\n\n.layout-editor .layout-canvas #dummy {\n  display: none;\n}\n\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) {\n  border-top: 1px dashed rgba(124, 124, 124, 0.6) !important;\n}\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) > .layout-element-wrapper {\n  margin-top: -1px !important;\n}\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element + .layout-element:not(.layout-element-active) {\n  border-left: 0 !important;\n}\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element + .layout-element:not(.layout-element-active) > .layout-element-wrapper {\n  margin-left: 0 !important;\n}\n\n.layout-editor .layout-column > .layout-element-wrapper > .layout-column-resize-bar {\n  display: none;\n  position: absolute;\n  z-index: 30;\n  top: 0;\n  width: 16px;\n  height: 100%;\n  cursor: col-resize;\n}\n.layout-editor .layout-column > .layout-element-wrapper > .layout-column-resize-bar-left {\n  left: -6px;\n}\n.layout-editor .layout-column > .layout-element-wrapper > .layout-column-resize-bar-right {\n  right: -6px;\n}\n.layout-editor .layout-column.layout-element-focused > .layout-element-wrapper > .layout-column-resize-bar {\n  display: block;\n}\n\n.layout-editor .layout-content,\n.layout-editor .layout-html {\n  min-height: 1em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup {\n  padding: 2px;\n  overflow-x: hidden;\n  line-height: normal;\n  /* this color is just a suggestion and can be changed based on implementation feedback */\n  /* this also has bidi implications */\n  /* this also has bidi implications */\n  /* LTR-specific: use 'margin-right' for rtl elements */\n  /* LTR-specific: use 'padding-right' for rtl elements */\n  /*table[rules=none i], table[rules=groups i], table[rules=rows i],\ntable[rules=cols i], table[rules=all i], table[frame=void i],\ntable[frame=above i], table[frame=below i], table[frame=hsides i],\ntable[frame=lhs i], table[frame=rhs i], table[frame=vsides i],\ntable[frame=box i], table[frame=border i],\ntable[rules=none i] > tr > td, table[rules=none i] > tr > th,\ntable[rules=groups i] > tr > td, table[rules=groups i] > tr > th,\ntable[rules=rows i] > tr > td, table[rules=rows i] > tr > th,\ntable[rules=cols i] > tr > td, table[rules=cols i] > tr > th,\ntable[rules=all i] > tr > td, table[rules=all i] > tr > th,\ntable[rules=none i] > thead > tr > td, table[rules=none i] > thead > tr > th,\ntable[rules=groups i] > thead > tr > td, table[rules=groups i] > thead > tr > th,\ntable[rules=rows i] > thead > tr > td, table[rules=rows i] > thead > tr > th,\ntable[rules=cols i] > thead > tr > td, table[rules=cols i] > thead > tr > th,\ntable[rules=all i] > thead > tr > td, table[rules=all i] > thead > tr > th,\ntable[rules=none i] > tbody > tr > td, table[rules=none i] > tbody > tr > th,\ntable[rules=groups i] > tbody > tr > td, table[rules=groups i] > tbody > tr > th,\ntable[rules=rows i] > tbody > tr > td, table[rules=rows i] > tbody > tr > th,\ntable[rules=cols i] > tbody > tr > td, table[rules=cols i] > tbody > tr > th,\ntable[rules=all i] > tbody > tr > td, table[rules=all i] > tbody > tr > th,\ntable[rules=none i] > tfoot > tr > td, table[rules=none i] > tfoot > tr > th,\ntable[rules=groups i] > tfoot > tr > td, table[rules=groups i] > tfoot > tr > th,\ntable[rules=rows i] > tfoot > tr > td, table[rules=rows i] > tfoot > tr > th,\ntable[rules=cols i] > tfoot > tr > td, table[rules=cols i] > tfoot > tr > th,\ntable[rules=all i] > tfoot > tr > td, table[rules=all i] > tfoot > tr > th {\n    border-color: black;\n}*/\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup *,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup * {\n  margin: 0;\n  padding: 0;\n  box-sizing: content-box;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup address,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup address,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup center,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup center,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup div,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup div,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figcaption,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figcaption,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup footer,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup footer,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup form,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup form,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup header,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup header,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup hr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup hr,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup legend,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup legend,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup xmp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup xmp {\n  display: block;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup xmp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup xmp {\n  margin-top: 1em;\n  margin-bottom: 1em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figure {\n  margin-left: 40px;\n  margin-right: 40px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup address,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup address {\n  font-style: italic;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup xmp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup xmp {\n  font-family: monospace;\n  white-space: pre;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup cite,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup cite,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dfn,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dfn,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup em,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup em,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup i,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup i,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup var,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup var {\n  font-style: italic;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup b,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup b,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup strong,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup strong {\n  font-weight: bolder;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup code,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup code,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup kbd,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup kbd,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup samp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup samp,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tt {\n  font-family: monospace;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup big,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup big {\n  font-size: larger;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup small,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup small {\n  font-size: smaller;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sub,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sub {\n  vertical-align: sub;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sup {\n  vertical-align: super;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sub,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sub,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sup {\n  line-height: normal;\n  font-size: smaller;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ruby,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ruby {\n  display: ruby;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rb,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rb {\n  display: ruby-base;\n  white-space: nowrap;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rt {\n  display: ruby-text;\n  white-space: nowrap;\n  font-size: 50%;\n  font-variant-east-asian: ruby;\n  text-emphasis: none;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rbc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rbc {\n  display: ruby-base-container;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rtc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rtc {\n  display: ruby-text-container;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ruby,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ruby,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rb,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rb,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rt,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rbc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rbc,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rtc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rtc {\n  unicode-bidi: isolate;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :link,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :link {\n  color: #0000EE;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :visited,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :visited {\n  color: #551A8B;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :link,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :link,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :visited,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :visited {\n  text-decoration: underline;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup a:link[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup a:link[rel~=help],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup a:visited[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup a:visited[rel~=help],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup area:link[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup area:link[rel~=help],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup area:visited[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup area:visited[rel~=help] {\n  cursor: help;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :focus,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :focus {\n  outline: auto;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup mark,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup mark {\n  background: yellow;\n  color: black;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup abbr[title],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup abbr[title],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup acronym[title],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup acronym[title] {\n  text-decoration: dotted underline;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ins,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ins,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup u,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup u {\n  text-decoration: underline;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup del,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup del,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup s,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup s,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup strike,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup strike {\n  text-decoration: line-through;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blink,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blink {\n  text-decoration: blink;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup q::before,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup q::before {\n  content: open-quote;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup q::after,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup q::after {\n  content: close-quote;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup br,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup br {\n  content: '\\A';\n  white-space: pre;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup nobr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup nobr {\n  white-space: nowrap;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup wbr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup wbr {\n  content: '\\200B';\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup nobr wbr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup nobr wbr {\n  white-space: normal;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup article,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup article,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup aside,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup aside,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h1,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h1,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h2,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h2,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h3,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h3,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h4,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h4,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h5,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h5,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h6,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h6,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup hgroup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup hgroup,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup nav,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup nav,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup section,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup section {\n  display: block;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h1,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h1 {\n  margin-top: 0.67em;\n  margin-bottom: 0.67em;\n  font-size: 2.00em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h2,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h2 {\n  margin-top: 0.83em;\n  margin-bottom: 0.83em;\n  font-size: 1.50em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h3,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h3 {\n  margin-top: 1.00em;\n  margin-bottom: 1.00em;\n  font-size: 1.17em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h4,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h4 {\n  margin-top: 1.33em;\n  margin-bottom: 1.33em;\n  font-size: 1.00em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h5,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h5 {\n  margin-top: 1.67em;\n  margin-bottom: 1.67em;\n  font-size: 0.83em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h6,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h6 {\n  margin-top: 2.33em;\n  margin-bottom: 2.33em;\n  font-size: 0.67em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dd,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dd,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dt,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  display: block;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup li,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup li {\n  display: list-item;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  margin-top: 1em;\n  margin-bottom: 1em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul {\n  margin-top: 0;\n  margin-bottom: 0;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dd,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dd {\n  margin-left: 40px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  padding-left: 40px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol {\n  list-style-type: decimal;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  list-style-type: disc;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul {\n  list-style-type: circle;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul ul {\n  list-style-type: square;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table {\n  display: table;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup caption,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup caption {\n  display: table-caption;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup colgroup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup colgroup,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup colgroup[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup colgroup[hidden] {\n  display: table-column-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup col,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup col,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup col[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup col[hidden] {\n  display: table-column;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead[hidden] {\n  display: table-header-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody[hidden] {\n  display: table-row-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot[hidden] {\n  display: table-footer-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr[hidden] {\n  display: table-row;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th[hidden] {\n  display: table-cell;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup colgroup[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup colgroup[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup col[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup col[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th[hidden] {\n  visibility: collapse;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table {\n  box-sizing: border-box;\n  border-spacing: 2px;\n  border-collapse: separate;\n  text-indent: initial;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  padding: 1px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table > tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table > tr {\n  vertical-align: middle;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  vertical-align: inherit;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  border-color: gray;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr {\n  border-color: inherit;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup option,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup option,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup optgroup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup optgroup,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup textarea,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup textarea,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup keygen,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup keygen {\n  text-indent: initial;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup textarea,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup textarea {\n  white-space: pre-wrap;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"radio\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"radio\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"checkbox\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"checkbox\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"reset\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"reset\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"submit\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"submit\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup button {\n  box-sizing: border-box;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup button {\n  padding: 0.3em 0.5em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup hr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup hr {\n  color: gray;\n  border-style: inset;\n  border-width: 1px;\n  margin: 0.5em auto;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup fieldset,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup fieldset {\n  margin-left: 2px;\n  margin-right: 2px;\n  border: groove 2px ThreeDFace;\n  padding: 0.35em 0.625em 0.75em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup legend,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup legend {\n  padding-left: 2px;\n  padding-right: 2px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup > *:first-child,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup > *:first-child {\n  margin-top: 0 !important;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup > *:last-child,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup > *:last-child {\n  margin-bottom: 0 !important;\n}\n.layout-editor .layout-content.layout-content-image > .layout-element-wrapper > .layout-content-markup > img,\n.layout-editor .layout-html.layout-content-image > .layout-element-wrapper > .layout-content-markup > img,\n.layout-editor .layout-content.layout-content-vector-image > .layout-element-wrapper > .layout-content-markup > img,\n.layout-editor .layout-html.layout-content-vector-image > .layout-element-wrapper > .layout-content-markup > img {\n  display: block;\n  width: 100%;\n  max-width: 100%;\n  height: auto;\n}\n.layout-editor .img-responsive,\n.layout-editor .img-responsive img {\n  display: block;\n  width: 100%;\n  max-width: 100%;\n  height: auto;\n}\n\n.layout-editor > .layout-toolbox-wrapper {\n  position: relative;\n  margin-left: 12px;\n  width: 220px;\n  flex-shrink: 0;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox {\n  border: 1px solid #e4e5e6;\n  width: 220px;\n  min-height: 400px;\n  padding: 6px;\n  background-color: #f3f4f5;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox.sticky-top {\n  position: fixed;\n  top: 0;\n  max-height: 100%;\n  overflow-y: auto;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox.sticky-bottom {\n  position: absolute;\n  bottom: 0;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group {\n  margin-top: 12px;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-group-heading {\n  display: block;\n  margin-bottom: 4px;\n  text-decoration: none;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-group-heading:before {\n  display: inline-block;\n  width: 10px;\n  margin-right: 4px;\n  font: normal normal normal 14px/1 FontAwesome;\n  text-align: center;\n  content: \"\\f0d7\";\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group.collapsed .layout-toolbox-group-heading:before {\n  content: \"\\f0da\";\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group.collapsed .layout-toolbox-items {\n  display: none;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-section + .layout-toolbox-section {\n  margin-top: 4px;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-item {\n  border: 1px solid #e4e5e6;\n  background-color: #fff;\n  padding: 9px 12px;\n  cursor: default;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-item i {\n  display: inline-block;\n  width: 16px;\n  font: normal normal normal 14px/1 FontAwesome;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-item + .layout-toolbox-item {\n  margin-top: 4px;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group + .layout-toolbox-group {\n  margin-top: 6px;\n}\n\n.layout-editor .layout-popup {\n  display: none;\n  position: absolute;\n  margin: 0;\n  -moz-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\n  -webkit-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\n  box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\n  border: 1px solid #e4e5e6;\n  padding: 2px 0;\n  background-color: #f7f7f7;\n  list-style: none;\n  z-index: 20;\n  color: #7c7c7c;\n  text-align: left;\n  cursor: default;\n  white-space: nowrap;\n  line-height: normal;\n  min-width: 300px;\n}\n.layout-editor .layout-popup.wide {\n  width: 600px;\n}\n.layout-editor .layout-popup .layout-popup-flex {\n  display: flex;\n  padding: 2px 5px;\n}\n.layout-editor .layout-popup .layout-popup-flex .layout-popup-column + .layout-popup-column {\n  margin-left: 4px;\n  border-left: 1px solid rgba(128, 128, 128, 0.15);\n  padding-left: 4px;\n}\n.layout-editor .layout-popup .layout-popup-item {\n  padding: 4px 6px;\n}\n.layout-editor .layout-popup .layout-popup-label {\n  font-size: 0.9em;\n  font-weight: bold;\n  color: #7c7c7c;\n}\n.layout-editor .layout-popup .layout-popup-action:hover {\n  background-color: #f3f4f5;\n  cursor: pointer;\n}\n.layout-editor .layout-popup .layout-popup-input input[type='text'],\n.layout-editor .layout-popup .layout-popup-input textarea {\n  width: 100%;\n}\n.layout-editor .layout-popup .layout-popup-input > label {\n  display: block;\n  font-size: 0.9em;\n}\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-element {\r\n        position: relative;\r\n        margin-top: 0;\r\n        margin-right: 0;\r\n        margin-bottom: 0;\r\n        padding: 0;\r\n\r\n        // We don't fuck with the left margin of columns, because Bootstrap uses these to render column offsets.\r\n        &:not(.layout-column) {\r\n            margin-left: 0;\r\n        }\r\n\r\n        > .layout-element-wrapper {\r\n            > .layout-panel {\r\n                display: none; // Shown only in active or focused states.\r\n                position: absolute;\r\n                margin: 0;\r\n                z-index: 20;\r\n                height: 25px;\r\n                padding: 0 6px;\r\n                list-style: none;\r\n                white-space: nowrap;\r\n                line-height: 25px;\r\n                vertical-align: middle;\r\n\r\n                > .layout-panel-item {\r\n                    display: inline-block;\r\n                    height: 25px;\r\n                    padding: 1px 6px 0;\r\n                }\r\n\r\n                > .layout-panel-label {\r\n                    font-size: @font-size - 1;\r\n                }\r\n\r\n                > .layout-panel-action {\r\n                    display: none; // Shown only in focused state.\r\n                    width: 28px;\r\n                    cursor: pointer;\r\n                    text-align: center;\r\n                }\r\n            }\r\n\r\n            > .layout-panel-main {\r\n                top: -27px;\r\n                left: -2px;\r\n            }\r\n        }\r\n    }\r\n\r\n    // When dragging from toolbox, elements will be li rather than div.\r\n    li.layout-element {\r\n        list-style: none;\r\n    }\r\n\r\n    &:not(.layout-editor-dragging) {\r\n\r\n        .layout-element-active, .layout-element-focused {\r\n            border-width: 2px;\r\n            border-style: solid;\r\n\r\n            > .layout-element-wrapper {\r\n                margin: -2px;\r\n            }\r\n        }\r\n\r\n        .layout-element-active {\r\n            @active-highlight: fade(#686868, 10%);\r\n\r\n            border-color: @active-highlight;\r\n\r\n            > .layout-element-wrapper > .layout-panel-main {\r\n                display: block; // To reveal.\r\n                z-index: 30;\r\n                background-color: @active-highlight;\r\n            }\r\n        }\r\n\r\n        .layout-element-focused {\r\n            @focused-highlight: #648721;\r\n            @focused-text: #fefefe;\r\n\r\n            border-color: @focused-highlight;\r\n\r\n            > .layout-element-wrapper > .layout-panel {\r\n                display: block; // To reveal.\r\n                background-color: @focused-highlight;\r\n                color: @focused-text;\r\n\r\n                > .layout-panel-action {\r\n                    display: inline-block; // To reveal.\r\n                    &:hover {\r\n                        background-color: lighten(@focused-highlight, 10%);\r\n                    }\r\n\r\n                    &.disabled {\r\n                        cursor: default;\r\n                        color: fade(@focused-text, 40%);\r\n\r\n                        &:hover {\r\n                            background-color: @focused-highlight;\r\n                        }\r\n                    }\r\n\r\n                    &.active {\r\n                        color: lighten(saturate(spin(@focused-highlight, -10), 100%), 30%);\r\n                        background-color: lighten(@focused-highlight, 5%);\r\n\r\n                        &:hover {\r\n                            background-color: lighten(@focused-highlight, 10%);\r\n                        }\r\n                    }\r\n                }\r\n            }\r\n        }\r\n\r\n        .layout-element-selected {\r\n            @selected-highlight: #648721;\r\n            background-color: fade(@selected-highlight, 8%);\r\n        }\r\n    }\r\n\r\n    .ui-sortable-placeholder {\r\n        display: none;\r\n    }\r\n\r\n    .layout-element-droptarget {\r\n        @droptarget-highlight: #648721;\r\n        box-shadow: inset 0 0 12px 6px fade(@droptarget-highlight, 50%);\r\n\r\n        .ui-sortable-placeholder {\r\n            display: block; // To reveal.\r\n            visibility: visible !important;\r\n            min-height: @element-min-height - 2px;\r\n            border: 2px dashed @droptarget-highlight !important;\r\n            background-color: fade(@droptarget-highlight, 16%);\r\n        }\r\n    }\r\n\r\n    // A CSS fix for media item elements.\r\n    .media-thumbnail {\r\n        img {\r\n            max-width:100%;\r\n            max-height: 100%;\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-container {\r\n\r\n        > .layout-element-wrapper {\r\n\r\n            > .layout-container-children-placeholder {\r\n                display: none;\r\n                flex-direction: column;\r\n                margin: @container-padding - 1px;\r\n                min-height: @element-min-height;\r\n                border: 1px dashed fade(@gray-text, 40%);\r\n                border-radius: 4px;\r\n                padding: @container-padding;\r\n                justify-content: center;\r\n                font-size: @font-size - 1;\r\n                font-style: italic;\r\n                opacity: 0.6;\r\n                text-align: center;\r\n            }\r\n\r\n            > .layout-children {\r\n                padding: @container-padding;\r\n\r\n                // All adjacent non-container children need some space between...\r\n                > .layout-element:not(.layout-container) + .layout-element:not(.layout-container) {\r\n                    margin-top: @content-spacing;\r\n                }\r\n\r\n                // ... except when jQuery UI sortable helper is the first element (because that one is absolutely positioned and \"floating\").\r\n                > .ui-sortable-helper:first-child + .layout-element:not(.layout-container) {\r\n                    margin-top: 0;\r\n                }\r\n            }\r\n\r\n            &.layout-container-empty {\r\n                > .layout-container-children-placeholder {\r\n                    display: flex; // To reveal.\r\n                }\r\n\r\n                > .layout-children {\r\n                    position: absolute;\r\n                    top: 0;\r\n                    right: 0;\r\n                    bottom: 0;\r\n                    left: 0;\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    &:not(.layout-editor-dragging) {\r\n        .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children {\r\n            > .layout-element:not(.layout-element-active) {\r\n                border: 1px dashed fade(@gray-text, 60%);\r\n\r\n                > .layout-element-wrapper {\r\n                    margin: -1px;\r\n                }\r\n            }\r\n\r\n            > .layout-container + .layout-container:not(.layout-element-active) {\r\n                border-top: 0;\r\n\r\n                > .layout-element-wrapper {\r\n                    margin-top: 0;\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-canvas {\r\n        #dummy { // Only added because WE doesn't compile if there are no rules.\r\n            display: none;\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-row {\r\n        &.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children {\r\n            > .layout-element:not(.layout-element-active) {\r\n                border-top: 1px dashed fade(@gray-text, 60%) !important;\r\n                > .layout-element-wrapper {\r\n                    margin-top: -1px !important;\r\n                }\r\n            }\r\n\r\n            > .layout-element + .layout-element:not(.layout-element-active) {\r\n                border-left: 0 !important;\r\n                > .layout-element-wrapper {\r\n                    margin-left: 0 !important;\r\n                }\r\n            }\r\n        }\r\n    }\r\n}","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-column {\r\n        > .layout-element-wrapper {\r\n            > .layout-column-resize-bar {\r\n                display: none;\r\n                position: absolute;\r\n                z-index: 30;\r\n                top: 0;\r\n                width: 16px;\r\n                height: 100%;\r\n                cursor: col-resize;\r\n            }\r\n\r\n            > .layout-column-resize-bar-left {\r\n                left: -6px;\r\n            }\r\n\r\n            > .layout-column-resize-bar-right {\r\n                right: -6px;\r\n            }\r\n        }\r\n\r\n        &.layout-element-focused > .layout-element-wrapper > .layout-column-resize-bar {\r\n            display: block;\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n@import \"Reset.less\";\r\n\r\n.layout-editor {\r\n    .layout-content, .layout-html {\r\n        min-height: 1em;\r\n\r\n        > .layout-element-wrapper {\r\n            .layout-content-markup {\r\n\r\n                padding: 2px;\r\n                overflow-x: hidden;\r\n                line-height: normal;\r\n\r\n                // Reset to HTML5 W3C standard default styling within content.\r\n                .reset();\r\n\r\n                > *:first-child {\r\n                    margin-top: 0 !important; // Important because site.css of the admin theme styles heading margins with a very high specificity.\r\n                }\r\n\r\n                > *:last-child {\r\n                    margin-bottom: 0 !important;\r\n                }\r\n            }\r\n        }\r\n\r\n        &.layout-content-image, &.layout-content-vector-image {\r\n            > .layout-element-wrapper > .layout-content-markup > img {\r\n                display: block;\r\n                width: 100%;\r\n                max-width: 100%;\r\n                height: auto;\r\n            }\r\n        }\r\n    }\r\n\r\n    .img-responsive, .img-responsive img {\r\n        display: block;\r\n        width: 100%;\r\n        max-width: 100%;\r\n        height: auto;\r\n    }\r\n}\r\n","﻿.reset() {\r\n\r\n    * {\r\n        margin: 0;\r\n        padding: 0;\r\n        box-sizing: content-box;\r\n    }\r\n\r\n    address, blockquote, center, div, figure, figcaption, footer, form,\r\n    header, hr, legend, listing, p, plaintext, pre, xmp {\r\n        display: block;\r\n    }\r\n\r\n    blockquote, figure, listing, p, plaintext, pre, xmp {\r\n        margin-top: 1em;\r\n        margin-bottom: 1em;\r\n    }\r\n\r\n    blockquote, figure {\r\n        margin-left: 40px;\r\n        margin-right: 40px;\r\n    }\r\n\r\n    address {\r\n        font-style: italic;\r\n    }\r\n\r\n    listing, plaintext, pre, xmp {\r\n        font-family: monospace;\r\n        white-space: pre;\r\n    }\r\n\r\n    cite, dfn, em, i, var {\r\n        font-style: italic;\r\n    }\r\n\r\n    b, strong {\r\n        font-weight: bolder;\r\n    }\r\n\r\n    code, kbd, samp, tt {\r\n        font-family: monospace;\r\n    }\r\n\r\n    big {\r\n        font-size: larger;\r\n    }\r\n\r\n    small {\r\n        font-size: smaller;\r\n    }\r\n\r\n    sub {\r\n        vertical-align: sub;\r\n    }\r\n\r\n    sup {\r\n        vertical-align: super;\r\n    }\r\n\r\n    sub, sup {\r\n        line-height: normal;\r\n        font-size: smaller;\r\n    }\r\n\r\n\r\n    ruby {\r\n        display: ruby;\r\n    }\r\n\r\n    rb {\r\n        display: ruby-base;\r\n        white-space: nowrap;\r\n    }\r\n\r\n    rt {\r\n        display: ruby-text;\r\n        white-space: nowrap;\r\n        font-size: 50%;\r\n        font-variant-east-asian: ruby;\r\n        text-emphasis: none;\r\n    }\r\n\r\n    rbc {\r\n        display: ruby-base-container;\r\n    }\r\n\r\n    rtc {\r\n        display: ruby-text-container;\r\n    }\r\n\r\n    ruby, rb, rt, rbc, rtc {\r\n        unicode-bidi: isolate;\r\n    }\r\n\r\n\r\n    :link {\r\n        color: #0000EE;\r\n    }\r\n\r\n    :visited {\r\n        color: #551A8B;\r\n    }\r\n\r\n    :link, :visited {\r\n        text-decoration: underline;\r\n    }\r\n\r\n    a:link[rel~=help], a:visited[rel~=help],\r\n    area:link[rel~=help], area:visited[rel~=help] {\r\n        cursor: help;\r\n    }\r\n\r\n    :focus {\r\n        outline: auto;\r\n    }\r\n\r\n    mark {\r\n        background: yellow;\r\n        color: black;\r\n    }\r\n    /* this color is just a suggestion and can be changed based on implementation feedback */\r\n    abbr[title], acronym[title] {\r\n        text-decoration: dotted underline;\r\n    }\r\n\r\n    ins, u {\r\n        text-decoration: underline;\r\n    }\r\n\r\n    del, s, strike {\r\n        text-decoration: line-through;\r\n    }\r\n\r\n    blink {\r\n        text-decoration: blink;\r\n    }\r\n\r\n    q::before {\r\n        content: open-quote;\r\n    }\r\n\r\n    q::after {\r\n        content: close-quote;\r\n    }\r\n\r\n    br {\r\n        content: '\\A';\r\n        white-space: pre;\r\n    }\r\n    /* this also has bidi implications */\r\n    nobr {\r\n        white-space: nowrap;\r\n    }\r\n\r\n    wbr {\r\n        content: '\\200B';\r\n    }\r\n    /* this also has bidi implications */\r\n    nobr wbr {\r\n        white-space: normal;\r\n    }\r\n\r\n    article, aside, h1, h2, h3, h4, h5, h6, hgroup, nav, section {\r\n        display: block;\r\n    }\r\n\r\n    h1 {\r\n        margin-top: 0.67em;\r\n        margin-bottom: 0.67em;\r\n        font-size: 2.00em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h2 {\r\n        margin-top: 0.83em;\r\n        margin-bottom: 0.83em;\r\n        font-size: 1.50em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h3 {\r\n        margin-top: 1.00em;\r\n        margin-bottom: 1.00em;\r\n        font-size: 1.17em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h4 {\r\n        margin-top: 1.33em;\r\n        margin-bottom: 1.33em;\r\n        font-size: 1.00em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h5 {\r\n        margin-top: 1.67em;\r\n        margin-bottom: 1.67em;\r\n        font-size: 0.83em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h6 {\r\n        margin-top: 2.33em;\r\n        margin-bottom: 2.33em;\r\n        font-size: 0.67em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    dir, dd, dl, dt, ol, ul {\r\n        display: block;\r\n    }\r\n\r\n    li {\r\n        display: list-item;\r\n    }\r\n\r\n    dir, dl, ol, ul {\r\n        margin-top: 1em;\r\n        margin-bottom: 1em;\r\n    }\r\n\r\n        dir dir, dir dl, dir ol, dir ul,\r\n        dl dir, dl dl, dl ol, dl ul,\r\n        ol dir, ol dl, ol ol, ol ul,\r\n        ul dir, ul dl, ul ol, ul ul {\r\n            margin-top: 0;\r\n            margin-bottom: 0;\r\n        }\r\n\r\n    dd {\r\n        margin-left: 40px;\r\n    }\r\n    /* LTR-specific: use 'margin-right' for rtl elements */\r\n    dir, ol, ul {\r\n        padding-left: 40px;\r\n    }\r\n    /* LTR-specific: use 'padding-right' for rtl elements */\r\n    ol {\r\n        list-style-type: decimal;\r\n    }\r\n\r\n    dir, ul {\r\n        list-style-type: disc;\r\n    }\r\n\r\n        dir dir, dir ul,\r\n        ol dir, ol ul,\r\n        ul dir, ul ul {\r\n            list-style-type: circle;\r\n        }\r\n\r\n            dir dir dir, dir dir ul,\r\n            dir ol dir, dir ol ul,\r\n            dir ul dir, dir ul ul,\r\n            ol dir dir, ol dir ul,\r\n            ol ol dir, ol ol ul,\r\n            ol ul dir, ol ul ul,\r\n            ul dir dir, ul dir ul,\r\n            ul ol dir, ul ol ul,\r\n            ul ul dir, ul ul ul {\r\n                list-style-type: square;\r\n            }\r\n\r\n    table {\r\n        display: table;\r\n    }\r\n\r\n    caption {\r\n        display: table-caption;\r\n    }\r\n\r\n    colgroup, colgroup[hidden] {\r\n        display: table-column-group;\r\n    }\r\n\r\n    col, col[hidden] {\r\n        display: table-column;\r\n    }\r\n\r\n    thead, thead[hidden] {\r\n        display: table-header-group;\r\n    }\r\n\r\n    tbody, tbody[hidden] {\r\n        display: table-row-group;\r\n    }\r\n\r\n    tfoot, tfoot[hidden] {\r\n        display: table-footer-group;\r\n    }\r\n\r\n    tr, tr[hidden] {\r\n        display: table-row;\r\n    }\r\n\r\n    td, th, td[hidden], th[hidden] {\r\n        display: table-cell;\r\n    }\r\n\r\n        colgroup[hidden], col[hidden], thead[hidden], tbody[hidden],\r\n        tfoot[hidden], tr[hidden], td[hidden], th[hidden] {\r\n            visibility: collapse;\r\n        }\r\n\r\n    table {\r\n        box-sizing: border-box;\r\n        border-spacing: 2px;\r\n        border-collapse: separate;\r\n        text-indent: initial;\r\n    }\r\n\r\n    td, th {\r\n        padding: 1px;\r\n    }\r\n\r\n    th {\r\n        font-weight: bold;\r\n    }\r\n\r\n    thead, tbody, tfoot, table > tr {\r\n        vertical-align: middle;\r\n    }\r\n\r\n    tr, td, th {\r\n        vertical-align: inherit;\r\n    }\r\n\r\n    table, td, th {\r\n        border-color: gray;\r\n    }\r\n\r\n    thead, tbody, tfoot, tr {\r\n        border-color: inherit;\r\n    }\r\n    /*table[rules=none i], table[rules=groups i], table[rules=rows i],\r\ntable[rules=cols i], table[rules=all i], table[frame=void i],\r\ntable[frame=above i], table[frame=below i], table[frame=hsides i],\r\ntable[frame=lhs i], table[frame=rhs i], table[frame=vsides i],\r\ntable[frame=box i], table[frame=border i],\r\ntable[rules=none i] > tr > td, table[rules=none i] > tr > th,\r\ntable[rules=groups i] > tr > td, table[rules=groups i] > tr > th,\r\ntable[rules=rows i] > tr > td, table[rules=rows i] > tr > th,\r\ntable[rules=cols i] > tr > td, table[rules=cols i] > tr > th,\r\ntable[rules=all i] > tr > td, table[rules=all i] > tr > th,\r\ntable[rules=none i] > thead > tr > td, table[rules=none i] > thead > tr > th,\r\ntable[rules=groups i] > thead > tr > td, table[rules=groups i] > thead > tr > th,\r\ntable[rules=rows i] > thead > tr > td, table[rules=rows i] > thead > tr > th,\r\ntable[rules=cols i] > thead > tr > td, table[rules=cols i] > thead > tr > th,\r\ntable[rules=all i] > thead > tr > td, table[rules=all i] > thead > tr > th,\r\ntable[rules=none i] > tbody > tr > td, table[rules=none i] > tbody > tr > th,\r\ntable[rules=groups i] > tbody > tr > td, table[rules=groups i] > tbody > tr > th,\r\ntable[rules=rows i] > tbody > tr > td, table[rules=rows i] > tbody > tr > th,\r\ntable[rules=cols i] > tbody > tr > td, table[rules=cols i] > tbody > tr > th,\r\ntable[rules=all i] > tbody > tr > td, table[rules=all i] > tbody > tr > th,\r\ntable[rules=none i] > tfoot > tr > td, table[rules=none i] > tfoot > tr > th,\r\ntable[rules=groups i] > tfoot > tr > td, table[rules=groups i] > tfoot > tr > th,\r\ntable[rules=rows i] > tfoot > tr > td, table[rules=rows i] > tfoot > tr > th,\r\ntable[rules=cols i] > tfoot > tr > td, table[rules=cols i] > tfoot > tr > th,\r\ntable[rules=all i] > tfoot > tr > td, table[rules=all i] > tfoot > tr > th {\r\n    border-color: black;\r\n}*/\r\n    input, select, option, optgroup, button, textarea, keygen {\r\n        text-indent: initial;\r\n    }\r\n\r\n    textarea {\r\n        white-space: pre-wrap;\r\n    }\r\n\r\n    input[type=\"radio\"], input[type=\"checkbox\"], input[type=\"reset\"], input[type=\"button\"],\r\n    input[type=\"submit\"], select, button {\r\n        box-sizing: border-box;\r\n    }\r\n\r\n    input[type=\"button\"], button {\r\n        padding: 0.3em 0.5em;\r\n    }\r\n\r\n    hr {\r\n        color: gray;\r\n        border-style: inset;\r\n        border-width: 1px;\r\n        margin: 0.5em auto;\r\n    }\r\n\r\n    fieldset {\r\n        margin-left: 2px;\r\n        margin-right: 2px;\r\n        border: groove 2px ThreeDFace;\r\n        padding: 0.35em 0.625em 0.75em;\r\n    }\r\n\r\n    legend {\r\n        padding-left: 2px;\r\n        padding-right: 2px;\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    > .layout-toolbox-wrapper {\r\n        position: relative;\r\n        margin-left: @container-padding;\r\n        width: 220px;\r\n        flex-shrink: 0;\r\n\r\n        > .layout-toolbox {\r\n            border: 1px solid @gray-border;\r\n            width: 220px;\r\n            min-height: 400px;\r\n            padding: @container-padding / 2;\r\n            background-color: @gray-bg;\r\n\r\n            &.sticky-top {\r\n                position: fixed;\r\n                top: 0;\r\n                max-height: 100%;\r\n                overflow-y: auto;\r\n            }\r\n\r\n            &.sticky-bottom {\r\n                position: absolute;\r\n                bottom: 0;\r\n            }\r\n\r\n            .layout-toolbox-group {\r\n                margin-top: @container-padding;\r\n\r\n                .layout-toolbox-group-heading {\r\n                    display: block;\r\n                    margin-bottom: @container-padding / 3;\r\n                    text-decoration: none;\r\n                \r\n                    &:before {\r\n                        display: inline-block;\r\n                        width: 10px;\r\n                        margin-right: @container-padding / 3;\r\n                        font: normal normal normal 14px/1 FontAwesome;\r\n                        text-align: center;\r\n                        content: \"\\f0d7\";\r\n                    }\r\n                }\r\n            \r\n                &.collapsed {\r\n                    .layout-toolbox-group-heading:before {\r\n                        content: \"\\f0da\";\r\n                    }\r\n\r\n                    .layout-toolbox-items {\r\n                        display: none;\r\n                    }\r\n                }\r\n\r\n                .layout-toolbox-section + .layout-toolbox-section {\r\n                    margin-top: @container-padding / 3;\r\n                }\r\n\r\n                .layout-toolbox-item {\r\n                    border: 1px solid @gray-border;\r\n                    background-color: #fff;\r\n                    padding: (@container-padding - 3) @container-padding;\r\n                    cursor: default;\r\n\r\n                    i {\r\n                        display: inline-block;\r\n                        width: 16px;\r\n                        font: normal normal normal 14px/1 FontAwesome;\r\n                    }\r\n\r\n                    + .layout-toolbox-item {\r\n                        margin-top: @container-padding / 3;\r\n                    }\r\n                }\r\n\r\n                + .layout-toolbox-group {\r\n                    margin-top: @container-padding / 2;\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-popup {\r\n        display: none; // Shown only in active or focused states.\r\n        position: absolute;\r\n        margin: 0;\r\n        -moz-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\r\n        -webkit-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\r\n        box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\r\n        border: 1px solid @gray-border;\r\n        padding: 2px 0;\r\n        background-color: #f7f7f7;\r\n        list-style: none;\r\n        z-index: 20;\r\n        color: @gray-text;\r\n        text-align: left;\r\n        cursor: default;\r\n        white-space: nowrap;\r\n        line-height: normal;\r\n        min-width: 300px;\r\n\r\n        &.wide {\r\n            width: 600px;\r\n        }\r\n\r\n        .layout-popup-flex {\r\n            display: flex;\r\n            padding: 2px 5px;\r\n\r\n            .layout-popup-column + .layout-popup-column {\r\n                margin-left: 4px;\r\n                border-left: 1px solid fade(gray, 15%);\r\n                padding-left: 4px;\r\n            }\r\n        }\r\n\r\n        .layout-popup-item {\r\n            padding: 4px 6px;\r\n        }\r\n\r\n        .layout-popup-label {\r\n            font-size: 0.9em;\r\n            font-weight: bold;\r\n            color: @gray-text;\r\n        }\r\n\r\n        .layout-popup-action:hover {\r\n            background-color: @gray-bg;\r\n            cursor: pointer;\r\n        }\r\n\r\n        .layout-popup-input {\r\n            input[type='text'], textarea {\r\n                width: 100%;\r\n            }\r\n\r\n            > label {\r\n                display: block;\r\n                font-size: 0.9em;\r\n            }\r\n        }\r\n    }\r\n}\r\n"],"sourceRoot":"/source/"} */ \ No newline at end of file +/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Editor.less","LayoutEditor.css","Element.less","Container.less","Canvas.less","Row.less","Column.less","Content.less","Reset.less","Toolbox.less","Popup.less"],"names":[],"mappings":"AAEA;EACI,sBAAA;EAAA,qBAAA;EAAA,cAAA;EACA,uCAAA;MAAA,uBAAA;UAAA,+BAAA;EACA,mBAAA;EACA,UAAA;CCDH;ADHD;EAOQ,sBAAA;EAAA,qBAAA;EAAA,cAAA;CCDP;ADND;EAUY,kBAAA;CCDX;ADMD;EACI,sBAAA;EAAA,qBAAA;EAAA,cAAA;EACA,gBAAA;EACA,gBAAA;EACA,6BAAA;MAAA,wBAAA;UAAA,qBAAA;CCJH;ADAD;EAOQ,qBAAA;MAAA,qBAAA;UAAA,aAAA;EACA,0BAAA;EACA,0BAAA;CCJP;ADLD;EAYY,cAAA;EACA,oBAAA;EACA,iBAAA;CCJX;ADVD;EAiBgB,uBAAA;CCJf;ADbD;EAsBY,wBAAA;EACA,qBAAA;EACA,oBAAA;CCNX;ADlBD;EA6BQ,oBAAA;CCRP;ADYD;EACI,cAAA;CCVH;ADaO;;EACI,aAAA;EACA,eAAA;CCVX;ADaO;EACI,YAAA;CCXX;ADCD;;EAcY,gBAAA;CCXX;ADHD;EAkBY,uBAAA;EACA,YAAA;EACA,WAAA;CCZX;ADcW;EACI,oBAAA;EACA,YAAA;CCZf;ADZD;EA6BY,gBAAA;CCdX;ADfD;EAkCQ,mBAAA;EACA,0BAAA;EACA,iBAAA;EACA,uBAAA;CChBP;ADrBD;EAyCQ,qBAAA;EACA,mBAAA;CCjBP;ADzBD;EA8CQ,oBAAA;CClBP;;AC5ED;EAEQ,mBAAA;EACA,cAAA;EACA,gBAAA;EACA,iBAAA;EACA,WAAA;CD8EP;AC3EO;EACI,eAAA;CD6EX;ACvFD;EAegB,cAAA;EACA,mBAAA;EACA,UAAA;EACA,YAAA;EACA,aAAA;EACA,eAAA;EACA,iBAAA;EACA,oBAAA;EACA,kBAAA;EACA,uBAAA;CD2Ef;ACnGD;EA2BoB,sBAAA;EACA,aAAA;EACA,mBAAA;CD2EnB;ACxGD;EAiCoB,gBAAA;CD0EnB;AC3GD;EAqCoB,cAAA;EACA,YAAA;EACA,gBAAA;EACA,mBAAA;CDyEnB;ACjHD;EA6CgB,WAAA;EACA,WAAA;CDuEf;ACrHD;EAqDQ,iBAAA;CDmEP;AChEG;;EAGQ,kBAAA;EACA,oBAAA;CDiEX;ACrEG;;EAOY,aAAA;CDkEf;ACzEG;EAcQ,uCAAA;CD8DX;AC5EG;EAiBY,eAAA;EACA,YAAA;EACA,2CAAA;CD8Df;ACjFG;EA2BQ,sBAAA;CDyDX;ACpFG;EA8BY,eAAA;EACA,0BAAA;EACA,eAAA;CDyDf;ACzFG;EAmCgB,sBAAA;CDyDnB;ACxDmB;EACI,0BAAA;CD0DvB;ACvDmB;EACI,gBAAA;EACA,gCAAA;CDyDvB;ACvDuB;EACI,0BAAA;CDyD3B;ACrDmB;EACI,eAAA;EACA,0BAAA;CDuDvB;ACrDuB;EACI,0BAAA;CDuD3B;AC7GG;EA+DQ,2CAAA;CDiDX;ACxKD;EA4HQ,cAAA;CD+CP;AC3KD;EAiIQ,uDAAA;CD6CP;AC9KD;EAoIY,eAAA;EACA,+BAAA;EACA,iBAAA;EACA,sCAAA;EACA,2CAAA;CD6CX;ACrLD;EA+IY,gBAAA;EACA,iBAAA;CDyCX;;AEzLD;EAMgB,cAAA;EACA,+BAAA;MAAA,2BAAA;UAAA,uBAAA;EACA,aAAA;EACA,iBAAA;EACA,4CAAA;EACA,mBAAA;EACA,cAAA;EACA,gCAAA;MAAA,sBAAA;UAAA,wBAAA;EACA,gBAAA;EACA,mBAAA;EACA,aAAA;EACA,mBAAA;CFuLf;AExMD;EAqBgB,cAAA;CFsLf;AE3MD;EAyBoB,iBAAA;CFqLnB;AE9MD;EA8BoB,cAAA;CFmLnB;AE/KW;EAEQ,sBAAA;EAAA,qBAAA;EAAA,cAAA;CFgLnB;AElLW;EAMQ,mBAAA;EACA,OAAA;EACA,SAAA;EACA,UAAA;EACA,QAAA;CF+KnB;AEzKG;EAGY,4CAAA;CFyKf;AE5KG;EAMgB,aAAA;CFyKnB;AE/KG;EAWY,cAAA;CFuKf;AElLG;EAcgB,cAAA;CFuKnB;;AGvOD;EAGY,cAAA;CHwOX;;AIzOO;EAEQ,2DAAA;CJ2Of;AI7OO;EAIY,4BAAA;CJ4OnB;AIhPO;EASQ,0BAAA;CJ0Of;AInPO;EAWY,0BAAA;CJ2OnB;;AKxPD;EAIgB,cAAA;EACA,mBAAA;EACA,YAAA;EACA,OAAA;EACA,YAAA;EACA,aAAA;EACA,mBAAA;CLwPf;AKlQD;EAcgB,WAAA;CLuPf;AKrQD;EAkBgB,YAAA;CLsPf;AKlPO;EACI,eAAA;CLoPX;;AM1QD;;EAEQ,gBAAA;CN6QP;AM/QD;;EAOgB,aAAA;EACA,mBAAA;EACA,oBAAA;EN4Qd,yFAAyF;EACzF,qCAAqC;EACrC,qCAAqC;EACrC,uDAAuD;EACvD,wDAAwD;EACxD;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BC;CACF;AMrTD;;ECAQ,UAAA;EACA,WAAA;EACA,wBAAA;CPyTP;AM3TD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECOQ,eAAA;CPsVP;AM7VD;;;;;;;;;;;;;;ECWQ,gBAAA;EACA,mBAAA;CPkWP;AM9WD;;;;ECgBQ,kBAAA;EACA,mBAAA;CPoWP;AMrXD;;ECqBQ,mBAAA;CPoWP;AMzXD;;;;;;;;ECyBQ,uBAAA;EACA,iBAAA;CP0WP;AMpYD;;;;;;;;;;EC8BQ,mBAAA;CPkXP;AMhZD;;;;ECkCQ,oBAAA;CPoXP;AMtZD;;;;;;;;ECsCQ,uBAAA;CP0XP;AMhaD;;EC0CQ,kBAAA;CP0XP;AMpaD;;EC8CQ,mBAAA;CP0XP;AMxaD;;ECkDQ,oBAAA;CP0XP;AM5aD;;ECsDQ,sBAAA;CP0XP;AMhbD;;;;EC0DQ,oBAAA;EACA,mBAAA;CP4XP;AMvbD;;ECgEQ,cAAA;CP2XP;AM3bD;;ECoEQ,mBAAA;EACA,oBAAA;CP2XP;AMhcD;;ECyEQ,mBAAA;EACA,oBAAA;EACA,eAAA;EACA,8BAAA;EACA,4BAAA;UAAA,oBAAA;CP2XP;AMxcD;;ECiFQ,6BAAA;CP2XP;AM5cD;;ECqFQ,6BAAA;CP2XP;AMhdD;;;;;;;;;;ECyFQ,sBAAA;CPmYP;AM5dD;;EC8FQ,eAAA;CPkYP;AMheD;;ECkGQ,eAAA;CPkYP;AMpeD;;;;ECsGQ,2BAAA;CPoYP;AM1eD;;;;;;;;EC2GQ,aAAA;CPyYP;AMpfD;;EC+GQ,cAAA;CPyYP;AMxfD;;ECmHQ,mBAAA;EACA,aAAA;CPyYP;AM7fD;;;;ECwHQ,kCAAA;CP2YP;AMngBD;;;;EC4HQ,2BAAA;CP6YP;AMzgBD;;;;;;ECgIQ,8BAAA;CPiZP;AMjhBD;;ECoIQ,uBAAA;CPiZP;AMrhBD;;ECwIQ,oBAAA;CPiZP;AMzhBD;;EC4IQ,qBAAA;CPiZP;AM7hBD;;ECgJQ,cAAA;EACA,iBAAA;CPiZP;AMliBD;;ECqJQ,oBAAA;CPiZP;AMtiBD;;ECyJQ,iBAAA;CPiZP;AM1iBD;;EC6JQ,oBAAA;CPiZP;AM9iBD;;;;;;;;;;;;;;;;;;;;;;ECiKQ,eAAA;CPqaP;AMtkBD;;ECqKQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AM7kBD;;EC4KQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMplBD;;ECmLQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AM3lBD;;EC0LQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMlmBD;;ECiMQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMzmBD;;ECwMQ,mBAAA;EACA,sBAAA;EACA,kBAAA;EACA,kBAAA;CPqaP;AMhnBD;;;;;;;;;;;;EC+MQ,eAAA;CP+aP;AM9nBD;;ECmNQ,mBAAA;CP+aP;AMloBD;;;;;;;;ECuNQ,gBAAA;EACA,mBAAA;CPqbP;AM7oBD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EC+NY,cAAA;EACA,iBAAA;CPgdX;AMhrBD;;ECoOQ,kBAAA;CPgdP;AMprBD;;;;;;ECwOQ,mBAAA;CPodP;AM5rBD;;EC4OQ,yBAAA;CPodP;AMhsBD;;;;ECgPQ,sBAAA;CPsdP;AMtsBD;;;;;;;;;;;;ECsPY,wBAAA;CP8dX;AMptBD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECkQgB,wBAAA;CPwff;AM1vBD;;ECsQQ,eAAA;CPwfP;AM9vBD;;EC0QQ,uBAAA;CPwfP;AMlwBD;;;;EC8QQ,4BAAA;CP0fP;AMxwBD;;;;ECkRQ,sBAAA;CP4fP;AM9wBD;;;;ECsRQ,4BAAA;CP8fP;AMpxBD;;;;EC0RQ,yBAAA;CPggBP;AM1xBD;;;;EC8RQ,4BAAA;CPkgBP;AMhyBD;;;;ECkSQ,mBAAA;CPogBP;AMtyBD;;;;;;;;ECsSQ,oBAAA;CP0gBP;AMhzBD;;;;;;;;;;;;;;;;EC2SY,qBAAA;CPuhBX;AMl0BD;;EC+SQ,uBAAA;EACA,oBAAA;EACA,0BAAA;EACA,qBAAA;CPuhBP;AMz0BD;;;;ECsTQ,aAAA;CPyhBP;AM/0BD;;EC0TQ,kBAAA;CPyhBP;AMn1BD;;;;;;;;EC8TQ,uBAAA;CP+hBP;AM71BD;;;;;;ECkUQ,wBAAA;CPmiBP;AMr2BD;;;;;;ECsUQ,mBAAA;CPuiBP;AM72BD;;;;;;;;EC0UQ,sBAAA;CP6iBP;AMv3BD;;;;;;;;;;;;;;ECwWQ,qBAAA;CP+hBP;AMv4BD;;EC4WQ,sBAAA;CP+hBP;AM34BD;;;;;;;;;;;;;;ECiXQ,uBAAA;CP0iBP;AM35BD;;;;ECqXQ,qBAAA;CP4iBP;AMj6BD;;ECyXQ,YAAA;EACA,oBAAA;EACA,kBAAA;EACA,mBAAA;CP4iBP;AMx6BD;;ECgYQ,iBAAA;EACA,kBAAA;EACA,8BAAA;EACA,+BAAA;CP4iBP;AM/6BD;;ECuYQ,kBAAA;EACA,mBAAA;CP4iBP;AMp7BD;;EAeoB,yBAAA;CNy6BnB;AMx7BD;;EAmBoB,4BAAA;CNy6BnB;AMp6BO;;;;EAEQ,eAAA;EACA,YAAA;EACA,gBAAA;EACA,aAAA;CNw6Bf;AMr8BD;;EAmCQ,eAAA;EACA,YAAA;EACA,gBAAA;EACA,aAAA;CNs6BP;;AQ78BD;EAEQ,mBAAA;EACA,kBAAA;EACA,aAAA;EACA,uBAAA;MAAA,qBAAA;UAAA,eAAA;CR+8BP;AQp9BD;EAQY,0BAAA;EACA,aAAA;EACA,kBAAA;EACA,aAAA;EACA,0BAAA;CR+8BX;AQ78BW;EACI,gBAAA;EACA,OAAA;EACA,iBAAA;EACA,iBAAA;CR+8Bf;AQ58BW;EACI,mBAAA;EACA,UAAA;CR88Bf;AQr+BD;EA2BgB,iBAAA;CR68Bf;AQx+BD;EA8BoB,eAAA;EACA,mBAAA;EACA,sBAAA;CR68BnB;AQ38BmB;EACI,sBAAA;EACA,YAAA;EACA,kBAAA;EACA,8CAAA;EACA,mBAAA;EACA,iBAAA;CR68BvB;AQz8Be;EAEQ,iBAAA;CR08BvB;AQ58Be;EAMQ,cAAA;CRy8BvB;AQ3/BD;EAuDoB,gBAAA;CRu8BnB;AQ9/BD;EA2DoB,0BAAA;EACA,uBAAA;EACA,kBAAA;EACA,gBAAA;CRs8BnB;AQpgCD;EAiEwB,sBAAA;EACA,YAAA;EACA,8CAAA;CRs8BvB;AQzgCD;EAuEwB,gBAAA;CRq8BvB;AQ5gCD;EA4EoB,gBAAA;CRm8BnB;;AS/gCD;EAEQ,cAAA;EACA,mBAAA;EACA,UAAA;EAGA,iDAAA;EACA,0BAAA;EACA,eAAA;EACA,0BAAA;EACA,iBAAA;EACA,YAAA;EACA,eAAA;EACA,iBAAA;EACA,gBAAA;EACA,oBAAA;EACA,oBAAA;EACA,iBAAA;CTihCP;AS/gCO;EACI,aAAA;CTihCX;AStiCD;EAyBY,sBAAA;EAAA,qBAAA;EAAA,cAAA;EACA,iBAAA;CTghCX;AS1iCD;EA6BgB,iBAAA;EACA,iDAAA;EACA,kBAAA;CTghCf;AS/iCD;EAoCY,iBAAA;CT8gCX;ASljCD;EAwCY,iBAAA;EACA,kBAAA;EACA,eAAA;CT6gCX;ASvjCD;EA8CY,0BAAA;EACA,gBAAA;CT4gCX;AS3jCD;;EAoDgB,YAAA;CT2gCf;AS/jCD;EAwDgB,eAAA;EACA,iBAAA;CT0gCf","file":"LayoutEditor.css","sourcesContent":["@import \"Variables.less\";\r\n\r\n.layout-editor-toolbar {\r\n    display: flex;\r\n    justify-content: space-between;\r\n    position: relative;\r\n    top: 10px;\r\n\r\n    .layout-editor-toolbar-group {\r\n        display: flex;\r\n\r\n        > li + li {\r\n            margin-left: 12px;\r\n        }\r\n    }\r\n}\r\n\r\n.layout-editor {\r\n    display: flex;\r\n    margin-top: 1em;\r\n    font-size: @font-size;\r\n    align-items: stretch;\r\n\r\n    > .layout-canvas-wrapper {\r\n        flex-grow: 1;\r\n        background-color: @gray-bg;\r\n        border: 1px solid @gray-border;\r\n\r\n        > .layout-toolbar-container {\r\n            display: none;\r\n            margin: @container-padding @container-padding 0;\r\n            min-height: 71px;\r\n\r\n            > .mce-panel {\r\n                width: 100% !important;\r\n            }\r\n        }\r\n\r\n        .layout-content > .layout-element-wrapper .layout-content-markup > .layout-placeholder {\r\n            border: 1px dashed #ccc;\r\n            padding: 0.2em 0.4em;\r\n            background: #e8e8e8;\r\n        }\r\n    }\r\n\r\n    .layout-snippet {\r\n        background: #e8e8e8;\r\n    }\r\n}\r\n\r\n.layout-editor-help-dialog {\r\n    display: none;\r\n\r\n    .help-row {\r\n        &:before, &:after {\r\n            content: \" \"; // 1\r\n            display: table; // 2\r\n        }\r\n\r\n        &:after {\r\n            clear: both;\r\n        }\r\n\r\n        > .help-column-full, > .help-column-half {\r\n            margin: 0.5em 0;\r\n        }\r\n\r\n        > .help-column-half {\r\n            box-sizing: border-box;\r\n            float: left;\r\n            width: 50%;\r\n\r\n            &:nth-child(2n) {\r\n                padding-right: 10px;\r\n                clear: left;\r\n            }\r\n        }\r\n\r\n        + .help-row {\r\n            margin-top: 1em;\r\n        }\r\n    }\r\n\r\n    code {\r\n        border-radius: 4px;\r\n        background-color: #f3f4f5;\r\n        padding: 2px 4px;\r\n        font-family: monospace;\r\n    }\r\n\r\n    p {\r\n        margin-bottom: 0.5em;\r\n        line-height: 1.6em;\r\n    }\r\n\r\n    table > tbody > tr > td:first-child {\r\n        padding-right: 10px;\r\n    }\r\n}\r\n",".layout-editor-toolbar {\n  display: flex;\n  justify-content: space-between;\n  position: relative;\n  top: 10px;\n}\n.layout-editor-toolbar .layout-editor-toolbar-group {\n  display: flex;\n}\n.layout-editor-toolbar .layout-editor-toolbar-group > li + li {\n  margin-left: 12px;\n}\n.layout-editor {\n  display: flex;\n  margin-top: 1em;\n  font-size: 14px;\n  align-items: stretch;\n}\n.layout-editor > .layout-canvas-wrapper {\n  flex-grow: 1;\n  background-color: #f3f4f5;\n  border: 1px solid #e4e5e6;\n}\n.layout-editor > .layout-canvas-wrapper > .layout-toolbar-container {\n  display: none;\n  margin: 12px 12px 0;\n  min-height: 71px;\n}\n.layout-editor > .layout-canvas-wrapper > .layout-toolbar-container > .mce-panel {\n  width: 100% !important;\n}\n.layout-editor > .layout-canvas-wrapper .layout-content > .layout-element-wrapper .layout-content-markup > .layout-placeholder {\n  border: 1px dashed #ccc;\n  padding: 0.2em 0.4em;\n  background: #e8e8e8;\n}\n.layout-editor .layout-snippet {\n  background: #e8e8e8;\n}\n.layout-editor-help-dialog {\n  display: none;\n}\n.layout-editor-help-dialog .help-row:before,\n.layout-editor-help-dialog .help-row:after {\n  content: \" \";\n  display: table;\n}\n.layout-editor-help-dialog .help-row:after {\n  clear: both;\n}\n.layout-editor-help-dialog .help-row > .help-column-full,\n.layout-editor-help-dialog .help-row > .help-column-half {\n  margin: 0.5em 0;\n}\n.layout-editor-help-dialog .help-row > .help-column-half {\n  box-sizing: border-box;\n  float: left;\n  width: 50%;\n}\n.layout-editor-help-dialog .help-row > .help-column-half:nth-child(2n) {\n  padding-right: 10px;\n  clear: left;\n}\n.layout-editor-help-dialog .help-row + .help-row {\n  margin-top: 1em;\n}\n.layout-editor-help-dialog code {\n  border-radius: 4px;\n  background-color: #f3f4f5;\n  padding: 2px 4px;\n  font-family: monospace;\n}\n.layout-editor-help-dialog p {\n  margin-bottom: 0.5em;\n  line-height: 1.6em;\n}\n.layout-editor-help-dialog table > tbody > tr > td:first-child {\n  padding-right: 10px;\n}\n\n.layout-editor .layout-element {\n  position: relative;\n  margin-top: 0;\n  margin-right: 0;\n  margin-bottom: 0;\n  padding: 0;\n}\n.layout-editor .layout-element:not(.layout-column) {\n  margin-left: 0;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel {\n  display: none;\n  position: absolute;\n  margin: 0;\n  z-index: 20;\n  height: 25px;\n  padding: 0 6px;\n  list-style: none;\n  white-space: nowrap;\n  line-height: 25px;\n  vertical-align: middle;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel > .layout-panel-item {\n  display: inline-block;\n  height: 25px;\n  padding: 1px 6px 0;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel > .layout-panel-label {\n  font-size: 13px;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel > .layout-panel-action {\n  display: none;\n  width: 28px;\n  cursor: pointer;\n  text-align: center;\n}\n.layout-editor .layout-element > .layout-element-wrapper > .layout-panel-main {\n  top: -27px;\n  left: -2px;\n}\n.layout-editor li.layout-element {\n  list-style: none;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active,\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused {\n  border-width: 2px;\n  border-style: solid;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active > .layout-element-wrapper,\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper {\n  margin: -2px;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active {\n  border-color: rgba(104, 104, 104, 0.1);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-active > .layout-element-wrapper > .layout-panel-main {\n  display: block;\n  z-index: 30;\n  background-color: rgba(104, 104, 104, 0.1);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused {\n  border-color: #648721;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel {\n  display: block;\n  background-color: #648721;\n  color: #fefefe;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action {\n  display: inline-block;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action:hover {\n  background-color: #82b02b;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.disabled {\n  cursor: default;\n  color: rgba(254, 254, 254, 0.4);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.disabled:hover {\n  background-color: #648721;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.active {\n  color: #deff42;\n  background-color: #739b26;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-focused > .layout-element-wrapper > .layout-panel > .layout-panel-action.active:hover {\n  background-color: #82b02b;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-element-selected {\n  background-color: rgba(100, 135, 33, 0.08);\n}\n.layout-editor .ui-sortable-placeholder {\n  display: none;\n}\n.layout-editor .layout-element-droptarget {\n  box-shadow: inset 0 0 12px 6px rgba(100, 135, 33, 0.5);\n}\n.layout-editor .layout-element-droptarget .ui-sortable-placeholder {\n  display: block;\n  visibility: visible !important;\n  min-height: 78px;\n  border: 2px dashed #648721 !important;\n  background-color: rgba(100, 135, 33, 0.16);\n}\n.layout-editor .media-thumbnail img {\n  max-width: 100%;\n  max-height: 100%;\n}\n\n.layout-editor .layout-container > .layout-element-wrapper > .layout-container-children-placeholder {\n  display: none;\n  flex-direction: column;\n  margin: 11px;\n  min-height: 80px;\n  border: 1px dashed rgba(124, 124, 124, 0.4);\n  border-radius: 4px;\n  padding: 12px;\n  justify-content: center;\n  font-size: 13px;\n  font-style: italic;\n  opacity: 0.6;\n  text-align: center;\n}\n.layout-editor .layout-container > .layout-element-wrapper > .layout-children {\n  padding: 12px;\n}\n.layout-editor .layout-container > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-container) + .layout-element:not(.layout-container) {\n  margin-top: 12px;\n}\n.layout-editor .layout-container > .layout-element-wrapper > .layout-children > .ui-sortable-helper:first-child + .layout-element:not(.layout-container) {\n  margin-top: 0;\n}\n.layout-editor .layout-container > .layout-element-wrapper.layout-container-empty > .layout-container-children-placeholder {\n  display: flex;\n}\n.layout-editor .layout-container > .layout-element-wrapper.layout-container-empty > .layout-children {\n  position: absolute;\n  top: 0;\n  right: 0;\n  bottom: 0;\n  left: 0;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) {\n  border: 1px dashed rgba(124, 124, 124, 0.6);\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) > .layout-element-wrapper {\n  margin: -1px;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-container + .layout-container:not(.layout-element-active) {\n  border-top: 0;\n}\n.layout-editor:not(.layout-editor-dragging) .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children > .layout-container + .layout-container:not(.layout-element-active) > .layout-element-wrapper {\n  margin-top: 0;\n}\n\n.layout-editor .layout-canvas #dummy {\n  display: none;\n}\n\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) {\n  border-top: 1px dashed rgba(124, 124, 124, 0.6) !important;\n}\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element:not(.layout-element-active) > .layout-element-wrapper {\n  margin-top: -1px !important;\n}\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element + .layout-element:not(.layout-element-active) {\n  border-left: 0 !important;\n}\n.layout-editor .layout-row.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children > .layout-element + .layout-element:not(.layout-element-active) > .layout-element-wrapper {\n  margin-left: 0 !important;\n}\n\n.layout-editor .layout-column > .layout-element-wrapper > .layout-column-resize-bar {\n  display: none;\n  position: absolute;\n  z-index: 30;\n  top: 0;\n  width: 16px;\n  height: 100%;\n  cursor: col-resize;\n}\n.layout-editor .layout-column > .layout-element-wrapper > .layout-column-resize-bar-left {\n  left: -6px;\n}\n.layout-editor .layout-column > .layout-element-wrapper > .layout-column-resize-bar-right {\n  right: -6px;\n}\n.layout-editor .layout-column.layout-element-focused > .layout-element-wrapper > .layout-column-resize-bar {\n  display: block;\n}\n\n.layout-editor .layout-content,\n.layout-editor .layout-html {\n  min-height: 1em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup {\n  padding: 2px;\n  overflow-x: hidden;\n  line-height: normal;\n  /* this color is just a suggestion and can be changed based on implementation feedback */\n  /* this also has bidi implications */\n  /* this also has bidi implications */\n  /* LTR-specific: use 'margin-right' for rtl elements */\n  /* LTR-specific: use 'padding-right' for rtl elements */\n  /*table[rules=none i], table[rules=groups i], table[rules=rows i],\ntable[rules=cols i], table[rules=all i], table[frame=void i],\ntable[frame=above i], table[frame=below i], table[frame=hsides i],\ntable[frame=lhs i], table[frame=rhs i], table[frame=vsides i],\ntable[frame=box i], table[frame=border i],\ntable[rules=none i] > tr > td, table[rules=none i] > tr > th,\ntable[rules=groups i] > tr > td, table[rules=groups i] > tr > th,\ntable[rules=rows i] > tr > td, table[rules=rows i] > tr > th,\ntable[rules=cols i] > tr > td, table[rules=cols i] > tr > th,\ntable[rules=all i] > tr > td, table[rules=all i] > tr > th,\ntable[rules=none i] > thead > tr > td, table[rules=none i] > thead > tr > th,\ntable[rules=groups i] > thead > tr > td, table[rules=groups i] > thead > tr > th,\ntable[rules=rows i] > thead > tr > td, table[rules=rows i] > thead > tr > th,\ntable[rules=cols i] > thead > tr > td, table[rules=cols i] > thead > tr > th,\ntable[rules=all i] > thead > tr > td, table[rules=all i] > thead > tr > th,\ntable[rules=none i] > tbody > tr > td, table[rules=none i] > tbody > tr > th,\ntable[rules=groups i] > tbody > tr > td, table[rules=groups i] > tbody > tr > th,\ntable[rules=rows i] > tbody > tr > td, table[rules=rows i] > tbody > tr > th,\ntable[rules=cols i] > tbody > tr > td, table[rules=cols i] > tbody > tr > th,\ntable[rules=all i] > tbody > tr > td, table[rules=all i] > tbody > tr > th,\ntable[rules=none i] > tfoot > tr > td, table[rules=none i] > tfoot > tr > th,\ntable[rules=groups i] > tfoot > tr > td, table[rules=groups i] > tfoot > tr > th,\ntable[rules=rows i] > tfoot > tr > td, table[rules=rows i] > tfoot > tr > th,\ntable[rules=cols i] > tfoot > tr > td, table[rules=cols i] > tfoot > tr > th,\ntable[rules=all i] > tfoot > tr > td, table[rules=all i] > tfoot > tr > th {\n    border-color: black;\n}*/\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup *,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup * {\n  margin: 0;\n  padding: 0;\n  box-sizing: content-box;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup address,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup address,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup center,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup center,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup div,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup div,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figcaption,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figcaption,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup footer,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup footer,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup form,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup form,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup header,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup header,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup hr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup hr,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup legend,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup legend,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup xmp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup xmp {\n  display: block;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup p,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup xmp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup xmp {\n  margin-top: 1em;\n  margin-bottom: 1em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blockquote,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup figure,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup figure {\n  margin-left: 40px;\n  margin-right: 40px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup address,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup address {\n  font-style: italic;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup listing,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup plaintext,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup pre,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup xmp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup xmp {\n  font-family: monospace;\n  white-space: pre;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup cite,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup cite,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dfn,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dfn,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup em,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup em,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup i,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup i,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup var,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup var {\n  font-style: italic;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup b,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup b,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup strong,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup strong {\n  font-weight: bolder;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup code,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup code,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup kbd,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup kbd,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup samp,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup samp,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tt {\n  font-family: monospace;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup big,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup big {\n  font-size: larger;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup small,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup small {\n  font-size: smaller;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sub,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sub {\n  vertical-align: sub;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sup {\n  vertical-align: super;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sub,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sub,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup sup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup sup {\n  line-height: normal;\n  font-size: smaller;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ruby,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ruby {\n  display: ruby;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rb,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rb {\n  display: ruby-base;\n  white-space: nowrap;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rt {\n  display: ruby-text;\n  white-space: nowrap;\n  font-size: 50%;\n  font-variant-east-asian: ruby;\n  text-emphasis: none;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rbc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rbc {\n  display: ruby-base-container;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rtc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rtc {\n  display: ruby-text-container;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ruby,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ruby,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rb,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rb,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rt,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rbc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rbc,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup rtc,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup rtc {\n  unicode-bidi: isolate;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :link,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :link {\n  color: #0000EE;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :visited,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :visited {\n  color: #551A8B;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :link,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :link,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :visited,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :visited {\n  text-decoration: underline;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup a:link[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup a:link[rel~=help],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup a:visited[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup a:visited[rel~=help],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup area:link[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup area:link[rel~=help],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup area:visited[rel~=help],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup area:visited[rel~=help] {\n  cursor: help;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup :focus,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup :focus {\n  outline: auto;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup mark,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup mark {\n  background: yellow;\n  color: black;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup abbr[title],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup abbr[title],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup acronym[title],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup acronym[title] {\n  text-decoration: dotted underline;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ins,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ins,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup u,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup u {\n  text-decoration: underline;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup del,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup del,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup s,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup s,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup strike,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup strike {\n  text-decoration: line-through;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup blink,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup blink {\n  text-decoration: blink;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup q::before,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup q::before {\n  content: open-quote;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup q::after,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup q::after {\n  content: close-quote;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup br,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup br {\n  content: '\\A';\n  white-space: pre;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup nobr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup nobr {\n  white-space: nowrap;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup wbr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup wbr {\n  content: '\\200B';\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup nobr wbr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup nobr wbr {\n  white-space: normal;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup article,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup article,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup aside,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup aside,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h1,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h1,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h2,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h2,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h3,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h3,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h4,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h4,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h5,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h5,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h6,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h6,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup hgroup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup hgroup,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup nav,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup nav,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup section,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup section {\n  display: block;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h1,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h1 {\n  margin-top: 0.67em;\n  margin-bottom: 0.67em;\n  font-size: 2.00em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h2,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h2 {\n  margin-top: 0.83em;\n  margin-bottom: 0.83em;\n  font-size: 1.50em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h3,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h3 {\n  margin-top: 1.00em;\n  margin-bottom: 1.00em;\n  font-size: 1.17em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h4,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h4 {\n  margin-top: 1.33em;\n  margin-bottom: 1.33em;\n  font-size: 1.00em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h5,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h5 {\n  margin-top: 1.67em;\n  margin-bottom: 1.67em;\n  font-size: 0.83em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup h6,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup h6 {\n  margin-top: 2.33em;\n  margin-bottom: 2.33em;\n  font-size: 0.67em;\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dd,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dd,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dt,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dt,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  display: block;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup li,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup li {\n  display: list-item;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  margin-top: 1em;\n  margin-bottom: 1em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dl ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dl ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dl,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dl,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul {\n  margin-top: 0;\n  margin-bottom: 0;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dd,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dd {\n  margin-left: 40px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  padding-left: 40px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol {\n  list-style-type: decimal;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul {\n  list-style-type: disc;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul {\n  list-style-type: circle;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup dir ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup dir ul ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ol ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ol ul ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul dir ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul dir ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ol dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ol dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ol ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ol ul,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul dir,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul dir,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup ul ul ul,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup ul ul ul {\n  list-style-type: square;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table {\n  display: table;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup caption,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup caption {\n  display: table-caption;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup colgroup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup colgroup,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup colgroup[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup colgroup[hidden] {\n  display: table-column-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup col,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup col,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup col[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup col[hidden] {\n  display: table-column;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead[hidden] {\n  display: table-header-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody[hidden] {\n  display: table-row-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot[hidden] {\n  display: table-footer-group;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr[hidden] {\n  display: table-row;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th[hidden] {\n  display: table-cell;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup colgroup[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup colgroup[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup col[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup col[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td[hidden],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th[hidden],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th[hidden] {\n  visibility: collapse;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table {\n  box-sizing: border-box;\n  border-spacing: 2px;\n  border-collapse: separate;\n  text-indent: initial;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  padding: 1px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  font-weight: bold;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table > tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table > tr {\n  vertical-align: middle;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  vertical-align: inherit;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup table,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup td,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup th,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup th {\n  border-color: gray;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup thead,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tbody,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tfoot,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup tr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup tr {\n  border-color: inherit;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup option,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup option,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup optgroup,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup optgroup,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup textarea,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup textarea,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup keygen,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup keygen {\n  text-indent: initial;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup textarea,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup textarea {\n  white-space: pre-wrap;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"radio\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"radio\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"checkbox\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"checkbox\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"reset\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"reset\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"submit\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"submit\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup select,\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup button {\n  box-sizing: border-box;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup input[type=\"button\"],\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup button,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup button {\n  padding: 0.3em 0.5em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup hr,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup hr {\n  color: gray;\n  border-style: inset;\n  border-width: 1px;\n  margin: 0.5em auto;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup fieldset,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup fieldset {\n  margin-left: 2px;\n  margin-right: 2px;\n  border: groove 2px ThreeDFace;\n  padding: 0.35em 0.625em 0.75em;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup legend,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup legend {\n  padding-left: 2px;\n  padding-right: 2px;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup > *:first-child,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup > *:first-child {\n  margin-top: 0 !important;\n}\n.layout-editor .layout-content > .layout-element-wrapper .layout-content-markup > *:last-child,\n.layout-editor .layout-html > .layout-element-wrapper .layout-content-markup > *:last-child {\n  margin-bottom: 0 !important;\n}\n.layout-editor .layout-content.layout-content-image > .layout-element-wrapper > .layout-content-markup > img,\n.layout-editor .layout-html.layout-content-image > .layout-element-wrapper > .layout-content-markup > img,\n.layout-editor .layout-content.layout-content-vector-image > .layout-element-wrapper > .layout-content-markup > img,\n.layout-editor .layout-html.layout-content-vector-image > .layout-element-wrapper > .layout-content-markup > img {\n  display: block;\n  width: 100%;\n  max-width: 100%;\n  height: auto;\n}\n.layout-editor .img-responsive,\n.layout-editor .img-responsive img {\n  display: block;\n  width: 100%;\n  max-width: 100%;\n  height: auto;\n}\n\n.layout-editor > .layout-toolbox-wrapper {\n  position: relative;\n  margin-left: 12px;\n  width: 220px;\n  flex-shrink: 0;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox {\n  border: 1px solid #e4e5e6;\n  width: 220px;\n  min-height: 400px;\n  padding: 6px;\n  background-color: #f3f4f5;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox.sticky-top {\n  position: fixed;\n  top: 0;\n  max-height: 100%;\n  overflow-y: auto;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox.sticky-bottom {\n  position: absolute;\n  bottom: 0;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group {\n  margin-top: 12px;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-group-heading {\n  display: block;\n  margin-bottom: 4px;\n  text-decoration: none;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-group-heading:before {\n  display: inline-block;\n  width: 10px;\n  margin-right: 4px;\n  font: normal normal normal 14px/1 FontAwesome;\n  text-align: center;\n  content: \"\\f0d7\";\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group.collapsed .layout-toolbox-group-heading:before {\n  content: \"\\f0da\";\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group.collapsed .layout-toolbox-items {\n  display: none;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-section + .layout-toolbox-section {\n  margin-top: 4px;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-item {\n  border: 1px solid #e4e5e6;\n  background-color: #fff;\n  padding: 9px 12px;\n  cursor: default;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-item i {\n  display: inline-block;\n  width: 16px;\n  font: normal normal normal 14px/1 FontAwesome;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group .layout-toolbox-item + .layout-toolbox-item {\n  margin-top: 4px;\n}\n.layout-editor > .layout-toolbox-wrapper > .layout-toolbox .layout-toolbox-group + .layout-toolbox-group {\n  margin-top: 6px;\n}\n\n.layout-editor .layout-popup {\n  display: none;\n  position: absolute;\n  margin: 0;\n  -moz-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\n  -webkit-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\n  box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\n  border: 1px solid #e4e5e6;\n  padding: 2px 0;\n  background-color: #f7f7f7;\n  list-style: none;\n  z-index: 20;\n  color: #7c7c7c;\n  text-align: left;\n  cursor: default;\n  white-space: nowrap;\n  line-height: normal;\n  min-width: 300px;\n}\n.layout-editor .layout-popup.wide {\n  width: 600px;\n}\n.layout-editor .layout-popup .layout-popup-flex {\n  display: flex;\n  padding: 2px 5px;\n}\n.layout-editor .layout-popup .layout-popup-flex .layout-popup-column + .layout-popup-column {\n  margin-left: 4px;\n  border-left: 1px solid rgba(128, 128, 128, 0.15);\n  padding-left: 4px;\n}\n.layout-editor .layout-popup .layout-popup-item {\n  padding: 4px 6px;\n}\n.layout-editor .layout-popup .layout-popup-label {\n  font-size: 0.9em;\n  font-weight: bold;\n  color: #7c7c7c;\n}\n.layout-editor .layout-popup .layout-popup-action:hover {\n  background-color: #f3f4f5;\n  cursor: pointer;\n}\n.layout-editor .layout-popup .layout-popup-input input[type='text'],\n.layout-editor .layout-popup .layout-popup-input textarea {\n  width: 100%;\n}\n.layout-editor .layout-popup .layout-popup-input > label {\n  display: block;\n  font-size: 0.9em;\n}\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-element {\r\n        position: relative;\r\n        margin-top: 0;\r\n        margin-right: 0;\r\n        margin-bottom: 0;\r\n        padding: 0;\r\n\r\n        // We don't fuck with the left margin of columns, because Bootstrap uses these to render column offsets.\r\n        &:not(.layout-column) {\r\n            margin-left: 0;\r\n        }\r\n\r\n        > .layout-element-wrapper {\r\n            > .layout-panel {\r\n                display: none; // Shown only in active or focused states.\r\n                position: absolute;\r\n                margin: 0;\r\n                z-index: 20;\r\n                height: 25px;\r\n                padding: 0 6px;\r\n                list-style: none;\r\n                white-space: nowrap;\r\n                line-height: 25px;\r\n                vertical-align: middle;\r\n\r\n                > .layout-panel-item {\r\n                    display: inline-block;\r\n                    height: 25px;\r\n                    padding: 1px 6px 0;\r\n                }\r\n\r\n                > .layout-panel-label {\r\n                    font-size: @font-size - 1;\r\n                }\r\n\r\n                > .layout-panel-action {\r\n                    display: none; // Shown only in focused state.\r\n                    width: 28px;\r\n                    cursor: pointer;\r\n                    text-align: center;\r\n                }\r\n            }\r\n\r\n            > .layout-panel-main {\r\n                top: -27px;\r\n                left: -2px;\r\n            }\r\n        }\r\n    }\r\n\r\n    // When dragging from toolbox, elements will be li rather than div.\r\n    li.layout-element {\r\n        list-style: none;\r\n    }\r\n\r\n    &:not(.layout-editor-dragging) {\r\n\r\n        .layout-element-active, .layout-element-focused {\r\n            border-width: 2px;\r\n            border-style: solid;\r\n\r\n            > .layout-element-wrapper {\r\n                margin: -2px;\r\n            }\r\n        }\r\n\r\n        .layout-element-active {\r\n            @active-highlight: fade(#686868, 10%);\r\n\r\n            border-color: @active-highlight;\r\n\r\n            > .layout-element-wrapper > .layout-panel-main {\r\n                display: block; // To reveal.\r\n                z-index: 30;\r\n                background-color: @active-highlight;\r\n            }\r\n        }\r\n\r\n        .layout-element-focused {\r\n            @focused-highlight: #648721;\r\n            @focused-text: #fefefe;\r\n\r\n            border-color: @focused-highlight;\r\n\r\n            > .layout-element-wrapper > .layout-panel {\r\n                display: block; // To reveal.\r\n                background-color: @focused-highlight;\r\n                color: @focused-text;\r\n\r\n                > .layout-panel-action {\r\n                    display: inline-block; // To reveal.\r\n                    &:hover {\r\n                        background-color: lighten(@focused-highlight, 10%);\r\n                    }\r\n\r\n                    &.disabled {\r\n                        cursor: default;\r\n                        color: fade(@focused-text, 40%);\r\n\r\n                        &:hover {\r\n                            background-color: @focused-highlight;\r\n                        }\r\n                    }\r\n\r\n                    &.active {\r\n                        color: lighten(saturate(spin(@focused-highlight, -10), 100%), 30%);\r\n                        background-color: lighten(@focused-highlight, 5%);\r\n\r\n                        &:hover {\r\n                            background-color: lighten(@focused-highlight, 10%);\r\n                        }\r\n                    }\r\n                }\r\n            }\r\n        }\r\n\r\n        .layout-element-selected {\r\n            @selected-highlight: #648721;\r\n            background-color: fade(@selected-highlight, 8%);\r\n        }\r\n    }\r\n\r\n    .ui-sortable-placeholder {\r\n        display: none;\r\n    }\r\n\r\n    .layout-element-droptarget {\r\n        @droptarget-highlight: #648721;\r\n        box-shadow: inset 0 0 12px 6px fade(@droptarget-highlight, 50%);\r\n\r\n        .ui-sortable-placeholder {\r\n            display: block; // To reveal.\r\n            visibility: visible !important;\r\n            min-height: @element-min-height - 2px;\r\n            border: 2px dashed @droptarget-highlight !important;\r\n            background-color: fade(@droptarget-highlight, 16%);\r\n        }\r\n    }\r\n\r\n    // A CSS fix for media item elements.\r\n    .media-thumbnail {\r\n        img {\r\n            max-width:100%;\r\n            max-height: 100%;\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-container {\r\n\r\n        > .layout-element-wrapper {\r\n\r\n            > .layout-container-children-placeholder {\r\n                display: none;\r\n                flex-direction: column;\r\n                margin: @container-padding - 1px;\r\n                min-height: @element-min-height;\r\n                border: 1px dashed fade(@gray-text, 40%);\r\n                border-radius: 4px;\r\n                padding: @container-padding;\r\n                justify-content: center;\r\n                font-size: @font-size - 1;\r\n                font-style: italic;\r\n                opacity: 0.6;\r\n                text-align: center;\r\n            }\r\n\r\n            > .layout-children {\r\n                padding: @container-padding;\r\n\r\n                // All adjacent non-container children need some space between...\r\n                > .layout-element:not(.layout-container) + .layout-element:not(.layout-container) {\r\n                    margin-top: @content-spacing;\r\n                }\r\n\r\n                // ... except when jQuery UI sortable helper is the first element (because that one is absolutely positioned and \"floating\").\r\n                > .ui-sortable-helper:first-child + .layout-element:not(.layout-container) {\r\n                    margin-top: 0;\r\n                }\r\n            }\r\n\r\n            &.layout-container-empty {\r\n                > .layout-container-children-placeholder {\r\n                    display: flex; // To reveal.\r\n                }\r\n\r\n                > .layout-children {\r\n                    position: absolute;\r\n                    top: 0;\r\n                    right: 0;\r\n                    bottom: 0;\r\n                    left: 0;\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    &:not(.layout-editor-dragging) {\r\n        .layout-container.layout-element-focused > .layout-element-wrapper > .layout-children {\r\n            > .layout-element:not(.layout-element-active) {\r\n                border: 1px dashed fade(@gray-text, 60%);\r\n\r\n                > .layout-element-wrapper {\r\n                    margin: -1px;\r\n                }\r\n            }\r\n\r\n            > .layout-container + .layout-container:not(.layout-element-active) {\r\n                border-top: 0;\r\n\r\n                > .layout-element-wrapper {\r\n                    margin-top: 0;\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-canvas {\r\n        #dummy { // Only added because WE doesn't compile if there are no rules.\r\n            display: none;\r\n        }\r\n    }\r\n}","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-row {\r\n        &.layout-element-focused:not(.layout-element-droptarget) > .layout-element-wrapper > .layout-children {\r\n            > .layout-element:not(.layout-element-active) {\r\n                border-top: 1px dashed fade(@gray-text, 60%) !important;\r\n                > .layout-element-wrapper {\r\n                    margin-top: -1px !important;\r\n                }\r\n            }\r\n\r\n            > .layout-element + .layout-element:not(.layout-element-active) {\r\n                border-left: 0 !important;\r\n                > .layout-element-wrapper {\r\n                    margin-left: 0 !important;\r\n                }\r\n            }\r\n        }\r\n    }\r\n}","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-column {\r\n        > .layout-element-wrapper {\r\n            > .layout-column-resize-bar {\r\n                display: none;\r\n                position: absolute;\r\n                z-index: 30;\r\n                top: 0;\r\n                width: 16px;\r\n                height: 100%;\r\n                cursor: col-resize;\r\n            }\r\n\r\n            > .layout-column-resize-bar-left {\r\n                left: -6px;\r\n            }\r\n\r\n            > .layout-column-resize-bar-right {\r\n                right: -6px;\r\n            }\r\n        }\r\n\r\n        &.layout-element-focused > .layout-element-wrapper > .layout-column-resize-bar {\r\n            display: block;\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n@import \"Reset.less\";\r\n\r\n.layout-editor {\r\n    .layout-content, .layout-html {\r\n        min-height: 1em;\r\n\r\n        > .layout-element-wrapper {\r\n            .layout-content-markup {\r\n\r\n                padding: 2px;\r\n                overflow-x: hidden;\r\n                line-height: normal;\r\n\r\n                // Reset to HTML5 W3C standard default styling within content.\r\n                .reset();\r\n\r\n                > *:first-child {\r\n                    margin-top: 0 !important; // Important because site.css of the admin theme styles heading margins with a very high specificity.\r\n                }\r\n\r\n                > *:last-child {\r\n                    margin-bottom: 0 !important;\r\n                }\r\n            }\r\n        }\r\n\r\n        &.layout-content-image, &.layout-content-vector-image {\r\n            > .layout-element-wrapper > .layout-content-markup > img {\r\n                display: block;\r\n                width: 100%;\r\n                max-width: 100%;\r\n                height: auto;\r\n            }\r\n        }\r\n    }\r\n\r\n    .img-responsive, .img-responsive img {\r\n        display: block;\r\n        width: 100%;\r\n        max-width: 100%;\r\n        height: auto;\r\n    }\r\n}\r\n","﻿.reset() {\r\n\r\n    * {\r\n        margin: 0;\r\n        padding: 0;\r\n        box-sizing: content-box;\r\n    }\r\n\r\n    address, blockquote, center, div, figure, figcaption, footer, form,\r\n    header, hr, legend, listing, p, plaintext, pre, xmp {\r\n        display: block;\r\n    }\r\n\r\n    blockquote, figure, listing, p, plaintext, pre, xmp {\r\n        margin-top: 1em;\r\n        margin-bottom: 1em;\r\n    }\r\n\r\n    blockquote, figure {\r\n        margin-left: 40px;\r\n        margin-right: 40px;\r\n    }\r\n\r\n    address {\r\n        font-style: italic;\r\n    }\r\n\r\n    listing, plaintext, pre, xmp {\r\n        font-family: monospace;\r\n        white-space: pre;\r\n    }\r\n\r\n    cite, dfn, em, i, var {\r\n        font-style: italic;\r\n    }\r\n\r\n    b, strong {\r\n        font-weight: bolder;\r\n    }\r\n\r\n    code, kbd, samp, tt {\r\n        font-family: monospace;\r\n    }\r\n\r\n    big {\r\n        font-size: larger;\r\n    }\r\n\r\n    small {\r\n        font-size: smaller;\r\n    }\r\n\r\n    sub {\r\n        vertical-align: sub;\r\n    }\r\n\r\n    sup {\r\n        vertical-align: super;\r\n    }\r\n\r\n    sub, sup {\r\n        line-height: normal;\r\n        font-size: smaller;\r\n    }\r\n\r\n\r\n    ruby {\r\n        display: ruby;\r\n    }\r\n\r\n    rb {\r\n        display: ruby-base;\r\n        white-space: nowrap;\r\n    }\r\n\r\n    rt {\r\n        display: ruby-text;\r\n        white-space: nowrap;\r\n        font-size: 50%;\r\n        font-variant-east-asian: ruby;\r\n        text-emphasis: none;\r\n    }\r\n\r\n    rbc {\r\n        display: ruby-base-container;\r\n    }\r\n\r\n    rtc {\r\n        display: ruby-text-container;\r\n    }\r\n\r\n    ruby, rb, rt, rbc, rtc {\r\n        unicode-bidi: isolate;\r\n    }\r\n\r\n\r\n    :link {\r\n        color: #0000EE;\r\n    }\r\n\r\n    :visited {\r\n        color: #551A8B;\r\n    }\r\n\r\n    :link, :visited {\r\n        text-decoration: underline;\r\n    }\r\n\r\n    a:link[rel~=help], a:visited[rel~=help],\r\n    area:link[rel~=help], area:visited[rel~=help] {\r\n        cursor: help;\r\n    }\r\n\r\n    :focus {\r\n        outline: auto;\r\n    }\r\n\r\n    mark {\r\n        background: yellow;\r\n        color: black;\r\n    }\r\n    /* this color is just a suggestion and can be changed based on implementation feedback */\r\n    abbr[title], acronym[title] {\r\n        text-decoration: dotted underline;\r\n    }\r\n\r\n    ins, u {\r\n        text-decoration: underline;\r\n    }\r\n\r\n    del, s, strike {\r\n        text-decoration: line-through;\r\n    }\r\n\r\n    blink {\r\n        text-decoration: blink;\r\n    }\r\n\r\n    q::before {\r\n        content: open-quote;\r\n    }\r\n\r\n    q::after {\r\n        content: close-quote;\r\n    }\r\n\r\n    br {\r\n        content: '\\A';\r\n        white-space: pre;\r\n    }\r\n    /* this also has bidi implications */\r\n    nobr {\r\n        white-space: nowrap;\r\n    }\r\n\r\n    wbr {\r\n        content: '\\200B';\r\n    }\r\n    /* this also has bidi implications */\r\n    nobr wbr {\r\n        white-space: normal;\r\n    }\r\n\r\n    article, aside, h1, h2, h3, h4, h5, h6, hgroup, nav, section {\r\n        display: block;\r\n    }\r\n\r\n    h1 {\r\n        margin-top: 0.67em;\r\n        margin-bottom: 0.67em;\r\n        font-size: 2.00em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h2 {\r\n        margin-top: 0.83em;\r\n        margin-bottom: 0.83em;\r\n        font-size: 1.50em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h3 {\r\n        margin-top: 1.00em;\r\n        margin-bottom: 1.00em;\r\n        font-size: 1.17em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h4 {\r\n        margin-top: 1.33em;\r\n        margin-bottom: 1.33em;\r\n        font-size: 1.00em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h5 {\r\n        margin-top: 1.67em;\r\n        margin-bottom: 1.67em;\r\n        font-size: 0.83em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    h6 {\r\n        margin-top: 2.33em;\r\n        margin-bottom: 2.33em;\r\n        font-size: 0.67em;\r\n        font-weight: bold;\r\n    }\r\n\r\n    dir, dd, dl, dt, ol, ul {\r\n        display: block;\r\n    }\r\n\r\n    li {\r\n        display: list-item;\r\n    }\r\n\r\n    dir, dl, ol, ul {\r\n        margin-top: 1em;\r\n        margin-bottom: 1em;\r\n    }\r\n\r\n        dir dir, dir dl, dir ol, dir ul,\r\n        dl dir, dl dl, dl ol, dl ul,\r\n        ol dir, ol dl, ol ol, ol ul,\r\n        ul dir, ul dl, ul ol, ul ul {\r\n            margin-top: 0;\r\n            margin-bottom: 0;\r\n        }\r\n\r\n    dd {\r\n        margin-left: 40px;\r\n    }\r\n    /* LTR-specific: use 'margin-right' for rtl elements */\r\n    dir, ol, ul {\r\n        padding-left: 40px;\r\n    }\r\n    /* LTR-specific: use 'padding-right' for rtl elements */\r\n    ol {\r\n        list-style-type: decimal;\r\n    }\r\n\r\n    dir, ul {\r\n        list-style-type: disc;\r\n    }\r\n\r\n        dir dir, dir ul,\r\n        ol dir, ol ul,\r\n        ul dir, ul ul {\r\n            list-style-type: circle;\r\n        }\r\n\r\n            dir dir dir, dir dir ul,\r\n            dir ol dir, dir ol ul,\r\n            dir ul dir, dir ul ul,\r\n            ol dir dir, ol dir ul,\r\n            ol ol dir, ol ol ul,\r\n            ol ul dir, ol ul ul,\r\n            ul dir dir, ul dir ul,\r\n            ul ol dir, ul ol ul,\r\n            ul ul dir, ul ul ul {\r\n                list-style-type: square;\r\n            }\r\n\r\n    table {\r\n        display: table;\r\n    }\r\n\r\n    caption {\r\n        display: table-caption;\r\n    }\r\n\r\n    colgroup, colgroup[hidden] {\r\n        display: table-column-group;\r\n    }\r\n\r\n    col, col[hidden] {\r\n        display: table-column;\r\n    }\r\n\r\n    thead, thead[hidden] {\r\n        display: table-header-group;\r\n    }\r\n\r\n    tbody, tbody[hidden] {\r\n        display: table-row-group;\r\n    }\r\n\r\n    tfoot, tfoot[hidden] {\r\n        display: table-footer-group;\r\n    }\r\n\r\n    tr, tr[hidden] {\r\n        display: table-row;\r\n    }\r\n\r\n    td, th, td[hidden], th[hidden] {\r\n        display: table-cell;\r\n    }\r\n\r\n        colgroup[hidden], col[hidden], thead[hidden], tbody[hidden],\r\n        tfoot[hidden], tr[hidden], td[hidden], th[hidden] {\r\n            visibility: collapse;\r\n        }\r\n\r\n    table {\r\n        box-sizing: border-box;\r\n        border-spacing: 2px;\r\n        border-collapse: separate;\r\n        text-indent: initial;\r\n    }\r\n\r\n    td, th {\r\n        padding: 1px;\r\n    }\r\n\r\n    th {\r\n        font-weight: bold;\r\n    }\r\n\r\n    thead, tbody, tfoot, table > tr {\r\n        vertical-align: middle;\r\n    }\r\n\r\n    tr, td, th {\r\n        vertical-align: inherit;\r\n    }\r\n\r\n    table, td, th {\r\n        border-color: gray;\r\n    }\r\n\r\n    thead, tbody, tfoot, tr {\r\n        border-color: inherit;\r\n    }\r\n    /*table[rules=none i], table[rules=groups i], table[rules=rows i],\r\ntable[rules=cols i], table[rules=all i], table[frame=void i],\r\ntable[frame=above i], table[frame=below i], table[frame=hsides i],\r\ntable[frame=lhs i], table[frame=rhs i], table[frame=vsides i],\r\ntable[frame=box i], table[frame=border i],\r\ntable[rules=none i] > tr > td, table[rules=none i] > tr > th,\r\ntable[rules=groups i] > tr > td, table[rules=groups i] > tr > th,\r\ntable[rules=rows i] > tr > td, table[rules=rows i] > tr > th,\r\ntable[rules=cols i] > tr > td, table[rules=cols i] > tr > th,\r\ntable[rules=all i] > tr > td, table[rules=all i] > tr > th,\r\ntable[rules=none i] > thead > tr > td, table[rules=none i] > thead > tr > th,\r\ntable[rules=groups i] > thead > tr > td, table[rules=groups i] > thead > tr > th,\r\ntable[rules=rows i] > thead > tr > td, table[rules=rows i] > thead > tr > th,\r\ntable[rules=cols i] > thead > tr > td, table[rules=cols i] > thead > tr > th,\r\ntable[rules=all i] > thead > tr > td, table[rules=all i] > thead > tr > th,\r\ntable[rules=none i] > tbody > tr > td, table[rules=none i] > tbody > tr > th,\r\ntable[rules=groups i] > tbody > tr > td, table[rules=groups i] > tbody > tr > th,\r\ntable[rules=rows i] > tbody > tr > td, table[rules=rows i] > tbody > tr > th,\r\ntable[rules=cols i] > tbody > tr > td, table[rules=cols i] > tbody > tr > th,\r\ntable[rules=all i] > tbody > tr > td, table[rules=all i] > tbody > tr > th,\r\ntable[rules=none i] > tfoot > tr > td, table[rules=none i] > tfoot > tr > th,\r\ntable[rules=groups i] > tfoot > tr > td, table[rules=groups i] > tfoot > tr > th,\r\ntable[rules=rows i] > tfoot > tr > td, table[rules=rows i] > tfoot > tr > th,\r\ntable[rules=cols i] > tfoot > tr > td, table[rules=cols i] > tfoot > tr > th,\r\ntable[rules=all i] > tfoot > tr > td, table[rules=all i] > tfoot > tr > th {\r\n    border-color: black;\r\n}*/\r\n    input, select, option, optgroup, button, textarea, keygen {\r\n        text-indent: initial;\r\n    }\r\n\r\n    textarea {\r\n        white-space: pre-wrap;\r\n    }\r\n\r\n    input[type=\"radio\"], input[type=\"checkbox\"], input[type=\"reset\"], input[type=\"button\"],\r\n    input[type=\"submit\"], select, button {\r\n        box-sizing: border-box;\r\n    }\r\n\r\n    input[type=\"button\"], button {\r\n        padding: 0.3em 0.5em;\r\n    }\r\n\r\n    hr {\r\n        color: gray;\r\n        border-style: inset;\r\n        border-width: 1px;\r\n        margin: 0.5em auto;\r\n    }\r\n\r\n    fieldset {\r\n        margin-left: 2px;\r\n        margin-right: 2px;\r\n        border: groove 2px ThreeDFace;\r\n        padding: 0.35em 0.625em 0.75em;\r\n    }\r\n\r\n    legend {\r\n        padding-left: 2px;\r\n        padding-right: 2px;\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    > .layout-toolbox-wrapper {\r\n        position: relative;\r\n        margin-left: @container-padding;\r\n        width: 220px;\r\n        flex-shrink: 0;\r\n\r\n        > .layout-toolbox {\r\n            border: 1px solid @gray-border;\r\n            width: 220px;\r\n            min-height: 400px;\r\n            padding: @container-padding / 2;\r\n            background-color: @gray-bg;\r\n\r\n            &.sticky-top {\r\n                position: fixed;\r\n                top: 0;\r\n                max-height: 100%;\r\n                overflow-y: auto;\r\n            }\r\n\r\n            &.sticky-bottom {\r\n                position: absolute;\r\n                bottom: 0;\r\n            }\r\n\r\n            .layout-toolbox-group {\r\n                margin-top: @container-padding;\r\n\r\n                .layout-toolbox-group-heading {\r\n                    display: block;\r\n                    margin-bottom: @container-padding / 3;\r\n                    text-decoration: none;\r\n                \r\n                    &:before {\r\n                        display: inline-block;\r\n                        width: 10px;\r\n                        margin-right: @container-padding / 3;\r\n                        font: normal normal normal 14px/1 FontAwesome;\r\n                        text-align: center;\r\n                        content: \"\\f0d7\";\r\n                    }\r\n                }\r\n            \r\n                &.collapsed {\r\n                    .layout-toolbox-group-heading:before {\r\n                        content: \"\\f0da\";\r\n                    }\r\n\r\n                    .layout-toolbox-items {\r\n                        display: none;\r\n                    }\r\n                }\r\n\r\n                .layout-toolbox-section + .layout-toolbox-section {\r\n                    margin-top: @container-padding / 3;\r\n                }\r\n\r\n                .layout-toolbox-item {\r\n                    border: 1px solid @gray-border;\r\n                    background-color: #fff;\r\n                    padding: (@container-padding - 3) @container-padding;\r\n                    cursor: default;\r\n\r\n                    i {\r\n                        display: inline-block;\r\n                        width: 16px;\r\n                        font: normal normal normal 14px/1 FontAwesome;\r\n                    }\r\n\r\n                    + .layout-toolbox-item {\r\n                        margin-top: @container-padding / 3;\r\n                    }\r\n                }\r\n\r\n                + .layout-toolbox-group {\r\n                    margin-top: @container-padding / 2;\r\n                }\r\n            }\r\n        }\r\n    }\r\n}\r\n","@import \"Variables.less\";\r\n\r\n.layout-editor {\r\n    .layout-popup {\r\n        display: none; // Shown only in active or focused states.\r\n        position: absolute;\r\n        margin: 0;\r\n        -moz-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\r\n        -webkit-box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\r\n        box-shadow: 3px 3px 11px 0 rgba(50, 50, 50, 0.5);\r\n        border: 1px solid @gray-border;\r\n        padding: 2px 0;\r\n        background-color: #f7f7f7;\r\n        list-style: none;\r\n        z-index: 20;\r\n        color: @gray-text;\r\n        text-align: left;\r\n        cursor: default;\r\n        white-space: nowrap;\r\n        line-height: normal;\r\n        min-width: 300px;\r\n\r\n        &.wide {\r\n            width: 600px;\r\n        }\r\n\r\n        .layout-popup-flex {\r\n            display: flex;\r\n            padding: 2px 5px;\r\n\r\n            .layout-popup-column + .layout-popup-column {\r\n                margin-left: 4px;\r\n                border-left: 1px solid fade(gray, 15%);\r\n                padding-left: 4px;\r\n            }\r\n        }\r\n\r\n        .layout-popup-item {\r\n            padding: 4px 6px;\r\n        }\r\n\r\n        .layout-popup-label {\r\n            font-size: 0.9em;\r\n            font-weight: bold;\r\n            color: @gray-text;\r\n        }\r\n\r\n        .layout-popup-action:hover {\r\n            background-color: @gray-bg;\r\n            cursor: pointer;\r\n        }\r\n\r\n        .layout-popup-input {\r\n            input[type='text'], textarea {\r\n                width: 100%;\r\n            }\r\n\r\n            > label {\r\n                display: block;\r\n                font-size: 0.9em;\r\n            }\r\n        }\r\n    }\r\n}\r\n"],"sourceRoot":"/source/"} */ \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Views/EditorTemplates/LayoutEditor.cshtml b/src/Orchard.Web/Modules/Orchard.Layouts/Views/EditorTemplates/LayoutEditor.cshtml index 0d928d4f7..816ebed41 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Views/EditorTemplates/LayoutEditor.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Views/EditorTemplates/LayoutEditor.cshtml @@ -17,7 +17,6 @@ Script.Require("jQueryUI_Resizable"); Script.Require("jQueryUI_Position"); Script.Require("jQueryUI_Dialog"); - Script.Require("TinyMce"); Script.Require("Layouts.LayoutEditor"); Script.Include("jquery.deserialize.js"); diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Views/Elements/ContentItem.Design.cshtml b/src/Orchard.Web/Modules/Orchard.Layouts/Views/Elements/ContentItem.Design.cshtml index c732cef6b..d84b1abe7 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Views/Elements/ContentItem.Design.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Views/Elements/ContentItem.Design.cshtml @@ -2,7 +2,7 @@ @foreach (var contentItemShape in Model.ContentItems) { var contentItem = (ContentItem)contentItemShape.ContentItem; var displayTextHtmlString = Html.ItemDisplayText(contentItem); - var displayText = displayTextHtmlString != null ? displayTextHtmlString.ToString() : T("-").ToString(); + var displayText = displayTextHtmlString != null ? (IHtmlString)displayTextHtmlString : T("-");
@displayText
diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Views/LayoutEditor.Template.Toolbox.cshtml b/src/Orchard.Web/Modules/Orchard.Layouts/Views/LayoutEditor.Template.Toolbox.cshtml index 7ccbcd2cd..05374bd38 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Views/LayoutEditor.Template.Toolbox.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Views/LayoutEditor.Template.Toolbox.cshtml @@ -1,10 +1,6 @@ 
- -
    +
    • @T("Layout")
        diff --git a/src/Orchard.Web/Modules/Orchard.Lists/Module.txt b/src/Orchard.Web/Modules/Orchard.Lists/Module.txt index 010fa8caa..be8fc36a6 100644 --- a/src/Orchard.Web/Modules/Orchard.Lists/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Lists/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Introduces a preconfigured container-enabled content type. FeatureDescription: A basic container-enabled content type. diff --git a/src/Orchard.Web/Modules/Orchard.Lists/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Lists/Properties/AssemblyInfo.cs index e6df16ec3..c729a5b99 100644 --- a/src/Orchard.Web/Modules/Orchard.Lists/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Lists/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Lists/Views/Parts.Container.Manage.cshtml b/src/Orchard.Web/Modules/Orchard.Lists/Views/Parts.Container.Manage.cshtml index 10a69f805..636dff05e 100644 --- a/src/Orchard.Web/Modules/Orchard.Lists/Views/Parts.Container.Manage.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Lists/Views/Parts.Container.Manage.cshtml @@ -5,17 +5,17 @@ }

        - @Html.ActionLink(T("{0} Properties", (string)Model.ContainerDisplayName).ToString(), "Edit", new { Area = "Contents", Id = (int)Model.ContainerId, ReturnUrl = Html.ViewContext.HttpContext.Request.RawUrl }) + @Html.ActionLink(T("{0} Properties", (string)Model.ContainerDisplayName), "Edit", new { Area = "Contents", Id = (int)Model.ContainerId, ReturnUrl = Html.ViewContext.HttpContext.Request.RawUrl })

        @if (itemContentTypes.Any()) { foreach (var contentType in itemContentTypes) { - @Html.ActionLink(T("New {0}", contentType.DisplayName).ToString(), "Create", "Admin", new { area = "Contents", id = contentType.Name, containerId }, new { @class = "button primaryAction create-content" }) + @Html.ActionLink(T("New {0}", contentType.DisplayName), "Create", "Admin", new { area = "Contents", id = contentType.Name, containerId }, new { @class = "button primaryAction create-content" }) } } else { - @Html.ActionLink(T("New Content").ToString(), "Create", "Admin", new { area = "Contents", containerId }, new { @class = "button primaryAction create-content" }) + @Html.ActionLink(T("New Content"), "Create", "Admin", new { area = "Contents", containerId }, new { @class = "button primaryAction create-content" }) } @T("Choose Items")
        diff --git a/src/Orchard.Web/Modules/Orchard.Localization/Controllers/AdminController.cs b/src/Orchard.Web/Modules/Orchard.Localization/Controllers/AdminController.cs index b0e7726dd..597c39439 100644 --- a/src/Orchard.Web/Modules/Orchard.Localization/Controllers/AdminController.cs +++ b/src/Orchard.Web/Modules/Orchard.Localization/Controllers/AdminController.cs @@ -53,10 +53,7 @@ namespace Orchard.Localization.Controllers { var contentItemTranslation = _contentManager.New(masterContentItem.ContentType); contentItemTranslation.MasterContentItem = masterContentItem; - // build the editor using the master content item so that - // the form is pre-populated with the original values - - var content = _contentManager.BuildEditor(masterContentItem); + var content = _contentManager.BuildEditor(contentItemTranslation); return View(content); } diff --git a/src/Orchard.Web/Modules/Orchard.Localization/Drivers/LocalizationPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Localization/Drivers/LocalizationPartDriver.cs index b6854a51d..2f59e3172 100644 --- a/src/Orchard.Web/Modules/Orchard.Localization/Drivers/LocalizationPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Localization/Drivers/LocalizationPartDriver.cs @@ -108,16 +108,19 @@ namespace Orchard.Localization.Drivers { } protected override void Importing(LocalizationPart part, ContentManagement.Handlers.ImportContentContext context) { - var masterContentItem = context.Attribute(part.PartDefinition.Name, "MasterContentItem"); - if (masterContentItem != null) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + + context.ImportAttribute(part.PartDefinition.Name, "MasterContentItem", masterContentItem => { var contentItem = context.GetItemFromSession(masterContentItem); if (contentItem != null) { part.MasterContentItem = contentItem; } - } + }); - var culture = context.Attribute(part.PartDefinition.Name, "Culture"); - if (culture != null) { + context.ImportAttribute(part.PartDefinition.Name, "Culture", culture => { var targetCulture = _cultureManager.GetCultureByName(culture); // Add Culture. if (targetCulture == null && _cultureManager.IsValidCulture(culture)) { @@ -125,7 +128,7 @@ namespace Orchard.Localization.Drivers { targetCulture = _cultureManager.GetCultureByName(culture); } part.Culture = targetCulture; - } + }); } protected override void Exporting(LocalizationPart part, ContentManagement.Handlers.ExportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Localization/Module.txt b/src/Orchard.Web/Modules/Orchard.Localization/Module.txt index 7e20add53..b019b5be1 100644 --- a/src/Orchard.Web/Modules/Orchard.Localization/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Localization/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The localization module enables the localization of content items. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Localization/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Localization/Properties/AssemblyInfo.cs index 08fde37af..cd3959a32 100644 --- a/src/Orchard.Web/Modules/Orchard.Localization/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Localization/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Media/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Media/Properties/AssemblyInfo.cs index 8d33e7cfb..b2b27b311 100644 --- a/src/Orchard.Web/Modules/Orchard.Media/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Media/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/AdminMenu.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/AdminMenu.cs index d83aff5e5..82c32328b 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/AdminMenu.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/AdminMenu.cs @@ -15,7 +15,7 @@ namespace Orchard.MediaLibrary { builder.AddImageSet("media-library") .Add(T("Media"), "6", menu => menu.Add(T("Media"), "0", item => item.Action("Index", "Admin", new { area = "Orchard.MediaLibrary" }) - .Permission(Permissions.ManageMediaContent))); + .Permission(Permissions.ManageOwnMedia))); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/AdminController.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/AdminController.cs index bd9ab44fa..c9158d4f7 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/AdminController.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/AdminController.cs @@ -14,6 +14,7 @@ using Orchard.Themes; using Orchard.UI.Navigation; using Orchard.ContentManagement.MetaData; using Orchard.Validation; +using System.Collections.Generic; namespace Orchard.MediaLibrary.Controllers { [ValidateInput(false)] @@ -41,17 +42,21 @@ namespace Orchard.MediaLibrary.Controllers { public ILogger Logger { get; set; } public ActionResult Index(string folderPath = "", bool dialog = false) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Cannot view media"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Cannot view media"))) return new HttpUnauthorizedResult(); + // If the user is trying to access a folder above his boundaries, redirect him to his home folder + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { + return RedirectToAction("Index", new { folderPath = rootMediaFolder.MediaPath, dialog }); + } + // let other modules enhance the ui by providing custom navigation and actions var explorer = Services.ContentManager.New("MediaLibraryExplorer"); explorer.Weld(new MediaLibraryExplorerPart()); var explorerShape = Services.ContentManager.BuildDisplay(explorer); - var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); - var viewModel = new MediaManagerIndexViewModel { DialogMode = dialog, FolderPath = folderPath, @@ -73,7 +78,7 @@ namespace Orchard.MediaLibrary.Controllers { } public ActionResult Import(string folderPath) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Cannot import media"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Cannot import media"))) return new HttpUnauthorizedResult(); var mediaProviderMenu = _navigationManager.BuildMenu("mediaproviders"); @@ -91,9 +96,20 @@ namespace Orchard.MediaLibrary.Controllers { [Themed(false)] public ActionResult MediaItems(string folderPath, int skip = 0, int count = 0, string order = "created", string mediaType = "") { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Cannot view media"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Cannot view media"))) return new HttpUnauthorizedResult(); + // Check permission.var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { + var model = new MediaManagerMediaItemsViewModel { + MediaItems = new List(), + MediaItemsCount = 0, + FolderPath = folderPath + }; + + return View(model); + } + var mediaParts = _mediaLibraryService.GetMediaContentItems(folderPath, skip, count, order, mediaType, VersionOptions.Latest); var mediaPartsCount = _mediaLibraryService.GetMediaContentItemsCount(folderPath, mediaType, VersionOptions.Latest); @@ -113,9 +129,19 @@ namespace Orchard.MediaLibrary.Controllers { [Themed(false)] public ActionResult ChildFolders(string folderPath = null) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Cannot get child folder listing"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Cannot get child folder listing"))) return new HttpUnauthorizedResult(); + // Check permission. + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { + var model = new MediaManagerChildFoldersViewModel { + Children = new IMediaFolder[0] + }; + + return View(model); + } + var viewModel = new MediaManagerChildFoldersViewModel { Children = _mediaLibraryService.GetMediaFolders(folderPath) }; @@ -127,11 +153,13 @@ namespace Orchard.MediaLibrary.Controllers { [Themed(false)] public ActionResult RecentMediaItems(int skip = 0, int count = 0, string order = "created", string mediaType = "") { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Cannot view media"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Cannot view media"))) return new HttpUnauthorizedResult(); - var mediaParts = _mediaLibraryService.GetMediaContentItems(skip, count, order, mediaType); - var mediaPartsCount = _mediaLibraryService.GetMediaContentItemsCount(mediaType); + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder().MediaPath; + + var mediaParts = _mediaLibraryService.GetMediaContentItems(rootMediaFolder, skip, count, order, mediaType); + var mediaPartsCount = _mediaLibraryService.GetMediaContentItemsCount(rootMediaFolder, mediaType); var mediaItems = mediaParts.Select(x => new MediaManagerMediaItemViewModel { @@ -149,12 +177,13 @@ namespace Orchard.MediaLibrary.Controllers { [Themed(false)] public ActionResult MediaItem(int id, string displayType = "SummaryAdmin") { - var contentItem = Services.ContentManager.Get(id, VersionOptions.Latest); + var contentItem = Services.ContentManager.Get(id, VersionOptions.Latest); if (contentItem == null) return HttpNotFound(); - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, contentItem, T("Cannot view media"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, contentItem, T("Cannot view media")) + || !_mediaLibraryService.CanManageMediaFolder(contentItem.FolderPath)) return new HttpUnauthorizedResult(); dynamic model = Services.ContentManager.BuildDisplay(contentItem, displayType); @@ -164,13 +193,20 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost] public ActionResult Delete(int[] mediaItemIds) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't delete media items"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't delete media items"))) return new HttpUnauthorizedResult(); + var mediaItems = Services.ContentManager + .Query(VersionOptions.Latest) + .ForContentItems(mediaItemIds) + .List() + .Select(x => x.As()) + .Where(x => x != null); + try { - foreach (var media in Services.ContentManager.Query(VersionOptions.Latest).ForContentItems(mediaItemIds).List()) { - if (media != null) { - Services.ContentManager.Remove(media); + foreach (var media in mediaItems) { + if (_mediaLibraryService.CanManageMediaFolder(media.FolderPath)) { + Services.ContentManager.Remove(media.ContentItem); } } @@ -184,12 +220,16 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost] public ActionResult Clone(int mediaItemId) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't clone media items"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't clone media items"))) return new HttpUnauthorizedResult(); try { var media = Services.ContentManager.Get(mediaItemId).As(); + if(!_mediaLibraryService.CanManageMediaFolder(media.FolderPath)) { + return new HttpUnauthorizedResult(); + } + var newFileName = Path.GetFileNameWithoutExtension(media.FileName) + " Copy" + Path.GetExtension(media.FileName); _mediaLibraryService.CopyFile(media.FolderPath, media.FileName, media.FolderPath, newFileName); diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/ClientStorageController.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/ClientStorageController.cs index 607dd1564..aedba54df 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/ClientStorageController.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/ClientStorageController.cs @@ -17,10 +17,13 @@ namespace Orchard.MediaLibrary.Controllers { public class ClientStorageController : Controller { private readonly IMediaLibraryService _mediaLibraryService; - public ClientStorageController(IMediaLibraryService mediaManagerService, IOrchardServices orchardServices) { + public ClientStorageController( + IMediaLibraryService mediaManagerService, + IContentManager contentManager, + IOrchardServices orchardServices) { _mediaLibraryService = mediaManagerService; Services = orchardServices; - + Services = orchardServices; T = NullLocalizer.Instance; } @@ -42,7 +45,13 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost] public ActionResult Upload(string folderPath, string type) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Cannot manage media"))) { + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia)) { + return new HttpUnauthorizedResult(); + } + + // Check permission. + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { return new HttpUnauthorizedResult(); } diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/FolderController.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/FolderController.cs index b08662cf8..aae4fc56f 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/FolderController.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/FolderController.cs @@ -32,9 +32,15 @@ namespace Orchard.MediaLibrary.Controllers { public Localizer T { get; set; } public ActionResult Create(string folderPath) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't create media folder"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't create media folder"))) return new HttpUnauthorizedResult(); + // If the user is trying to access a folder above his boundaries, redirect him to his home folder + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { + return RedirectToAction("Create", new { folderPath = rootMediaFolder.MediaPath }); + } + var viewModel = new MediaManagerFolderCreateViewModel { Hierarchy = _mediaLibraryService.GetMediaFolders(folderPath), FolderPath = folderPath @@ -45,12 +51,16 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost, ActionName("Create")] public ActionResult Create() { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't create media folder"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't create media folder"))) return new HttpUnauthorizedResult(); var viewModel = new MediaManagerFolderCreateViewModel(); UpdateModel(viewModel); + if (!_mediaLibraryService.CanManageMediaFolder(viewModel.FolderPath)) { + return new HttpUnauthorizedResult(); + } + try { _mediaLibraryService.CreateFolder(viewModel.FolderPath, viewModel.Name); Services.Notifier.Information(T("Media folder created")); @@ -66,9 +76,13 @@ namespace Orchard.MediaLibrary.Controllers { } public ActionResult Edit(string folderPath) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't edit media folder"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't edit media folder"))) return new HttpUnauthorizedResult(); - + + if (!_mediaLibraryService.CanManageMediaFolder(folderPath)) { + return new HttpUnauthorizedResult(); + } + var viewModel = new MediaManagerFolderEditViewModel { FolderPath = folderPath, Name = folderPath.Split(Path.DirectorySeparatorChar, Path.AltDirectorySeparatorChar).Last() @@ -80,12 +94,16 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost, ActionName("Edit")] [FormValueRequired("submit.Save")] public ActionResult Edit() { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't edit media folder"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't edit media folder"))) return new HttpUnauthorizedResult(); var viewModel = new MediaManagerFolderEditViewModel(); UpdateModel(viewModel); + if (!_mediaLibraryService.CanManageMediaFolder(viewModel.FolderPath)) { + return new HttpUnauthorizedResult(); + } + try { _mediaLibraryService.RenameFolder(viewModel.FolderPath, viewModel.Name); Services.Notifier.Information(T("Media folder renamed")); @@ -101,12 +119,16 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost, ActionName("Edit")] [FormValueRequired("submit.Delete")] public ActionResult Delete() { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't delete media folder"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't delete media folder"))) return new HttpUnauthorizedResult(); var viewModel = new MediaManagerFolderEditViewModel(); UpdateModel(viewModel); + if (!_mediaLibraryService.CanManageMediaFolder(viewModel.FolderPath)) { + return new HttpUnauthorizedResult(); + + } try { _mediaLibraryService.DeleteFolder(viewModel.FolderPath); Services.Notifier.Information(T("Media folder deleted")); @@ -122,9 +144,13 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost] public ActionResult Move(string folderPath, int[] mediaItemIds) { - if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent, T("Couldn't move media items"))) + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia, T("Couldn't move media items"))) return new HttpUnauthorizedResult(); + if (!_mediaLibraryService.CanManageMediaFolder(folderPath)) { + return new HttpUnauthorizedResult(); + } + foreach (var media in Services.ContentManager.Query().ForPart().ForContentItems(mediaItemIds).List()) { // don't try to rename the file if there is no associated media file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/OEmbedController.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/OEmbedController.cs index 0ed803cbe..e5368bd0a 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/OEmbedController.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/OEmbedController.cs @@ -9,12 +9,18 @@ using Orchard.MediaLibrary.ViewModels; using Orchard.Themes; using Orchard.UI.Admin; using Orchard.ContentManagement; +using Orchard.MediaLibrary.Services; namespace Orchard.MediaLibrary.Controllers { [Admin, Themed(false)] public class OEmbedController : Controller { - public OEmbedController(IOrchardServices services) { + private readonly IMediaLibraryService _mediaLibraryService; + + public OEmbedController( + IOrchardServices services, + IMediaLibraryService mediaManagerService) { Services = services; + _mediaLibraryService = mediaManagerService; } public IOrchardServices Services { get; set; } @@ -32,6 +38,15 @@ namespace Orchard.MediaLibrary.Controllers { [ActionName("Index")] [ValidateInput(false)] public ActionResult IndexPOST(string folderPath, string url, string type, string title, string html, string thumbnail, string width, string height, string description) { + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia)) + return new HttpUnauthorizedResult(); + + // Check permission. + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { + return new HttpUnauthorizedResult(); + } + var viewModel = new OEmbedViewModel { Url = url, FolderPath = folderPath diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/WebSearchController.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/WebSearchController.cs index 83154cf9c..23f7b576d 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/WebSearchController.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Controllers/WebSearchController.cs @@ -13,12 +13,19 @@ namespace Orchard.MediaLibrary.Controllers { public class WebSearchController : Controller { private readonly IMediaLibraryService _mediaLibraryService; private readonly IContentManager _contentManager; - - public WebSearchController(IMediaLibraryService mediaManagerService, IContentManager contentManager) { + + public WebSearchController( + IMediaLibraryService mediaManagerService, + IContentManager contentManager, + IOrchardServices orchardServices) { _mediaLibraryService = mediaManagerService; _contentManager = contentManager; + + Services = orchardServices; } + public IOrchardServices Services { get; set; } + public ActionResult Index(string folderPath, string type) { var viewModel = new ImportMediaViewModel { FolderPath = folderPath, @@ -30,7 +37,15 @@ namespace Orchard.MediaLibrary.Controllers { [HttpPost] - public JsonResult ImagePost(string folderPath, string type, string url) { + public ActionResult ImagePost(string folderPath, string type, string url) { + if (!Services.Authorizer.Authorize(Permissions.ManageOwnMedia)) + return new HttpUnauthorizedResult(); + + // Check permission. + var rootMediaFolder = _mediaLibraryService.GetRootMediaFolder(); + if (!Services.Authorizer.Authorize(Permissions.ManageMediaContent) && !_mediaLibraryService.CanManageMediaFolder(folderPath)) { + return new HttpUnauthorizedResult(); + } try { var buffer = new WebClient().DownloadData(url); diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/AudioPartDriver.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/AudioPartDriver.cs index 14f17ce73..32fdb65e7 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/AudioPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/AudioPartDriver.cs @@ -17,10 +17,14 @@ namespace Orchard.MediaLibrary.Drivers { } protected override void Importing(AudioPart part, ContentManagement.Handlers.ImportContentContext context) { - var length = context.Attribute(part.PartDefinition.Name, "Length"); - if (length != null) { - part.Length = int.Parse(length); + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } + + context.ImportAttribute(part.PartDefinition.Name, "Length", length => + part.Length = int.Parse(length) + ); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/DocumentPartDriver.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/DocumentPartDriver.cs index 2a0ac0d06..28d721d4b 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/DocumentPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/DocumentPartDriver.cs @@ -17,10 +17,14 @@ namespace Orchard.MediaLibrary.Drivers { } protected override void Importing(DocumentPart part, ContentManagement.Handlers.ImportContentContext context) { - var length = context.Attribute(part.PartDefinition.Name, "Length"); - if (length != null) { - part.Length = int.Parse(length); + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } + + context.ImportAttribute(part.PartDefinition.Name, "Length", length => + part.Length = int.Parse(length) + ); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/ImagePartDriver.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/ImagePartDriver.cs index 77163dbf4..5c12f4e34 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/ImagePartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/ImagePartDriver.cs @@ -20,15 +20,18 @@ namespace Orchard.MediaLibrary.Drivers { } protected override void Importing(ImagePart part, ContentManagement.Handlers.ImportContentContext context) { - var height = context.Attribute(part.PartDefinition.Name, "Height"); - if (height != null) { - part.Height = int.Parse(height); + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - var width = context.Attribute(part.PartDefinition.Name, "Width"); - if (width != null) { - part.Width = int.Parse(width); - } + context.ImportAttribute(part.PartDefinition.Name, "Height", height => + part.Height = int.Parse(height) + ); + + context.ImportAttribute(part.PartDefinition.Name, "Width", width => + part.Width = int.Parse(width) + ); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/MediaPartDriver.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/MediaPartDriver.cs index 65df8c8d4..7ee9d8bf9 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/MediaPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/MediaPartDriver.cs @@ -36,30 +36,30 @@ namespace Orchard.MediaLibrary.Drivers { } protected override void Importing(MediaPart part, ContentManagement.Handlers.ImportContentContext context) { - var mimeType = context.Attribute(part.PartDefinition.Name, "MimeType"); - if (mimeType != null) { - part.MimeType = mimeType; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - var caption = context.Attribute(part.PartDefinition.Name, "Caption"); - if (caption != null) { - part.Caption = caption; - } + context.ImportAttribute(part.PartDefinition.Name, "MimeType", mimeType => + part.MimeType = mimeType + ); - var alternateText = context.Attribute(part.PartDefinition.Name, "AlternateText"); - if (alternateText != null) { - part.AlternateText = alternateText; - } + context.ImportAttribute(part.PartDefinition.Name, "Caption", caption => + part.Caption = caption + ); - var folderPath = context.Attribute(part.PartDefinition.Name, "FolderPath"); - if (folderPath != null) { - part.FolderPath = folderPath; - } + context.ImportAttribute(part.PartDefinition.Name, "AlternateText", alternateText => + part.AlternateText = alternateText + ); - var fileName = context.Attribute(part.PartDefinition.Name, "FileName"); - if (fileName != null) { - part.FileName = fileName; - } + context.ImportAttribute(part.PartDefinition.Name, "FolderPath", folderPath => + part.FolderPath = folderPath + ); + + context.ImportAttribute(part.PartDefinition.Name, "FileName", fileName => + part.FileName = fileName + ); } protected override void Exporting(MediaPart part, ContentManagement.Handlers.ExportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/OEmbedPartDriver.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/OEmbedPartDriver.cs index 1ac59f54b..3f0fd1adf 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/OEmbedPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/OEmbedPartDriver.cs @@ -17,10 +17,14 @@ namespace Orchard.MediaLibrary.Drivers { } protected override void Importing(OEmbedPart part, ContentManagement.Handlers.ImportContentContext context) { - var source = context.Attribute(part.PartDefinition.Name, "Source"); - if (source != null) { - part.Source = source; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } + + context.ImportAttribute(part.PartDefinition.Name, "Source", source => + part.Source = source + ); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/VideoPartDriver.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/VideoPartDriver.cs index 57ef97145..a9fdce3bb 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/VideoPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Drivers/VideoPartDriver.cs @@ -17,10 +17,14 @@ namespace Orchard.MediaLibrary.Drivers { } protected override void Importing(VideoPart part, ContentManagement.Handlers.ImportContentContext context) { - var length = context.Attribute(part.PartDefinition.Name, "Length"); - if (length != null) { - part.Length = int.Parse(length); + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } + + context.ImportAttribute(part.PartDefinition.Name, "Length", length => + part.Length = int.Parse(length) + ); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Module.txt b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Module.txt index b43640f3c..700b9f1a5 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides enhanced Media management tools. Features: diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Orchard.MediaLibrary.csproj b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Orchard.MediaLibrary.csproj index 5cc81bb4b..08b413e12 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Orchard.MediaLibrary.csproj +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Orchard.MediaLibrary.csproj @@ -178,6 +178,7 @@ + diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Permissions.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Permissions.cs index 3efd53f8e..8e9497683 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Permissions.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Permissions.cs @@ -4,13 +4,15 @@ using Orchard.Security.Permissions; namespace Orchard.MediaLibrary { public class Permissions : IPermissionProvider { - public static readonly Permission ManageMediaContent = new Permission { Description = "Managing Media", Name = "ManageMediaContent" }; + public static readonly Permission ManageMediaContent = new Permission { Description = "Manage Media", Name = "ManageMediaContent" }; + public static readonly Permission ManageOwnMedia = new Permission { Description = "Manage Own Media", Name = "ManageOwnMedia", ImpliedBy = new[] { ManageMediaContent } }; public virtual Feature Feature { get; set; } public IEnumerable GetPermissions() { return new[] { ManageMediaContent, + ManageOwnMedia, }; } @@ -33,6 +35,7 @@ namespace Orchard.MediaLibrary { }, new PermissionStereotype { Name = "Contributor", + Permissions = new[] {ManageOwnMedia} }, }; } diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Properties/AssemblyInfo.cs index d0375acf9..1b1728893 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/ClientStorageMenu.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/ClientStorageMenu.cs index 7f8b8a973..005764bab 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/ClientStorageMenu.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/ClientStorageMenu.cs @@ -15,7 +15,7 @@ namespace Orchard.MediaLibrary.Providers { builder.AddImageSet("clientstorage") .Add(T("My Computer"), "5", menu => menu.Action("Index", "ClientStorage", new { area = "Orchard.MediaLibrary" }) - .Permission(Permissions.ManageMediaContent)); + .Permission(Permissions.ManageOwnMedia)); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/OEmbedMenu.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/OEmbedMenu.cs index 82875e1fb..71aff8019 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/OEmbedMenu.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/OEmbedMenu.cs @@ -15,7 +15,7 @@ namespace Orchard.MediaLibrary.Providers { builder.AddImageSet("oembed") .Add(T("Media Url"), "10", menu => menu.Action("Index", "OEmbed", new { area = "Orchard.MediaLibrary" }) - .Permission(Permissions.ManageMediaContent)); + .Permission(Permissions.ManageOwnMedia)); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/WebSearchMenu.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/WebSearchMenu.cs index 896205d0f..39fb04322 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/WebSearchMenu.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Providers/WebSearchMenu.cs @@ -15,7 +15,7 @@ namespace Orchard.MediaLibrary.Providers { builder.AddImageSet("websearch") .Add(T("Web Search"), "7", menu => menu.Action("Index", "WebSearch", new { area = "Orchard.MediaLibrary" }) - .Permission(Permissions.ManageMediaContent)); + .Permission(Permissions.ManageOwnMedia)); } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Security/MediaAuthorizationEventHandler.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Security/MediaAuthorizationEventHandler.cs new file mode 100644 index 000000000..14bb48916 --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Security/MediaAuthorizationEventHandler.cs @@ -0,0 +1,35 @@ +using Orchard.ContentManagement; +using Orchard.MediaLibrary.Models; +using Orchard.MediaLibrary.Services; +using Orchard.Security; + +namespace Orchard.MediaLibrary.Security { + public class MediaAuthorizationEventHandler : IAuthorizationServiceEventHandler { + private readonly IAuthorizer _authorizer; + private readonly IMediaLibraryService _mediaLibraryService; + + public MediaAuthorizationEventHandler( + IAuthorizer authorizer, + IMediaLibraryService mediaLibraryService) { + _authorizer = authorizer; + _mediaLibraryService = mediaLibraryService; + } + + public void Checking(CheckAccessContext context) { } + public void Complete(CheckAccessContext context) { } + + public void Adjust(CheckAccessContext context) { + var mediaPart = context.Content.As(); + if (mediaPart != null) { + if(_authorizer.Authorize(Permissions.ManageMediaContent)) { + context.Granted = true; + return; + } + + if(_authorizer.Authorize(Permissions.ManageOwnMedia)) { + context.Granted = _mediaLibraryService.CanManageMediaFolder(mediaPart.FolderPath); + } + } + } + } +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/IMediaLibraryService.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/IMediaLibraryService.cs index 7424c9fa5..2291f4c6f 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/IMediaLibraryService.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/IMediaLibraryService.cs @@ -1,4 +1,5 @@ -using System.Collections.Generic; +using System; +using System.Collections.Generic; using System.IO; using System.Web; using Orchard.ContentManagement; @@ -36,7 +37,7 @@ namespace Orchard.MediaLibrary.Services { /// The public URL for the media. string GetMediaPublicUrl(string mediaPath, string fileName); - MediaFolder GetRootMediaFolder(); + IMediaFolder GetRootMediaFolder(); /// /// Retrieves the media folders within a given relative path. @@ -131,4 +132,30 @@ namespace Orchard.MediaLibrary.Services { /// The path to the uploaded file. string UploadMediaFile(string folderPath, string fileName, Stream inputStream); } + + public static class MediaLibrayServiceExtensions { + public static bool CanManageMediaFolder(this IMediaLibraryService service, string folderPath) { + // The current user can manage a media if he has access to the whole hierarchy + // or the media is under his personal storage folder. + + var rootMediaFolder = service.GetRootMediaFolder(); + if (rootMediaFolder == null) { + return true; + } + + var mediaPath = folderPath + "\\"; + var rootPath = rootMediaFolder.MediaPath + "\\"; + + return mediaPath.StartsWith(rootPath, StringComparison.OrdinalIgnoreCase); + } + + public static string GetRootedFolderPath(this IMediaLibraryService service, string folderPath) { + var rootMediaFolder = service.GetRootMediaFolder(); + if (rootMediaFolder != null) { + return Path.Combine(rootMediaFolder.MediaPath, folderPath ?? ""); + } + + return folderPath; + } + } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/MediaLibraryService.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/MediaLibraryService.cs index e7fb1a4fe..21949b99d 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/MediaLibraryService.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/MediaLibraryService.cs @@ -210,7 +210,21 @@ namespace Orchard.MediaLibrary.Services { return GetPublicUrl(Path.Combine(mediaPath, fileName)); } - public MediaFolder GetRootMediaFolder() { + public IMediaFolder GetRootMediaFolder() { + if (_orchardServices.Authorizer.Authorize(Permissions.ManageMediaContent)) { + return null; + } + + if (_orchardServices.Authorizer.Authorize(Permissions.ManageOwnMedia)) { + var currentUser = _orchardServices.WorkContext.CurrentUser; + var userPath = _storageProvider.Combine("Users", currentUser.UserName); + + return new MediaFolder() { + Name = currentUser.UserName, + MediaPath = userPath + }; + } + return null; } diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/XmlRpcHandler.cs b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/XmlRpcHandler.cs index 88eb0be87..14a35ab00 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/XmlRpcHandler.cs +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Services/XmlRpcHandler.cs @@ -60,7 +60,7 @@ namespace Orchard.MediaLibrary.Services { UrlHelper url) { var user = _membershipService.ValidateUser(userName, password); - if (!_authorizationService.TryCheckAccess(Permissions.ManageMediaContent, user, null)) { + if (!_authorizationService.TryCheckAccess(Permissions.ManageOwnMedia, user, null)) { throw new OrchardCoreException(T("Access denied")); } @@ -72,6 +72,11 @@ namespace Orchard.MediaLibrary.Services { directoryName = "media"; } + // If the user only has access to his own folder, rewrite the folder name + if (!_authorizationService.TryCheckAccess(Permissions.ManageMediaContent, user, null)) { + directoryName = Path.Combine(_mediaLibraryService.GetRootedFolderPath(directoryName)); + } + try { // delete the file if it already exists, e.g. an updated image in a blog post // it's safe to delete the file as each content item gets a specific folder diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Admin/Index.cshtml b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Admin/Index.cshtml index 5a9dce068..57eaa43e4 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Admin/Index.cshtml +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Admin/Index.cshtml @@ -112,7 +112,7 @@ var mediaLibrarySettings = {
        -
          +
diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Folder/Create.cshtml b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Folder/Create.cshtml index 3ed81ac0f..c817607d0 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Folder/Create.cshtml +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Views/Folder/Create.cshtml @@ -6,10 +6,12 @@
\ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Drivers/SslSettingsPartDriver.cs b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Drivers/SslSettingsPartDriver.cs index 89dd0cd63..794df2d37 100644 --- a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Drivers/SslSettingsPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Drivers/SslSettingsPartDriver.cs @@ -36,7 +36,6 @@ namespace Orchard.SecureSocketsLayer.Drivers { } protected override void Importing(SslSettingsPart part, ImportContentContext context) { - base.Importing(part, context); _signals.Trigger(SslSettingsPart.CacheKey); } } diff --git a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Module.txt b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Module.txt index 916d28c75..140f74b56 100644 --- a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Module.txt @@ -2,7 +2,7 @@ Name: Secure Sockets Layer AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: This module will ensure SSL is used when accessing specific parts of the website like the dashboard, authentication pages or custom pages. FeatureName: Secure Sockets Layer diff --git a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Properties/AssemblyInfo.cs index 168f1af0d..263450c07 100644 --- a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Commands/SetupCommand.cs b/src/Orchard.Web/Modules/Orchard.Setup/Commands/SetupCommand.cs index e56c2cd61..306c626fe 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Commands/SetupCommand.cs +++ b/src/Orchard.Web/Modules/Orchard.Setup/Commands/SetupCommand.cs @@ -2,14 +2,17 @@ using System.Collections.Generic; using System.Linq; using Orchard.Commands; +using Orchard.Recipes.Services; using Orchard.Setup.Services; namespace Orchard.Setup.Commands { public class SetupCommand : DefaultOrchardCommandHandler { private readonly ISetupService _setupService; + private readonly IRecipeHarvester _recipeHarvester; - public SetupCommand(ISetupService setupService) { + public SetupCommand(ISetupService setupService, IRecipeHarvester recipeHarvester) { _setupService = setupService; + _recipeHarvester = recipeHarvester; } [OrchardSwitch] @@ -51,6 +54,7 @@ namespace Orchard.Setup.Commands { .Where(s => !String.IsNullOrEmpty(s)); } Recipe = String.IsNullOrEmpty(Recipe) ? "Default" : Recipe; + var recipe = _setupService.Recipes().GetRecipeByName(Recipe); var setupContext = new SetupContext { SiteName = SiteName, @@ -60,7 +64,7 @@ namespace Orchard.Setup.Commands { DatabaseConnectionString = DatabaseConnectionString, DatabaseTablePrefix = DatabaseTablePrefix, EnabledFeatures = enabledFeatures, - Recipe = Recipe, + Recipe = recipe, }; var executionId = _setupService.Setup(setupContext); diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Controllers/SetupController.cs b/src/Orchard.Web/Modules/Orchard.Setup/Controllers/SetupController.cs index fba6a53a5..8c2eb4687 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Controllers/SetupController.cs +++ b/src/Orchard.Web/Modules/Orchard.Setup/Controllers/SetupController.cs @@ -7,6 +7,7 @@ using Orchard.Logging; using Orchard.Setup.Services; using Orchard.Setup.ViewModels; using Orchard.Localization; +using Orchard.Recipes.Services; using Orchard.Themes; using Orchard.UI.Notify; @@ -20,10 +21,11 @@ namespace Orchard.Setup.Controllers { private const string DefaultRecipe = "Default"; public SetupController( - INotifier notifier, - ISetupService setupService, + INotifier notifier, + ISetupService setupService, IViewsBackgroundCompilation viewsBackgroundCompilation, ShellSettings shellSettings) { + _viewsBackgroundCompilation = viewsBackgroundCompilation; _shellSettings = shellSettings; _notifier = notifier; @@ -31,10 +33,12 @@ namespace Orchard.Setup.Controllers { T = NullLocalizer.Instance; Logger = NullLogger.Instance; + RecipeExecutionTimeout = 600; } public Localizer T { get; set; } public ILogger Logger { get; set; } + public int RecipeExecutionTimeout { get; set; } private ActionResult IndexViewResult(SetupViewModel model) { return View(model); @@ -44,10 +48,11 @@ namespace Orchard.Setup.Controllers { var initialSettings = _setupService.Prime(); var recipes = _setupService.Recipes().ToList(); string recipeDescription = null; + if (recipes.Count > 0) { recipeDescription = recipes[0].Description; } - + // On the first time installation of Orchard, the user gets to the setup screen, which // will take a while to finish (user inputting data and the setup process itself). // We use this opportunity to start a background task to "pre-compile" all the known @@ -56,10 +61,10 @@ namespace Orchard.Setup.Controllers { if (StringComparer.OrdinalIgnoreCase.Equals(initialSettings.Name, ShellSettings.DefaultName)) { _viewsBackgroundCompilation.Start(); } - + return IndexViewResult(new SetupViewModel { AdminUsername = "admin", - DatabaseIsPreconfigured = !string.IsNullOrEmpty(initialSettings.DataProvider), + DatabaseIsPreconfigured = !String.IsNullOrEmpty(initialSettings.DataProvider), Recipes = recipes, RecipeDescription = recipeDescription }); @@ -67,8 +72,8 @@ namespace Orchard.Setup.Controllers { [HttpPost, ActionName("Index")] public ActionResult IndexPOST(SetupViewModel model) { - // Sets the setup request timeout to 10 minutes to give enough time to execute custom recipes. - HttpContext.Server.ScriptTimeout = 600; + // Sets the setup request timeout to a configurable amount of seconds to give enough time to execute custom recipes. + HttpContext.Server.ScriptTimeout = RecipeExecutionTimeout; var recipes = _setupService.Recipes().ToList(); @@ -76,17 +81,17 @@ namespace Orchard.Setup.Controllers { if (model.DatabaseProvider != SetupDatabaseType.Builtin && string.IsNullOrEmpty(model.DatabaseConnectionString)) ModelState.AddModelError("DatabaseConnectionString", T("A connection string is required.").Text); - if (!string.IsNullOrWhiteSpace(model.ConfirmPassword) && model.AdminPassword != model.ConfirmPassword ) { + if (!String.IsNullOrWhiteSpace(model.ConfirmPassword) && model.AdminPassword != model.ConfirmPassword) { ModelState.AddModelError("ConfirmPassword", T("Password confirmation must match.").Text); } if (model.DatabaseProvider != SetupDatabaseType.Builtin && !string.IsNullOrWhiteSpace(model.DatabaseTablePrefix)) { model.DatabaseTablePrefix = model.DatabaseTablePrefix.Trim(); - if(!char.IsLetter(model.DatabaseTablePrefix[0])) { + if (!Char.IsLetter(model.DatabaseTablePrefix[0])) { ModelState.AddModelError("DatabaseTablePrefix", T("The table prefix must begin with a letter.").Text); } - if(model.DatabaseTablePrefix.Any(x => !Char.IsLetterOrDigit(x))) { + if (model.DatabaseTablePrefix.Any(x => !Char.IsLetterOrDigit(x))) { ModelState.AddModelError("DatabaseTablePrefix", T("The table prefix must contain letters or digits.").Text); } } @@ -103,16 +108,15 @@ namespace Orchard.Setup.Controllers { foreach (var recipe in recipes.Where(recipe => recipe.Name == model.Recipe)) { model.RecipeDescription = recipe.Description; } - model.DatabaseIsPreconfigured = !string.IsNullOrEmpty(_setupService.Prime().DataProvider); - + model.DatabaseIsPreconfigured = !String.IsNullOrEmpty(_setupService.Prime().DataProvider); + return IndexViewResult(model); } try { string providerName = null; - switch (model.DatabaseProvider) - { + switch (model.DatabaseProvider) { case SetupDatabaseType.Builtin: providerName = "SqlCe"; break; @@ -133,6 +137,7 @@ namespace Orchard.Setup.Controllers { throw new ApplicationException("Unknown database type: " + model.DatabaseProvider); } + var recipe = _setupService.Recipes().GetRecipeByName(model.Recipe); var setupContext = new SetupContext { SiteName = model.SiteName, AdminUsername = model.AdminUsername, @@ -140,11 +145,11 @@ namespace Orchard.Setup.Controllers { DatabaseProvider = providerName, DatabaseConnectionString = model.DatabaseConnectionString, DatabaseTablePrefix = model.DatabaseTablePrefix, - EnabledFeatures = null, // default list - Recipe = model.Recipe + EnabledFeatures = null, // Default list + Recipe = recipe }; - string executionId = _setupService.Setup(setupContext); + var executionId = _setupService.Setup(setupContext); // First time installation if finally done. Tell the background views compilation // process to stop, so that it doesn't interfere with the user (asp.net compilation @@ -153,7 +158,8 @@ namespace Orchard.Setup.Controllers { // Redirect to the welcome page. return Redirect("~/" + _shellSettings.RequestUrlPrefix); - } catch (Exception ex) { + } + catch (Exception ex) { Logger.Error(ex, "Setup failed"); _notifier.Error(T("Setup failed: {0}", ex.Message)); @@ -161,7 +167,7 @@ namespace Orchard.Setup.Controllers { foreach (var recipe in recipes.Where(recipe => recipe.Name == model.Recipe)) { model.RecipeDescription = recipe.Description; } - model.DatabaseIsPreconfigured = !string.IsNullOrEmpty(_setupService.Prime().DataProvider); + model.DatabaseIsPreconfigured = !String.IsNullOrEmpty(_setupService.Prime().DataProvider); return IndexViewResult(model); } diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Controllers/SetupDatabaseType.cs b/src/Orchard.Web/Modules/Orchard.Setup/Controllers/SetupDatabaseType.cs deleted file mode 100644 index 48265336c..000000000 --- a/src/Orchard.Web/Modules/Orchard.Setup/Controllers/SetupDatabaseType.cs +++ /dev/null @@ -1,15 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using System.Web; - -namespace Orchard.Setup.Controllers -{ - public enum SetupDatabaseType - { - Builtin, - SqlServer, - MySql, - PostgreSql, - } -} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Module.txt b/src/Orchard.Web/Modules/Orchard.Setup/Module.txt index 9895478e3..10e6b8582 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Setup/Module.txt @@ -1,8 +1,14 @@ Name: Setup Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The setup module is creating the application's setup experience. FeatureDescription: Standard site setup. This feature is disabled automatically once setup is over. Category: Core +Dependencies: Orchard.Setup.Services +Features: + Orchard.Setup.Services + Name: Setup Services + Description: Provides programmatic APIs to the setup service. + Category: Core diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Orchard.Setup.csproj b/src/Orchard.Web/Modules/Orchard.Setup/Orchard.Setup.csproj index 94d8c676d..6e7182827 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Orchard.Setup.csproj +++ b/src/Orchard.Web/Modules/Orchard.Setup/Orchard.Setup.csproj @@ -25,6 +25,7 @@ + true @@ -78,7 +79,7 @@ - + diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Setup/Properties/AssemblyInfo.cs index b64d7d032..72f4c97ed 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Setup/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Routes.cs b/src/Orchard.Web/Modules/Orchard.Setup/Routes.cs index 2bc323f36..faafaa50e 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Routes.cs +++ b/src/Orchard.Web/Modules/Orchard.Setup/Routes.cs @@ -4,8 +4,7 @@ using System.Web.Routing; using Orchard.Mvc.Routes; namespace Orchard.Setup { - public class Routes : IRouteProvider - { + public class Routes : IRouteProvider { public void GetRoutes(ICollection routes) { foreach (var routeDescriptor in GetRoutes()) routes.Add(routeDescriptor); @@ -13,24 +12,24 @@ namespace Orchard.Setup { public IEnumerable GetRoutes() { return new[] { - new RouteDescriptor { - Route = new Route( - "{controller}/{action}", - new RouteValueDictionary { - {"area", "Orchard.Setup"}, - {"controller", "Setup"}, - {"action", "Index"} - }, - new RouteValueDictionary { - {"area", "Orchard.Setup"}, - {"controller", "Setup"}, - }, - new RouteValueDictionary { - {"area", "Orchard.Setup"} - }, - new MvcRouteHandler()) - } - }; + new RouteDescriptor { + Route = new Route( + "{controller}/{action}", + new RouteValueDictionary { + {"area", "Orchard.Setup"}, + {"controller", "Setup"}, + {"action", "Index"} + }, + new RouteValueDictionary { + {"area", "Orchard.Setup"}, + {"controller", "Setup"}, + }, + new RouteValueDictionary { + {"area", "Orchard.Setup"} + }, + new MvcRouteHandler()) + } + }; } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupContext.cs b/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupContext.cs index 06c32c914..0c87402bc 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupContext.cs +++ b/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupContext.cs @@ -1,4 +1,5 @@ using System.Collections.Generic; +using Orchard.Recipes.Models; namespace Orchard.Setup.Services { public class SetupContext { @@ -9,6 +10,6 @@ namespace Orchard.Setup.Services { public string DatabaseConnectionString { get; set; } public string DatabaseTablePrefix { get; set; } public IEnumerable EnabledFeatures { get; set; } - public string Recipe { get; set; } + public Recipe Recipe { get; set; } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupService.cs b/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupService.cs index 16b3e465b..7549405cb 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupService.cs +++ b/src/Orchard.Web/Modules/Orchard.Setup/Services/SetupService.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using System.Linq; using System.Security.Cryptography; using System.Web; +using System.Xml.Linq; using Orchard.ContentManagement; using Orchard.Core.Settings.Models; using Orchard.Data; @@ -16,7 +17,6 @@ using Orchard.Environment.Descriptor.Models; using Orchard.Environment.Extensions; using Orchard.Environment.ShellBuilders; using Orchard.Environment.State; -using Orchard.Localization; using Orchard.Localization.Services; using Orchard.Logging; using Orchard.Recipes.Models; @@ -26,14 +26,14 @@ using Orchard.Settings; using Orchard.Utility.Extensions; namespace Orchard.Setup.Services { - public class SetupService : ISetupService { + [OrchardFeature("Orchard.Setup.Services")] + public class SetupService : Component, ISetupService { private readonly ShellSettings _shellSettings; private readonly IOrchardHost _orchardHost; private readonly IShellSettingsManager _shellSettingsManager; private readonly IShellContainerFactory _shellContainerFactory; private readonly ICompositionStrategy _compositionStrategy; private readonly IProcessingEngine _processingEngine; - private readonly IExtensionManager _extensionManager; private readonly IRecipeHarvester _recipeHarvester; private IEnumerable _recipes; @@ -44,23 +44,17 @@ namespace Orchard.Setup.Services { IShellContainerFactory shellContainerFactory, ICompositionStrategy compositionStrategy, IProcessingEngine processingEngine, - IExtensionManager extensionManager, IRecipeHarvester recipeHarvester) { + _shellSettings = shellSettings; _orchardHost = orchardHost; _shellSettingsManager = shellSettingsManager; _shellContainerFactory = shellContainerFactory; _compositionStrategy = compositionStrategy; _processingEngine = processingEngine; - _extensionManager = extensionManager; _recipeHarvester = recipeHarvester; - T = NullLocalizer.Instance; - Logger = NullLogger.Instance; } - public Localizer T { get; set; } - public ILogger Logger { get; set; } - public ShellSettings Prime() { return _shellSettings; } @@ -68,54 +62,60 @@ namespace Orchard.Setup.Services { public IEnumerable Recipes() { if (_recipes == null) { var recipes = new List(); - - foreach (var extension in _extensionManager.AvailableExtensions()) { - recipes.AddRange(_recipeHarvester.HarvestRecipes(extension.Id).Where(recipe => recipe.IsSetupRecipe)); - } - + recipes.AddRange(_recipeHarvester.HarvestRecipes().Where(recipe => recipe.IsSetupRecipe)); _recipes = recipes; } - return _recipes; } public string Setup(SetupContext context) { + var initialState = _shellSettings.State; + try { + return SetupInternal(context); + } + catch { + _shellSettings.State = initialState; + throw; + } + } + + private string SetupInternal(SetupContext context) { string executionId; Logger.Information("Running setup for tenant '{0}'.", _shellSettings.Name); // The vanilla Orchard distibution has the following features enabled. string[] hardcoded = { - // Framework - "Orchard.Framework", - // Core - "Common", "Containers", "Contents", "Dashboard", "Feeds", "Navigation","Scheduling", "Settings", "Shapes", "Title", - // Modules - "Orchard.Pages", "Orchard.ContentPicker", "Orchard.Themes", "Orchard.Users", "Orchard.Roles", "Orchard.Modules", - "PackagingServices", "Orchard.Packaging", "Gallery", "Orchard.Recipes" - }; + // Framework + "Orchard.Framework", + // Core + "Common", "Containers", "Contents", "Dashboard", "Feeds", "Navigation","Scheduling", "Settings", "Shapes", "Title", + // Modules + "Orchard.Pages", "Orchard.ContentPicker", "Orchard.Themes", "Orchard.Users", "Orchard.Roles", "Orchard.Modules", + "PackagingServices", "Orchard.Packaging", "Gallery", "Orchard.Recipes" + }; context.EnabledFeatures = hardcoded.Union(context.EnabledFeatures ?? Enumerable.Empty()).Distinct().ToList(); + // Set shell state to "Initializing" so that subsequent HTTP requests are responded to with "Service Unavailable" while Orchard is setting up. + _shellSettings.State = TenantState.Initializing; + var shellSettings = new ShellSettings(_shellSettings); - if (string.IsNullOrEmpty(shellSettings.DataProvider)) { + if (String.IsNullOrEmpty(shellSettings.DataProvider)) { shellSettings.DataProvider = context.DatabaseProvider; shellSettings.DataConnectionString = context.DatabaseConnectionString; shellSettings.DataTablePrefix = context.DatabaseTablePrefix; } - #region Encryption Settings - shellSettings.EncryptionAlgorithm = "AES"; - // randomly generated key + + // Randomly generated key. shellSettings.EncryptionKey = SymmetricAlgorithm.Create(shellSettings.EncryptionAlgorithm).Key.ToHexString(); - shellSettings.HashAlgorithm = "HMACSHA256"; - // randomly generated key - shellSettings.HashKey = HMAC.Create(shellSettings.HashAlgorithm).Key.ToHexString(); - #endregion + // Randomly generated key. + shellSettings.HashKey = HMAC.Create(shellSettings.HashAlgorithm).Key.ToHexString(); var shellDescriptor = new ShellDescriptor { Features = context.EnabledFeatures.Select(name => new ShellFeature { Name = name }) @@ -123,12 +123,12 @@ namespace Orchard.Setup.Services { var shellBlueprint = _compositionStrategy.Compose(shellSettings, shellDescriptor); - // initialize database explicitly, and store shell descriptor + // Initialize database explicitly, and store shell descriptor. using (var bootstrapLifetimeScope = _shellContainerFactory.CreateContainer(shellSettings, shellBlueprint)) { using (var environment = bootstrapLifetimeScope.CreateWorkContextScope()) { - // check if the database is already created (in case an exception occured in the second phase) + // Check if the database is already created (in case an exception occured in the second phase). var schemaBuilder = new SchemaBuilder(environment.Resolve()); var installationPresent = true; try { @@ -143,7 +143,7 @@ namespace Orchard.Setup.Services { throw new OrchardException(T("A previous Orchard installation was detected in this database with this table prefix.")); } - // Make a workaround to avoid the Transaction issue for PostgreSQL + // Workaround to avoid some Transaction issue for PostgreSQL. environment.Resolve().RequireNew(); schemaBuilder.CreateTable("Orchard_Framework_DataMigrationRecord", table => table @@ -168,16 +168,13 @@ namespace Orchard.Setup.Services { } } - // in effect "pump messages" see PostMessage circa 1980 + // In effect "pump messages" see PostMessage circa 1980. while (_processingEngine.AreTasksPending()) _processingEngine.ExecuteNextTask(); - // creating a standalone environment. + // Creating a standalone environment. // in theory this environment can be used to resolve any normal components by interface, and those - // components will exist entirely in isolation - no crossover between the safemode container currently in effect - - // must mark state as Running - otherwise standalone enviro is created "for setup" - shellSettings.State = TenantState.Running; + // components will exist entirely in isolation - no crossover between the safemode container currently in effect. using (var environment = _orchardHost.CreateStandaloneEnvironment(shellSettings)) { try { executionId = CreateTenantData(context, environment); @@ -189,23 +186,21 @@ namespace Orchard.Setup.Services { } _shellSettingsManager.SaveSettings(shellSettings); - return executionId; } private string CreateTenantData(SetupContext context, IWorkContextScope environment) { - // create superuser + // Create site owner. var membershipService = environment.Resolve(); - var user = - membershipService.CreateUser(new CreateUserParams(context.AdminUsername, context.AdminPassword, - String.Empty, String.Empty, String.Empty, - true)); + var user = membershipService.CreateUser( + new CreateUserParams(context.AdminUsername, context.AdminPassword, + String.Empty, String.Empty, String.Empty, true)); - // set superuser as current user for request (it will be set as the owner of all content items) + // Set site owner as current user for request (it will be set as the owner of all content items). var authenticationService = environment.Resolve(); authenticationService.SetAuthenticatedUserForRequest(user); - // set site name and settings + // Set site name and settings. var siteService = environment.Resolve(); var siteSettings = siteService.GetSiteSettings().As(); siteSettings.SiteSalt = Guid.NewGuid().ToString("N"); @@ -213,19 +208,27 @@ namespace Orchard.Setup.Services { siteSettings.SuperUser = context.AdminUsername; siteSettings.SiteCulture = "en-US"; - // add default culture + // Add default culture. var cultureManager = environment.Resolve(); cultureManager.AddCulture("en-US"); var recipeManager = environment.Resolve(); - var recipe = Recipes().FirstOrDefault(r => r.Name.Equals(context.Recipe, StringComparison.OrdinalIgnoreCase)); - - if (recipe == null) - throw new OrchardException(T("The recipe '{0}' could not be found.", context.Recipe)); - + var recipe = context.Recipe; var executionId = recipeManager.Execute(recipe); - // null check: temporary fix for running setup in command line + // Once the recipe has finished executing, we need to update the shell state to "Running", so add a recipe step that does exactly that. + var recipeStepQueue = environment.Resolve(); + var recipeStepResultRecordRepository = environment.Resolve>(); + var activateShellStep = new RecipeStep(Guid.NewGuid().ToString("N"), recipe.Name, "ActivateShell", new XElement("ActivateShell")); + recipeStepQueue.Enqueue(executionId, activateShellStep); + recipeStepResultRecordRepository.Create(new RecipeStepResultRecord { + ExecutionId = executionId, + RecipeName = recipe.Name, + StepId = activateShellStep.Id, + StepName = activateShellStep.Name + }); + + // Null check: temporary fix for running setup in command line. if (HttpContext.Current != null) { authenticationService.SignIn(user, true); } diff --git a/src/Orchard.Web/Modules/Orchard.Setup/SetupMode.cs b/src/Orchard.Web/Modules/Orchard.Setup/SetupMode.cs index ec675dac8..e0ee00076 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/SetupMode.cs +++ b/src/Orchard.Web/Modules/Orchard.Setup/SetupMode.cs @@ -1,5 +1,4 @@ using System; -using System.Collections.Generic; using System.Web.Routing; using Autofac; using Orchard.Caching; diff --git a/src/Orchard.Web/Modules/Orchard.Setup/ViewModels/SetupDatabaseType.cs b/src/Orchard.Web/Modules/Orchard.Setup/ViewModels/SetupDatabaseType.cs new file mode 100644 index 000000000..f57a86748 --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.Setup/ViewModels/SetupDatabaseType.cs @@ -0,0 +1,10 @@ +namespace Orchard.Setup.ViewModels +{ + public enum SetupDatabaseType + { + Builtin, + SqlServer, + MySql, + PostgreSql, + } +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Setup/Views/Setup/Index.cshtml b/src/Orchard.Web/Modules/Orchard.Setup/Views/Setup/Index.cshtml index b1990c468..9f6fe9130 100644 --- a/src/Orchard.Web/Modules/Orchard.Setup/Views/Setup/Index.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Setup/Views/Setup/Index.cshtml @@ -1,5 +1,6 @@ @model Orchard.Setup.ViewModels.SetupViewModel @using Orchard.Recipes.Models; +@using Orchard.Setup.ViewModels @{ Script.Require("jQuery"); Script.Require("ShapesBase"); @@ -47,19 +48,19 @@ if (!Model.DatabaseIsPreconfigured) { @T("How would you like to store your data?") @Html.ValidationMessage("DatabaseOptions", "Unable to setup data storage.")
- @Html.RadioButtonFor(svm => svm.DatabaseProvider, Orchard.Setup.Controllers.SetupDatabaseType.Builtin.ToString(), new { id = "builtin" }) + @Html.RadioButtonFor(svm => svm.DatabaseProvider, SetupDatabaseType.Builtin.ToString(), new { id = "builtin" })
- @Html.RadioButtonFor(svm => svm.DatabaseProvider, Orchard.Setup.Controllers.SetupDatabaseType.SqlServer.ToString(), new { id = "sqlserver" }) + @Html.RadioButtonFor(svm => svm.DatabaseProvider, SetupDatabaseType.SqlServer.ToString(), new { id = "sqlserver" })
- @Html.RadioButtonFor(svm => svm.DatabaseProvider, Orchard.Setup.Controllers.SetupDatabaseType.MySql.ToString(), new { id = "mysql" }) + @Html.RadioButtonFor(svm => svm.DatabaseProvider, SetupDatabaseType.MySql.ToString(), new { id = "mysql" })
- @Html.RadioButtonFor(svm => svm.DatabaseProvider, Orchard.Setup.Controllers.SetupDatabaseType.PostgreSql.ToString(), new { id = "postgresql" }) + @Html.RadioButtonFor(svm => svm.DatabaseProvider, SetupDatabaseType.PostgreSql.ToString(), new { id = "postgresql" })
diff --git a/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagCloudDriver.cs b/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagCloudDriver.cs index 345c8f576..eb3fe1538 100644 --- a/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagCloudDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagCloudDriver.cs @@ -43,6 +43,11 @@ namespace Orchard.Tags.Drivers { } protected override void Importing(TagCloudPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + part.Slug = context.Attribute(part.PartDefinition.Name, "Slug"); part.Buckets = Convert.ToInt32(context.Attribute(part.PartDefinition.Name, "Buckets")); } diff --git a/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagsPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagsPartDriver.cs index 2c42a27c8..669ed9df7 100644 --- a/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagsPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Tags/Drivers/TagsPartDriver.cs @@ -70,6 +70,11 @@ namespace Orchard.Tags.Drivers { } protected override void Importing(TagsPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + var tagString = context.Attribute(part.PartDefinition.Name, "Tags"); if (tagString != null) { var tags = tagString.Split(new[] {","}, StringSplitOptions.RemoveEmptyEntries); diff --git a/src/Orchard.Web/Modules/Orchard.Tags/Module.txt b/src/Orchard.Web/Modules/Orchard.Tags/Module.txt index 5fedb9b0e..c29f0a14b 100644 --- a/src/Orchard.Web/Modules/Orchard.Tags/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Tags/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Features: Orchard.Tags: diff --git a/src/Orchard.Web/Modules/Orchard.Tags/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Tags/Properties/AssemblyInfo.cs index 62e65e560..2c420e1bd 100644 --- a/src/Orchard.Web/Modules/Orchard.Tags/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Tags/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Tags/Services/TagCloudService.cs b/src/Orchard.Web/Modules/Orchard.Tags/Services/TagCloudService.cs index 26dbf2c52..a05644f69 100644 --- a/src/Orchard.Web/Modules/Orchard.Tags/Services/TagCloudService.cs +++ b/src/Orchard.Web/Modules/Orchard.Tags/Services/TagCloudService.cs @@ -32,7 +32,7 @@ namespace Orchard.Tags.Services { public IEnumerable GetPopularTags(int buckets, string slug) { var cacheKey = "Orchard.Tags.TagCloud." + (slug ?? "") + '.' + buckets; - return _cacheManager.Get(cacheKey, + return _cacheManager.Get(cacheKey, true, ctx => { ctx.Monitor(_signals.When(TagCloudTagsChanged)); IEnumerable tagCounts; diff --git a/src/Orchard.Web/Modules/Orchard.TaskLease/Models/TaskLeaseRecord.cs b/src/Orchard.Web/Modules/Orchard.TaskLease/Models/TaskLeaseRecord.cs index 3e28a67f3..039e0d27e 100644 --- a/src/Orchard.Web/Modules/Orchard.TaskLease/Models/TaskLeaseRecord.cs +++ b/src/Orchard.Web/Modules/Orchard.TaskLease/Models/TaskLeaseRecord.cs @@ -1,7 +1,4 @@ using System; -using System.Collections.Generic; -using System.Linq; -using System.Web; using Orchard.Data.Conventions; namespace Orchard.TaskLease.Models diff --git a/src/Orchard.Web/Modules/Orchard.TaskLease/Module.txt b/src/Orchard.Web/Modules/Orchard.TaskLease/Module.txt index 7748ef35a..868d44247 100644 --- a/src/Orchard.Web/Modules/Orchard.TaskLease/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.TaskLease/Module.txt @@ -2,7 +2,7 @@ Name: Task Lease AntiForgery: enabled Author: The Orchard Team Website: http://orchardtasklease.codeplex.com -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 LifecycleStatus: Deprecated Description: Provides services to synchronize tasks in a farm environment. diff --git a/src/Orchard.Web/Modules/Orchard.TaskLease/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.TaskLease/Properties/AssemblyInfo.cs index 3dabfb343..b2b8e12fd 100644 --- a/src/Orchard.Web/Modules/Orchard.TaskLease/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.TaskLease/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.TaskLease/Services/ITaskLeaseService.cs b/src/Orchard.Web/Modules/Orchard.TaskLease/Services/ITaskLeaseService.cs index 625472446..c32e3d37e 100644 --- a/src/Orchard.Web/Modules/Orchard.TaskLease/Services/ITaskLeaseService.cs +++ b/src/Orchard.Web/Modules/Orchard.TaskLease/Services/ITaskLeaseService.cs @@ -6,7 +6,7 @@ namespace Orchard.TaskLease.Services { /// Describes a service to save and acquire task leases. A task lease can't be acquired by two different machines, /// for a specific amount of time. Optionnally a State can be saved along with the lease. /// - [Obsolete("Use Orchard.Tasks.Locking.IDistributedLockService and the AcquireLockForMachine/TryAcquireLockForMachine methods instead.")] + [Obsolete("Use Orchard.Tasks.Locking.IDistributedLockService instead.")] public interface ITaskLeaseService : IDependency { /// diff --git a/src/Orchard.Web/Modules/Orchard.TaskLease/Services/TaskLeaseService.cs b/src/Orchard.Web/Modules/Orchard.TaskLease/Services/TaskLeaseService.cs index 8cf04b3fe..0c1025173 100644 --- a/src/Orchard.Web/Modules/Orchard.TaskLease/Services/TaskLeaseService.cs +++ b/src/Orchard.Web/Modules/Orchard.TaskLease/Services/TaskLeaseService.cs @@ -9,25 +9,25 @@ namespace Orchard.TaskLease.Services { /// /// Provides a database driven implementation of /// - [Obsolete("Use Orchard.Tasks.Locking.DistributedLockService and the AcquireLockForMachine/TryAcquireLockForMachine methods instead.")] + [Obsolete("Use Orchard.Tasks.Locking.DistributedLockService instead.")] public class TaskLeaseService : ITaskLeaseService { private readonly IRepository _repository; private readonly IClock _clock; - private readonly IMachineNameProvider _machineNameProvider; + private readonly IApplicationEnvironment _applicationEnvironment; public TaskLeaseService( IRepository repository, IClock clock, - IMachineNameProvider machineNameProvider) { + IApplicationEnvironment applicationEnvironment) { _repository = repository; _clock = clock; - _machineNameProvider = machineNameProvider; + _applicationEnvironment = applicationEnvironment; } public string Acquire(string taskName, DateTime expiredUtc) { - var machineName = _machineNameProvider.GetMachineName(); + var environmentIdentifier = _applicationEnvironment.GetEnvironmentIdentifier(); // retrieve current lease for the specified task var taskLease = _repository.Get(x => x.TaskName == taskName); @@ -36,7 +36,7 @@ namespace Orchard.TaskLease.Services { if (taskLease == null) { taskLease = new TaskLeaseRecord { TaskName = taskName, - MachineName = machineName, + MachineName = environmentIdentifier, State = String.Empty, UpdatedUtc = _clock.UtcNow, ExpiredUtc = expiredUtc @@ -49,12 +49,12 @@ namespace Orchard.TaskLease.Services { } // lease can't be aquired only if for a different machine and it has not expired - if (taskLease.MachineName != machineName && taskLease.ExpiredUtc >= _clock.UtcNow) { + if (taskLease.MachineName != environmentIdentifier && taskLease.ExpiredUtc >= _clock.UtcNow) { return null; } // otherwise update information - taskLease.MachineName = machineName; + taskLease.MachineName = environmentIdentifier; taskLease.UpdatedUtc = _clock.UtcNow; taskLease.ExpiredUtc = expiredUtc; @@ -64,10 +64,10 @@ namespace Orchard.TaskLease.Services { } public void Update(string taskName, string state) { - var machineName = _machineNameProvider.GetMachineName(); + var environmentIdentifier = _applicationEnvironment.GetEnvironmentIdentifier(); // retrieve current lease for the specified task - var taskLease = _repository.Get(x => x.TaskName == taskName && x.MachineName == machineName); + var taskLease = _repository.Get(x => x.TaskName == taskName && x.MachineName == environmentIdentifier); if (taskLease == null) { return; @@ -78,10 +78,10 @@ namespace Orchard.TaskLease.Services { } public void Update(string taskName, string state, DateTime expiredUtc) { - var machineName = _machineNameProvider.GetMachineName(); + var environmentIdentifier = _applicationEnvironment.GetEnvironmentIdentifier(); // retrieve current lease for the specified task - var taskLease = _repository.Get(x => x.TaskName == taskName && x.MachineName == machineName); + var taskLease = _repository.Get(x => x.TaskName == taskName && x.MachineName == environmentIdentifier); if (taskLease == null) { return; diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Controllers/TagsController.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Controllers/TagsController.cs new file mode 100644 index 000000000..bc1d929f3 --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Controllers/TagsController.cs @@ -0,0 +1,63 @@ +using System; +using System.Collections.Generic; +using System.Globalization; +using System.Web.Http; +using Orchard.ContentManagement; +using Orchard.Core.Title.Models; +using Orchard.Localization; +using Orchard.Logging; +using Orchard.Taxonomies.Helpers; +using Orchard.Taxonomies.Models; +using Orchard.Taxonomies.Services; +using System.Linq; +using Orchard.Security; +using Orchard.Taxonomies.ViewModels; + +namespace Orchard.Taxonomies.Controllers { + public class TagsController : ApiController { + private readonly ITaxonomyService _taxonomyService; + private readonly IContentManager _contentManager; + private readonly IAuthorizer _authorizer; + public Localizer T { get; set; } + protected ILogger Logger { get; set; } + + public TagsController( + ITaxonomyService taxonomyService, + IContentManager contentManager, + IAuthorizer authorizer) { + _taxonomyService = taxonomyService; + T = NullLocalizer.Instance; + _contentManager = contentManager; + _authorizer = authorizer; + Logger = NullLogger.Instance; + } + + public IEnumerable Get(int taxonomyId, bool leavesOnly, string query) { + if (!_authorizer.Authorize(StandardPermissions.AccessAdminPanel)) { + throw new UnauthorizedAccessException("Can't access the admin"); + } + if (string.IsNullOrEmpty(query)) return new List(); + var allTerms = leavesOnly + ? _taxonomyService.GetTerms(taxonomyId).ToList() + : new List(); + var matchingTerms = _contentManager.Query() + .Where(t => t.TaxonomyId == taxonomyId) + .Join() + .Where(r => r.Title.Contains(query)) + .List() + .Select(t => BuildTag(t, leavesOnly, allTerms)) + .OrderBy(t => t.Label) + .ToList(); + return matchingTerms; + } + + private static Tag BuildTag(TermPart term, bool leavesOnly, IEnumerable terms) { + return new Tag { + Value = term.Id, + Label = term.Name, + Disabled = !term.Selectable || (leavesOnly && terms.Any(t => t.Path.Contains(term.Path + term.Id))), + Levels = term.GetLevels() + }; + } + } +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyFieldDriver.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyFieldDriver.cs index 7cbc32363..eed4d476c 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyFieldDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyFieldDriver.cs @@ -60,7 +60,7 @@ namespace Orchard.Taxonomies.Drivers { var settings = field.PartFieldDefinition.Settings.GetModel(); var appliedTerms = GetAppliedTerms(part, field, VersionOptions.Latest).ToDictionary(t => t.Id, t => t); var taxonomy = _taxonomyService.GetTaxonomyByName(settings.Taxonomy); - var terms = taxonomy != null + var terms = taxonomy != null && !settings.Autocomplete ? _taxonomyService.GetTerms(taxonomy.Id).Where(t => !string.IsNullOrWhiteSpace(t.Name)).Select(t => t.CreateTermEntry()).ToList() : new List(0); @@ -70,9 +70,11 @@ namespace Orchard.Taxonomies.Drivers { DisplayName = field.DisplayName, Name = field.Name, Terms = terms, + SelectedTerms = appliedTerms.Select(t => t.Value), Settings = settings, - SingleTermId = terms.Where(t => t.IsChecked).Select(t => t.Id).FirstOrDefault(), - TaxonomyId = taxonomy != null ? taxonomy.Id : 0 + SingleTermId = appliedTerms.Select(t => t.Key).FirstOrDefault(), + TaxonomyId = taxonomy != null ? taxonomy.Id : 0, + HasTerms = taxonomy != null && _taxonomyService.GetTermsCount(taxonomy.Id) > 0 }; var templateName = settings.Autocomplete ? "Fields/TaxonomyField.Autocomplete" : "Fields/TaxonomyField"; @@ -108,10 +110,10 @@ namespace Orchard.Taxonomies.Drivers { var appliedTerms = _taxonomyService.GetTermsForContentItem(part.ContentItem.Id, field.Name); // stores all content items associated to this field - var termIdentities = appliedTerms.Select(x => Services.ContentManager.GetItemMetadata(x).Identity.ToString()) - .ToArray(); + var termIdentities = appliedTerms.Select(x => Services.ContentManager.GetItemMetadata(x).Identity.ToString()).ToArray(); - context.Element(XmlConvert.EncodeLocalName(field.FieldDefinition.Name + "." + field.Name)).SetAttributeValue("Terms", String.Join(",", termIdentities)); + if (termIdentities.Any()) + context.Element(XmlConvert.EncodeLocalName(field.FieldDefinition.Name + "." + field.Name)).SetAttributeValue("Terms", String.Join(",", termIdentities)); } protected override void Importing(ContentPart part, TaxonomyField field, ImportContentContext context) { diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyNavigationPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyNavigationPartDriver.cs index 2e94695b4..58bf62dc9 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyNavigationPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyNavigationPartDriver.cs @@ -101,6 +101,11 @@ namespace Orchard.Taxonomies.Drivers { } protected override void Importing(TaxonomyNavigationPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + part.DisplayContentCount = Boolean.Parse(context.Attribute(part.PartDefinition.Name, "DisplayContentCount")); part.DisplayRootTerm = Boolean.Parse(context.Attribute(part.PartDefinition.Name, "DisplayRootTerm")); part.HideEmptyTerms = Boolean.Parse(context.Attribute(part.PartDefinition.Name, "HideEmptyTerms")); diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyPartDriver.cs index 017664b74..e796aa789 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TaxonomyPartDriver.cs @@ -65,6 +65,11 @@ namespace Orchard.Taxonomies.Drivers { } protected override void Importing(TaxonomyPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + part.TermTypeName = context.Attribute(part.PartDefinition.Name, "TermTypeName"); } } diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TermPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TermPartDriver.cs index 10440230b..c99b638a6 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TermPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Drivers/TermPartDriver.cs @@ -126,6 +126,11 @@ namespace Orchard.Taxonomies.Drivers { } protected override void Importing(TermPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + part.Count = Int32.Parse(context.Attribute(part.PartDefinition.Name, "Count")); part.Selectable = Boolean.Parse(context.Attribute(part.PartDefinition.Name, "Selectable")); part.Weight = Int32.Parse(context.Attribute(part.PartDefinition.Name, "Weight")); diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Module.txt b/src/Orchard.Web/Modules/Orchard.Taxonomies/Module.txt index bf7079a4e..294a899c6 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The taxonomy module is providing custom categorization of arbitrary content types. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Orchard.Taxonomies.csproj b/src/Orchard.Web/Modules/Orchard.Taxonomies/Orchard.Taxonomies.csproj index 28d48a98d..b3725af1a 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Orchard.Taxonomies.csproj +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Orchard.Taxonomies.csproj @@ -50,6 +50,9 @@ + + ..\..\..\..\lib\newtonsoft.json\Newtonsoft.Json.dll + 3.5 @@ -59,6 +62,10 @@ + + False + ..\..\..\..\lib\aspnetwebapi\System.Web.Http.dll + False ..\..\..\..\lib\aspnetmvc\System.Web.Mvc.dll @@ -72,6 +79,8 @@ + + diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Properties/AssemblyInfo.cs index 2e45f4835..92f4e3d3d 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Properties/AssemblyInfo.cs @@ -29,5 +29,5 @@ using System.Runtime.InteropServices; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Scripts/admin-taxonomy-tags.js b/src/Orchard.Web/Modules/Orchard.Taxonomies/Scripts/admin-taxonomy-tags.js index 26518b153..95033e84b 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Scripts/admin-taxonomy-tags.js +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Scripts/admin-taxonomy-tags.js @@ -2,49 +2,24 @@ /* Helper functions **********************************************************************/ - var addTag = function ($plugin, label) { - $plugin.tagit("add", label); - }; - - var removeTag = function ($plugin, label) { - var tags = $plugin.tagit("tags"); - var index = findTagIndexByLabel(tags, label); - - if (index == -1) - return; - - tags.splice(index, 1); - $plugin.tagit("fill", tags); - }; - - var findTagIndexByLabel = function (tags, label) { - for (var i = 0; i < tags.length; i++) { - var tag = tags[i]; - - if (tag.label == label) { - return i; - } - } - return -1; - }; - var createTermCheckbox = function ($wrapper, tag) { var $ul = $("ul.terms", $wrapper); var singleChoice = $(".terms-editor", $wrapper).data("singlechoice"); var namePrefix = $wrapper.data("name-prefix"); var idPrefix = $wrapper.data("id-prefix"); var nextIndex = $("li", $ul).length; + var id = isNaN(tag.value) ? -nextIndex : tag.value; var checkboxId = idPrefix + "_Terms_" + nextIndex + "__IsChecked"; var checkboxName = namePrefix + ".Terms[" + nextIndex + "].IsChecked"; var radioName = namePrefix + ".SingleTermId"; - var checkboxHtml = ""; - var radioHtml = ""; + var checkboxHtml = ""; + var radioHtml = ""; var inputHtml = singleChoice ? radioHtml : checkboxHtml; var $li = $("
  • " + inputHtml + - "" + - "" + - "" + + "" + + "" + + "" + "
  • ").hide(); if (singleChoice) { @@ -68,67 +43,20 @@ var $tagIt = $("ul.tagit", $wrapper); var singleChoice = $(".terms-editor", $wrapper).data("singlechoice"); var $terms = $("ul.terms", $wrapper); - var $termCheckbox = $("input[data-term-identity='" + tag.label.toLowerCase() + "']", $terms).filter(function () { - return $(this).siblings("input[value='" + tag.value + "']").length; - }); - - if ($termCheckbox.is(":disabled")) { - removeTag($tagIt, tagLabelOrValue); - return; - } - - if ($termCheckbox.length == 0 && action == "added") { - createTermCheckbox($wrapper, tag.label, this); - } - - $termCheckbox.prop("checked", action == "added"); if (singleChoice && action == "added") { - $tagIt.tagit("fill", [tag.label]); + $tagIt.tagit("fill", tag); } - if (singleChoice) { - if (action == "added") { - var $option = $("select.term-picker", $wrapper).find("option[data-term-identity='" + tag.label.toLowerCase() + "']"); - $option.remove(); - } else { - $("select.term-picker", $wrapper).append(""); - } - } + $terms.empty(); + + $($tagIt.tagit("tags")).each(function (index, tag) { + createTermCheckbox($wrapper, tag, this); + }); $(".no-terms", $wrapper).hide(); }; - $("fieldset.taxonomy-wrapper .expando").on("change", "input[data-term]:enabled", function(e) { - var $checkbox = $(this); - var term = $checkbox.data("term"); - var $wrapper = $checkbox.parents("fieldset.taxonomy-wrapper:first"); - var $tagIt = $("ul.tagit", $wrapper); - var isChecked = $checkbox.is(":checked"); - - isChecked ? addTag($tagIt, term) : removeTag($tagIt, term); - }); - - $("select.term-picker").change(function (e) { - var $select = $(this); - var $firstOption = $("option:first", $select); - - if ($firstOption.is(":selected")) - return; - - var $selecedOption = $("option:selected", $select); - var $wrapper = $select.parents("fieldset.taxonomy-wrapper:first"); - var $tagIt = $("ul.tagit", $wrapper); - var term = $selecedOption.text(); - var singleChoice = $(".terms-editor", $wrapper).data("singlechoice"); - - addTag($tagIt, term); - $select.val(""); - - if (!singleChoice) - $selecedOption.remove(); - }); - var renderAutocompleteItem = function (ul, item) { var label = item.label; @@ -148,20 +76,28 @@ /* Initialization **********************************************************************/ $(".terms-editor").each(function () { - var allTerms = $(this).data("all-terms"); var selectedTerms = $(this).data("selected-terms"); var $tagit = $("> ul", this).tagit({ - tagSource: allTerms, + tagSource: function (request, response) { + var termsEditor = $(this.element).parents(".terms-editor"); + $.getJSON("/api/taxonomies/tags", { taxonomyId: termsEditor.data("taxonomy-id"), leavesOnly: termsEditor.data("leaves-only"), query: request.term }, function (data, status, xhr) { + response(data); + }); + }, initialTags: selectedTerms, triggerKeys: ['enter', 'tab'], // default is ['enter', 'space', 'comma', 'tab'] but we remove comma and space to allow them in the terms allowNewTags: $(this).data("allow-new-terms"), tagsChanged: onTagsChanged, caseSensitive: false, - minLength: 0 - }).data("uiTagit"); + minLength: 0, + sortable: true + }).data("ui-tagit"); - $tagit.input.autocomplete().data("uiAutocomplete")._renderItem = renderAutocompleteItem; + $tagit.input.autocomplete().data("ui-autocomplete")._renderItem = renderAutocompleteItem; + $tagit.input.autocomplete().on("autocompletefocus", function (event, ui) { + event.preventDefault(); + }); }); diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/ITaxonomyService.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/ITaxonomyService.cs index ab31ba19b..fd3cd01a7 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/ITaxonomyService.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/ITaxonomyService.cs @@ -41,6 +41,7 @@ namespace Orchard.Taxonomies.Services { IEnumerable GetTerms(int taxonomyId); + int GetTermsCount(int taxonomyId); TermPart GetTerm(int id); TermPart GetTermByName(int taxonomyId, string name); void DeleteTerm(TermPart termPart); diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TaxonomyService.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TaxonomyService.cs index a6d6b14d0..296f41a2b 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TaxonomyService.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TaxonomyService.cs @@ -185,6 +185,12 @@ namespace Orchard.Taxonomies.Services { return TermPart.Sort(result); } + public int GetTermsCount(int taxonomyId) { + return _contentManager.Query() + .Where(x => x.TaxonomyId == taxonomyId) + .Count(); + } + public TermPart GetTerm(int id) { return _contentManager .Query() diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TermCountProcessor.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TermCountProcessor.cs index b10459ea8..3d9251b07 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TermCountProcessor.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Services/TermCountProcessor.cs @@ -1,17 +1,11 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using System.Web; using Orchard.ContentManagement; using Orchard.Taxonomies.Models; namespace Orchard.Taxonomies.Services { public class TermCountProcessor : ITermCountProcessor { - private readonly IContentManager _contentManager; private readonly ITaxonomyService _taxonomyService; - public TermCountProcessor(IContentManager contentManager, ITaxonomyService taxonomyService) { - _contentManager = contentManager; + public TermCountProcessor(ITaxonomyService taxonomyService) { _taxonomyService = taxonomyService; } @@ -31,4 +25,4 @@ namespace Orchard.Taxonomies.Services { } } } -} \ No newline at end of file +} diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Styles/admin-taxonomy-tags.css b/src/Orchard.Web/Modules/Orchard.Taxonomies/Styles/admin-taxonomy-tags.css index 48db0b876..115a4fffd 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Styles/admin-taxonomy-tags.css +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Styles/admin-taxonomy-tags.css @@ -2,6 +2,10 @@ opacity: 0.3; } +.ui-autocomplete .ui-menu-item { + float: none; +} + .ui-autocomplete .ui-menu-item a { font-size: 0.8em; } diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Tokens/TaxonomyTokens.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/Tokens/TaxonomyTokens.cs index 3b2a9edc2..e921232ed 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Tokens/TaxonomyTokens.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Tokens/TaxonomyTokens.cs @@ -31,21 +31,33 @@ namespace Orchard.Taxonomies.Tokens { context.For("TaxonomyField") .Token("Terms", field => String.Join(", ", field.Terms.Select(t => t.Name).ToArray())) - .Token( - token => { - var index = 0; - return (token.StartsWith("Terms:", StringComparison.OrdinalIgnoreCase) && int.TryParse(token.Substring("Terms:".Length), out index)) ? index.ToString() : null; - }, - (token, t) => { - var index = Convert.ToInt32(token); - return index + 1 > t.Terms.Count() ? null : t.Terms.ElementAt(index).Name; + .Token(FilterTokenParam, + (index, field) => { + var term = field.Terms.ElementAtOrDefault(Convert.ToInt32(index)); + return term != null ? term.Name : null; }) - // todo: extend Chain() in order to accept a filter like in Token() so that we can chain on an expression - .Chain("Terms:0", "Content", t => t.Terms.ElementAt(0)) - .Chain("Terms:1", "Content", t => t.Terms.ElementAt(1)) - .Chain("Terms:2", "Content", t => t.Terms.ElementAt(2)) - .Chain("Terms:3", "Content", t => t.Terms.ElementAt(3)) + .Chain(FilterChainParam, "Content", (index, field) => field.Terms.ElementAtOrDefault(Convert.ToInt32(index))) ; } + + private static string FilterTokenParam(string token) { + int index = 0; + return (token.StartsWith("Terms:", StringComparison.OrdinalIgnoreCase) && int.TryParse(token.Substring("Terms:".Length), out index)) ? index.ToString() : null; + } + + private static Tuple FilterChainParam(string token) { + int tokenLength = "Terms:".Length; + int index = 0; + int chainIndex = token.IndexOf('.'); + if (!token.StartsWith("Terms:", StringComparison.OrdinalIgnoreCase) || chainIndex <= tokenLength) + return null; + + if (int.TryParse(token.Substring(tokenLength, chainIndex - tokenLength), out index)) { + return new Tuple(index.ToString(), token.Substring(chainIndex + 1)); + } + else { + return null; + } + } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/ViewModels/Tag.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/ViewModels/Tag.cs new file mode 100644 index 000000000..e379d0653 --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/ViewModels/Tag.cs @@ -0,0 +1,17 @@ +using Newtonsoft.Json; + +namespace Orchard.Taxonomies.ViewModels { + public class Tag { + [JsonProperty("label")] + public string Label { get; set; } + + [JsonProperty("value")] + public int Value { get; set; } + + [JsonProperty("levels")] + public int Levels { get; set; } + + [JsonProperty("disabled")] + public bool Disabled { get; set; } + } +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/ViewModels/TaxonomyFieldViewModel.cs b/src/Orchard.Web/Modules/Orchard.Taxonomies/ViewModels/TaxonomyFieldViewModel.cs index 1eccb5eda..948227fe7 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/ViewModels/TaxonomyFieldViewModel.cs +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/ViewModels/TaxonomyFieldViewModel.cs @@ -1,4 +1,5 @@ using System.Collections.Generic; +using Orchard.Taxonomies.Models; using Orchard.Taxonomies.Settings; namespace Orchard.Taxonomies.ViewModels { @@ -8,6 +9,8 @@ namespace Orchard.Taxonomies.ViewModels { public string DisplayName { get; set; } public TaxonomyFieldSettings Settings { get; set; } public IList Terms { get; set; } + public IEnumerable SelectedTerms { get; set; } public int SingleTermId { get; set; } + public bool HasTerms { get; set; } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.Autocomplete.cshtml b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.Autocomplete.cshtml index 173b44a6e..11f5e0f1b 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.Autocomplete.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.Autocomplete.cshtml @@ -1,5 +1,6 @@ @model TaxonomyFieldViewModel @using Orchard.Taxonomies.Helpers; +@using Orchard.Taxonomies.Models @using Orchard.Utility.Extensions; @using Orchard.Taxonomies.ViewModels; @@ -18,60 +19,61 @@ Script.Include("~/Themes/TheAdmin/scripts/admin.js").AtFoot(); Script.Include("admin-taxonomy-expando.js").AtFoot(); - var termIndex = 0; } @functions { - bool IsTermDisabled(TermEntry term) { + bool IsTermDisabled(TermPart term) { return !term.Selectable || (Model.Settings.LeavesOnly && Model.Terms.Any(t => t.Path.Contains(term.Path + term.Id))); } } @{ - var allTerms = Newtonsoft.Json.JsonConvert.SerializeObject(Model.Terms.Select(x => new { label = x.Name, value = x.Id, levels = x.GetLevels(), disabled = IsTermDisabled(x) })); - var selectedTerms = Newtonsoft.Json.JsonConvert.SerializeObject(Model.Terms.Where(x => x.IsChecked).Select(x => new { label = x.Name, value = x.Id, levels = 0, disabled = true })); + var termIndex = 0; + var checkedTerms = Model.SelectedTerms.ToList(); + var selectedTerms = Newtonsoft.Json.JsonConvert.SerializeObject(checkedTerms.Select(x => new { label = x.Name, value = x.Id, levels = 0, disabled = true })); + }
    - class="required" }>@Model.DisplayName + @if (Model.Settings.Autocomplete) { -
    -
      - @if (Model.Settings.SingleChoice) { -
      @T("Enter a single term. Hit tab or enter to apply the term.") @if (!Model.Settings.AllowCustomTerms) { @T("This taxonomy does not allow you to create new terms.") }
      - } - else { -
      @T("Enter multiple terms. Hit tab, enter or , to add multiple terms.") @if (!Model.Settings.AllowCustomTerms) { @T("This taxonomy does not allow you to create new terms.") }
      - } -
      +
      +
        + @if (Model.Settings.SingleChoice) { +
        @T("Enter a single term. Hit tab or enter to apply the term.") @if (!Model.Settings.AllowCustomTerms) { @T("This taxonomy does not allow you to create new terms.") }
        + } + else { +
        @T("Enter multiple terms. Hit tab, enter or , to add multiple terms.") @if (!Model.Settings.AllowCustomTerms) { @T("This taxonomy does not allow you to create new terms.") }
        + } +
        } @if (!String.IsNullOrWhiteSpace(Model.Settings.Hint)) { - @Model.Settings.Hint + @Model.Settings.Hint }
          - @foreach (var entry in Model.Terms) { - var ti = termIndex; -
        • - @{ - var disabled = IsTermDisabled(entry); - if (Model.Settings.SingleChoice) { - disabled="disabled" } type="radio" value="@Model.Terms[ti].Id" @if (entry.Id == Model.SingleTermId) { checked="checked" } name="@Html.FieldNameFor(m => m.SingleTermId)" id="@Html.FieldIdFor(m => m.Terms[ti].IsChecked)" data-term="@entry.Name" data-term-identity="@entry.Name.ToLower()" /> + @foreach (var entry in checkedTerms) { + var ti = termIndex; +
        • + @{ + var disabled = IsTermDisabled(entry); + if (Model.Settings.SingleChoice) { + disabled="disabled" } type="radio" value="@entry.Id" @if (entry.Id == Model.SingleTermId){ checked="checked" } name="@Html.FieldNameFor(m => m.SingleTermId)" id="@Html.FieldIdFor(m => m.Terms[ti].IsChecked)" data-term="@entry.Name" data-term-identity="@entry.Name.ToLower()" /> + } + else { + disabled="disabled" } type="checkbox" value="true" checked="checked" name="@Html.FieldNameFor(m => m.Terms[ti].IsChecked)" id="@Html.FieldIdFor(m => m.Terms[ti].IsChecked)" data-term="@entry.Name" data-term-identity="@entry.Name.ToLower()" /> + } } - else { - disabled="disabled" } type="checkbox" value="true" @if (entry.IsChecked) { checked="checked" } name="@Html.FieldNameFor(m => m.Terms[ti].IsChecked)" id="@Html.FieldIdFor(m => m.Terms[ti].IsChecked)" data-term="@entry.Name" data-term-identity="@entry.Name.ToLower()" /> - } - } - - @Html.HiddenFor(m => m.Terms[ti].Id) -
        • - termIndex++; - } + + + + termIndex++; + }
        - @if (!Model.Terms.Any()) { -
        - @T("There are no terms defined for {0} yet.", Model.DisplayName) - @T("Create some terms") -
        + @if (!Model.HasTerms && AuthorizedFor(Orchard.Taxonomies.Permissions.CreateTerm)) { +
        + @T("There are no terms defined for {0} yet.", Model.DisplayName.CamelFriendly()) + @T("Create some terms") +
        } @Html.HiddenFor(m => m.TaxonomyId)
        diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.cshtml b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.cshtml index 4e5656b4d..376b7169c 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/EditorTemplates/Fields/TaxonomyField.cshtml @@ -42,7 +42,7 @@ } - @if (!Model.Terms.Any()) { + @if (!Model.Terms.Any() && AuthorizedFor(Orchard.Taxonomies.Permissions.CreateTerm)) {
        @T("There are no terms defined for {0} yet.", Model.DisplayName) @T("Create some terms") diff --git a/src/Orchard.Web/Modules/Orchard.Templates/Drivers/ShapePartDriver.cs b/src/Orchard.Web/Modules/Orchard.Templates/Drivers/ShapePartDriver.cs index 0cd70d87f..986760e0e 100644 --- a/src/Orchard.Web/Modules/Orchard.Templates/Drivers/ShapePartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Templates/Drivers/ShapePartDriver.cs @@ -58,6 +58,11 @@ namespace Orchard.Templates.Drivers { } protected override void Importing(ShapePart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + var shapeElement = context.Data.Element(part.PartDefinition.Name); if (shapeElement != null) diff --git a/src/Orchard.Web/Modules/Orchard.Templates/Module.txt b/src/Orchard.Web/Modules/Orchard.Templates/Module.txt index 31f31fbd3..04dd4a09c 100644 --- a/src/Orchard.Web/Modules/Orchard.Templates/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Templates/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides a Template type that can be used to store template code and used as a shape. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Templates/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Templates/Properties/AssemblyInfo.cs index 2e9f3d1ab..5f2092b87 100644 --- a/src/Orchard.Web/Modules/Orchard.Templates/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Templates/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Templates/Services/RazorTemplateProcessor.cs b/src/Orchard.Web/Modules/Orchard.Templates/Services/RazorTemplateProcessor.cs index d752ca2e8..2aed18c8d 100644 --- a/src/Orchard.Web/Modules/Orchard.Templates/Services/RazorTemplateProcessor.cs +++ b/src/Orchard.Web/Modules/Orchard.Templates/Services/RazorTemplateProcessor.cs @@ -2,6 +2,7 @@ using System.Collections.Generic; using System.IO; using System.Text; +using System.Web; using System.Web.Mvc; using System.Web.UI; using System.Web.WebPages; @@ -14,6 +15,7 @@ namespace Orchard.Templates.Services { [OrchardFeature("Orchard.Templates.Razor")] public class RazorTemplateProcessor : TemplateProcessorImpl { private readonly IRazorCompiler _compiler; + private readonly HttpContextBase _httpContextBase; private readonly IWorkContextAccessor _wca; public override string Type { @@ -22,9 +24,11 @@ namespace Orchard.Templates.Services { public RazorTemplateProcessor( IRazorCompiler compiler, + HttpContextBase httpContextBase, IWorkContextAccessor wca) { _compiler = compiler; + _httpContextBase = httpContextBase; _wca = wca; Logger = NullLogger.Instance; } @@ -37,7 +41,7 @@ namespace Orchard.Templates.Services { public override string Process(string template, string name, DisplayContext context = null, dynamic model = null) { if (String.IsNullOrEmpty(template)) - return String.Empty; + return string.Empty; var compiledTemplate = _compiler.CompileRazor(template, name, new Dictionary()); var result = ActivateAndRenderTemplate(compiledTemplate, context, null, model); @@ -69,10 +73,9 @@ namespace Orchard.Templates.Services { // Setup a fake view context in order to support razor syntax inside of HTML attributes, // for instance: Homepage. var viewData = new ViewDataDictionary(model); - var httpContext = _wca.GetContext().HttpContext; obj.ViewContext = new ViewContext( new ControllerContext( - httpContext.Request.RequestContext, + _httpContextBase.Request.RequestContext, new StubController()), new StubView(), viewData, @@ -80,7 +83,7 @@ namespace Orchard.Templates.Services { htmlWriter); obj.ViewData = viewData; - obj.WebPageContext = new WebPageContext(httpContext, obj as WebPageRenderingBase, model); + obj.WebPageContext = new WebPageContext(_httpContextBase, obj as WebPageRenderingBase, model); obj.WorkContext = _wca.GetContext(); } diff --git a/src/Orchard.Web/Modules/Orchard.Templates/Services/TemplateShapeBindingResolver.cs b/src/Orchard.Web/Modules/Orchard.Templates/Services/TemplateShapeBindingResolver.cs index 6d8e1743a..4fc5eb6a1 100644 --- a/src/Orchard.Web/Modules/Orchard.Templates/Services/TemplateShapeBindingResolver.cs +++ b/src/Orchard.Web/Modules/Orchard.Templates/Services/TemplateShapeBindingResolver.cs @@ -55,7 +55,7 @@ namespace Orchard.Templates.Services { } private IDictionary BuildShapeProcessors() { - return _cacheManager.Get("Template.ShapeProcessors", ctx => { + return _cacheManager.Get("Template.ShapeProcessors", true, ctx => { ctx.Monitor(_signals.When(DefaultTemplateService.TemplatesSignal)); // select all name of types which contains ShapePart diff --git a/src/Orchard.Web/Modules/Orchard.Themes/Module.txt b/src/Orchard.Web/Modules/Orchard.Themes/Module.txt index 0d11ff0e6..71eadea38 100644 --- a/src/Orchard.Web/Modules/Orchard.Themes/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Themes/Module.txt @@ -3,7 +3,7 @@ AntiForgery: enabled SessionState: required Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The themes module makes it possible for Orchard applications to customize the look and feel of an Orchard web site. FeatureDescription: Basic theming capability. @@ -11,4 +11,4 @@ Category: Core Features: Orchard.Themes: Description: Basic theming capability. - Category: Core \ No newline at end of file + Category: Core diff --git a/src/Orchard.Web/Modules/Orchard.Themes/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Themes/Properties/AssemblyInfo.cs index bf91a7235..ae631558f 100644 --- a/src/Orchard.Web/Modules/Orchard.Themes/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Themes/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/EvaluateFor.cs b/src/Orchard.Web/Modules/Orchard.Tokens/EvaluateFor.cs index 39bda24bc..70438592c 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/EvaluateFor.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/EvaluateFor.cs @@ -7,5 +7,6 @@ namespace Orchard.Tokens { public abstract EvaluateFor Chain(string token, string chainTarget, Func chainValue); public abstract EvaluateFor Token(Func tokenValue); public abstract EvaluateFor Token(Func filter, Func tokenValue); + public abstract EvaluateFor Chain(Func> filter, string chainTarget, Func chainValue); } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Implementation/TokenManager.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Implementation/TokenManager.cs index b87d6f353..c0a967914 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Implementation/TokenManager.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Implementation/TokenManager.cs @@ -153,6 +153,27 @@ namespace Orchard.Tokens.Implementation { } return this; } + + public override EvaluateFor Chain(Func> filter, string chainTarget, Func chainValue) { + var subTokens = _context.Tokens + .Where(kv => kv.Key.Contains('.')) + .Select(kv => { + var filterResult = filter(kv.Key); + return filterResult != null ? new Tuple(filterResult.Item1, filterResult.Item2, kv.Value) : null; + }) + .Where(st => st != null) + .ToList(); + if (!subTokens.Any()) { + return this; + } + foreach(var chainGroup in subTokens.GroupBy(st => st.Item1)) { + var subValues = _context._manager.Evaluate(chainTarget, chainGroup.ToDictionary(cg => cg.Item2, cg => cg.Item3), new Dictionary { { chainTarget, chainValue(chainGroup.Key, _data) } }); + foreach (var subValue in subValues) { + _context.Values[subValue.Key] = subValue.Value; + } + } + return this; + } } private class EvaluateForSilent : EvaluateFor { @@ -175,6 +196,10 @@ namespace Orchard.Tokens.Implementation { public override EvaluateFor Chain(string token, string chainTarget, Func chainValue) { return this; } + + public override EvaluateFor Chain(Func> filter, string chainTarget, Func chainValue) { + return this; + } } } diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Module.txt b/src/Orchard.Web/Modules/Orchard.Tokens/Module.txt index 52abb0e05..5d0ec1c24 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Module.txt @@ -2,7 +2,7 @@ Name: Tokens AntiForgery: enabled Author: The Orchard Team Website: http://orchardtokens.codeplex.com -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides a system for performing string replacements with common site values. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Properties/AssemblyInfo.cs index fb3ec5952..bda12f596 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Providers/ContentTokens.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Providers/ContentTokens.cs index 257394d88..6e6dad645 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Providers/ContentTokens.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Providers/ContentTokens.cs @@ -36,7 +36,7 @@ namespace Orchard.Tokens.Providers { .Token("DisplayUrl", T("Display Url"), T("Url to display the content."), "Url") .Token("EditUrl", T("Edit Url"), T("Url to edit the content."), "Url") .Token("Container", T("Container"), T("The container Content Item."), "Content") - .Token("Body", T("Body"), T("The body text of the content item."), "Content") + .Token("Body", T("Body"), T("The body text of the content item."), "Text") ; // Token descriptors for fields diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Providers/TextTokens.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Providers/TextTokens.cs index 89f77c018..255cee809 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Providers/TextTokens.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Providers/TextTokens.cs @@ -13,38 +13,45 @@ namespace Orchard.Tokens.Providers { public void Describe(DescribeContext context) { context.For("Text", T("Text"), T("Tokens for text strings")) .Token("Limit:*", T("Limit:[,]"), T("Limit text to specified length and append an optional ellipsis text.")) - .Token("Format:*", T("Format:"), T("Optional format specifier (e.g. foo{0}bar). See format strings at Standard Formats and Custom Formats"), "DateTime") - .Token("TrimEnd:*", T("TrimEnd:"), T("Trims the specified characters or number of them from the end of the string."), "Text") + .Token("Format:*", T("Format:"), T("Optional format specifier (e.g. foo{0}bar).")) + .Token("TrimEnd:*", T("TrimEnd:"), T("Trims the specified characters or number of them from the end of the string.")) .Token("UrlEncode", T("Url Encode"), T("Encodes a URL string."), "Text") .Token("HtmlEncode", T("Html Encode"), T("Encodes an HTML string."), "Text") - .Token("LineEncode", T("Line Encode"), T("Replaces new lines with
        tags.")) + .Token("JavaScriptEncode", T("JavaScript Encode"), T("Encodes a JavaScript string."), "Text") + .Token("LineEncode", T("Line Encode"), T("Replaces new lines with
        tags."), "Text") ; } public void Evaluate(EvaluateContext context) { context.For("Text", () => "") - .Token( // {Text} + // {Text} + .Token( token => token == String.Empty ? String.Empty : null, (token, d) => d.ToString()) - .Token( // {Text.Limit:[,]} - token => { - if (token.StartsWith("Limit:", StringComparison.OrdinalIgnoreCase)) { - var param = token.Substring("Limit:".Length); - return param; - } - return null; - }, + // {Text.Limit:[,]} + .Token( + token => FilterTokenParam("Limit:", token), (token, t) => Limit(t, token)) // {Text.Format:} .Token( - token => token.StartsWith("Format:", StringComparison.OrdinalIgnoreCase) ? token.Substring("Format:".Length) : null, - (token, d) => String.Format(d,token)) - .Token(token => token.StartsWith("TrimEnd:", StringComparison.OrdinalIgnoreCase) ? token.Substring("TrimEnd:".Length) : null, TrimEnd) + token => FilterTokenParam("Format:", token), + (token, d) => String.Format(d, token)) + // {Text.TrimEnd:} + .Token(token => FilterTokenParam("TrimEnd:", token), TrimEnd) .Token("UrlEncode", HttpUtility.UrlEncode) + .Chain("UrlEncode", "Text", HttpUtility.UrlEncode) .Token("HtmlEncode", HttpUtility.HtmlEncode) + .Chain("HtmlEncode", "Text", HttpUtility.HtmlEncode) + .Token("JavaScriptEncode", HttpUtility.JavaScriptStringEncode) + .Chain("JavaScriptEncode", "Text", HttpUtility.JavaScriptStringEncode) .Token("LineEncode", text => text.Replace(System.Environment.NewLine, "
        ")) + .Chain("LineEncode", "Text", text => text.Replace(System.Environment.NewLine, "
        ")) ; - + + } + + private static string FilterTokenParam(string tokenName, string token) { + return token.StartsWith(tokenName, StringComparison.OrdinalIgnoreCase) ? token.Substring(tokenName.Length) : null; } private static string TrimEnd(string token, string param) { diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/Properties/AssemblyInfo.cs index 2238161de..27fa698fb 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/Properties/AssemblyInfo.cs @@ -32,5 +32,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/StubWorkContextAccessor.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/StubWorkContextAccessor.cs index b68a02f27..9a957179d 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/StubWorkContextAccessor.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/StubWorkContextAccessor.cs @@ -131,10 +131,18 @@ namespace Orchard.Tokens.Tests { return _lifetimeScope.Resolve(); } + public override object Resolve(Type serviceType) { + return _lifetimeScope.Resolve(serviceType); + } + public override bool TryResolve(out T service) { return _lifetimeScope.TryResolve(out service); } + public override bool TryResolve(Type serviceType, out object service) { + return _lifetimeScope.TryResolve(serviceType, out service); + } + public override T GetState(string name) { return (T)_contextDictonary[name]; } diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TestTokenProvider.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TestTokenProvider.cs index 105f705d4..c65b76a65 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TestTokenProvider.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TestTokenProvider.cs @@ -1,4 +1,5 @@ using System; +using System.Linq; using Orchard.ContentManagement; using Orchard.Localization; using Orchard.Security; @@ -24,6 +25,9 @@ namespace Orchard.Tokens.Tests { context.For("Date") .Token("Now", T("Now"), T("Current system date in short date format. You can chain a .NET DateTime format string to customize.")); + + context.For("Users") + .Token("Users[*:]", T("Users"), T("A user by its username"), "User"); } public void Evaluate(EvaluateContext context) { @@ -36,6 +40,7 @@ namespace Orchard.Tokens.Tests { context.For("User", () => new TestUser { UserName = "CurrentUser" }) .Token("Name", u => u.UserName) + .Token("Email", u => u.Email) .Token("Birthdate", u => "Nov 15") .Chain("Birthdate", "DateTime", u => new DateTime(1978, 11, 15)); @@ -45,6 +50,29 @@ namespace Orchard.Tokens.Tests { context.For("DateTime") .Token((token, value) => value.ToString(token)); + + context.For("Users", () => new TestUser[] { + new TestUser { UserName = "User1", Email = "user1@test.com" }, + new TestUser { UserName = "User2", Email = "user2@test.com" }, + new TestUser { UserName = "User3", Email = "user3@test.com" } + }) + .Token( + (token) => token.StartsWith("User:", StringComparison.OrdinalIgnoreCase) ? token.Substring("User:".Length) : null, + (userName, users) => users.Where(u => u.UserName == userName).Select(u => u.UserName).FirstOrDefault() + ) + .Chain( + (token) => { + int tokenLength = "User:".Length; + int chainIndex = token.IndexOf('.'); + if (token.StartsWith("User:", StringComparison.OrdinalIgnoreCase) && chainIndex > tokenLength) + return new Tuple(token.Substring(tokenLength, chainIndex - tokenLength), token.Substring(chainIndex + 1)); + else + return null; + }, + "User", + (userName, users) => users.Where(u => u.UserName == userName).FirstOrDefault() + ); + } } diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenManagerTests.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenManagerTests.cs index 341e5ed37..073442bb9 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenManagerTests.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenManagerTests.cs @@ -29,10 +29,11 @@ namespace Orchard.Tokens.Tests { [Test] public void TestDescribe() { var allTokens = _tokenManager.Describe(null); - Assert.That(allTokens.Count(), Is.EqualTo(3)); + Assert.That(allTokens.Count(), Is.EqualTo(4)); Assert.That(allTokens.Any(d => d.Target == "Site")); Assert.That(allTokens.Any(d => d.Target == "User")); Assert.That(allTokens.Any(d => d.Target == "Date")); + Assert.That(allTokens.Any(d => d.Target == "Users")); var tokens = allTokens.Single(d => d.Target == "Site").Tokens; Assert.That(string.Join(",", tokens.Select(td => td.Target)), Is.EqualTo("Site,Site,Site,Site")); @@ -59,7 +60,7 @@ namespace Orchard.Tokens.Tests { [Test] public void TestDescribeFilter() { var tokenDescriptors = _tokenManager.Describe(null); - Assert.That(tokenDescriptors.Count(), Is.EqualTo(3)); + Assert.That(tokenDescriptors.Count(), Is.EqualTo(4)); tokenDescriptors = _tokenManager.Describe(new[] { "Site" }); Assert.That(tokenDescriptors.Count(), Is.EqualTo(1)); Assert.That(tokenDescriptors.First().Target, Is.EqualTo("Site")); diff --git a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenizerTests.cs b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenizerTests.cs index 999c3d87b..8c976f8d0 100644 --- a/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenizerTests.cs +++ b/src/Orchard.Web/Modules/Orchard.Tokens/Tests/TokenizerTests.cs @@ -40,6 +40,18 @@ namespace Orchard.Tokens.Tests { Assert.That(_tokenizer.Replace("{Site.CurrentUser.Birthdate.yyyy}", null), Is.EqualTo("1978")); } + [Test] + public void TestParameterizedTokens() { + Assert.That(_tokenizer.Replace("{Users.User:User2}", null), Is.EqualTo("User2")); + Assert.That(_tokenizer.Replace("{Users.User:FakeUser}", null), Is.EqualTo("")); + } + + [Test] + public void TestParameterizedChainedTokens() { + Assert.That(_tokenizer.Replace("{Users.User:User2.Email}", null), Is.EqualTo("user2@test.com")); + Assert.That(_tokenizer.Replace("{Users.User:FakeUser.Email}", null), Is.EqualTo("")); + } + [Test] public void TestMissingTokens() { Assert.That(_tokenizer.Replace("[{Site.NotAToken}]", null), Is.EqualTo("[]")); diff --git a/src/Orchard.Web/Modules/Orchard.Users/Drivers/UserPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Users/Drivers/UserPartDriver.cs index f9a4526d6..1990a03d9 100644 --- a/src/Orchard.Web/Modules/Orchard.Users/Drivers/UserPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Users/Drivers/UserPartDriver.cs @@ -11,6 +11,11 @@ namespace Orchard.Users.Drivers { public class UserPartDriver : ContentPartDriver { protected override void Importing(UserPart part, ContentManagement.Handlers.ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + part.Email = context.Attribute(part.PartDefinition.Name, "Email"); part.EmailChallengeToken = context.Attribute(part.PartDefinition.Name, "EmailChallengeToken"); part.EmailStatus = (UserStatus)Enum.Parse(typeof(UserStatus), context.Attribute(part.PartDefinition.Name, "EmailStatus")); diff --git a/src/Orchard.Web/Modules/Orchard.Users/Module.txt b/src/Orchard.Web/Modules/Orchard.Users/Module.txt index 3abf74381..259aecc33 100644 --- a/src/Orchard.Web/Modules/Orchard.Users/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Users/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: The users module enables user management. Features: diff --git a/src/Orchard.Web/Modules/Orchard.Users/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Users/Properties/AssemblyInfo.cs index 80da788cb..95948dc3a 100644 --- a/src/Orchard.Web/Modules/Orchard.Users/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Users/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Warmup/Module.txt b/src/Orchard.Web/Modules/Orchard.Warmup/Module.txt index 4a59ab9e3..661208a95 100644 --- a/src/Orchard.Web/Modules/Orchard.Warmup/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Warmup/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: Provides a mecanism to generate a static version of pages for being used during application warm up. FeatureDescription: Generates the static version of specific pages periodically. diff --git a/src/Orchard.Web/Modules/Orchard.Warmup/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Warmup/Properties/AssemblyInfo.cs index 31aeabb61..1bba58d73 100644 --- a/src/Orchard.Web/Modules/Orchard.Warmup/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Warmup/Properties/AssemblyInfo.cs @@ -34,6 +34,6 @@ using System.Security; // CHANGE ON THIS VERSION NEEDS TO BE APPLIED ON WEB.CONFIG TOO -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Widgets/Controllers/AdminController.cs b/src/Orchard.Web/Modules/Orchard.Widgets/Controllers/AdminController.cs index c5ad05276..9431da0bc 100644 --- a/src/Orchard.Web/Modules/Orchard.Widgets/Controllers/AdminController.cs +++ b/src/Orchard.Web/Modules/Orchard.Widgets/Controllers/AdminController.cs @@ -69,7 +69,8 @@ namespace Orchard.Widgets.Controllers { } LayerPart currentLayer = layerId == null - ? layers.FirstOrDefault(layer => layer.Name == "Default") ?? layers.FirstOrDefault() + // look for the "Default" layer, or take the first if it doesn't exist + ? layers.FirstOrDefault(x => x.Name == "Default") ?? layers.FirstOrDefault() : layers.FirstOrDefault(layer => layer.Id == layerId); if (currentLayer == null && layerId != null) { // Incorrect layer id passed diff --git a/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/LayerPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/LayerPartDriver.cs index 09aea2464..202ea2692 100644 --- a/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/LayerPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/LayerPartDriver.cs @@ -64,6 +64,11 @@ namespace Orchard.Widgets.Drivers { } protected override void Importing(LayerPart part, ImportContentContext context) { + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; + } + var name = context.Attribute(part.PartDefinition.Name, "Name"); if (name != null) { part.Name = name; diff --git a/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/WidgetPartDriver.cs b/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/WidgetPartDriver.cs index ccaaa0c44..098944941 100644 --- a/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/WidgetPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.Widgets/Drivers/WidgetPartDriver.cs @@ -75,30 +75,25 @@ namespace Orchard.Widgets.Drivers { } protected override void Importing(WidgetPart part, ContentManagement.Handlers.ImportContentContext context) { - var title = context.Attribute(part.PartDefinition.Name, "Title"); - if (title != null) { - part.Title = title; + // Don't do anything if the tag is not specified. + if (context.Data.Element(part.PartDefinition.Name) == null) { + return; } - var position = context.Attribute(part.PartDefinition.Name, "Position"); - if (position != null) { - part.Position = position; - } - - var zone = context.Attribute(part.PartDefinition.Name, "Zone"); - if (zone != null) { - part.Zone = zone; - } - - var renderTitle = context.Attribute(part.PartDefinition.Name, "RenderTitle"); - if (!string.IsNullOrWhiteSpace(renderTitle)) { - part.RenderTitle = Convert.ToBoolean(renderTitle); - } - - var name = context.Attribute(part.PartDefinition.Name, "Name"); - if (name != null) { - part.Name = name; - } + context.ImportAttribute(part.PartDefinition.Name, "Title", x => part.Title = x); + context.ImportAttribute(part.PartDefinition.Name, "Position", x => part.Position = x); + context.ImportAttribute(part.PartDefinition.Name, "Zone", x => part.Zone = x); + context.ImportAttribute(part.PartDefinition.Name, "RenderTitle", x => { + if (!string.IsNullOrWhiteSpace(x)) { + part.RenderTitle = Convert.ToBoolean(x); + } + }); + context.ImportAttribute(part.PartDefinition.Name, "Name", x => part.Name = x); + context.ImportAttribute(part.PartDefinition.Name, "Title", x => { + if (!String.IsNullOrWhiteSpace(x)) + part.Title = x; + }); + context.ImportAttribute(part.PartDefinition.Name, "CssClasses", x => part.CssClasses = x); } protected override void Exporting(WidgetPart part, ContentManagement.Handlers.ExportContentContext context) { @@ -107,6 +102,7 @@ namespace Orchard.Widgets.Drivers { context.Element(part.PartDefinition.Name).SetAttributeValue("Zone", part.Zone); context.Element(part.PartDefinition.Name).SetAttributeValue("RenderTitle", part.RenderTitle); context.Element(part.PartDefinition.Name).SetAttributeValue("Name", part.Name); + context.Element(part.PartDefinition.Name).SetAttributeValue("CssClasses", part.CssClasses); } } -} \ No newline at end of file +} diff --git a/src/Orchard.Web/Modules/Orchard.Widgets/Module.txt b/src/Orchard.Web/Modules/Orchard.Widgets/Module.txt index f8e587adc..1a0eb5338 100644 --- a/src/Orchard.Web/Modules/Orchard.Widgets/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Widgets/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Description: An implementation of widgets for Orchard. FeatureDescription: An implementation of widgets. diff --git a/src/Orchard.Web/Modules/Orchard.Widgets/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Widgets/Properties/AssemblyInfo.cs index f75595450..bb6e1188a 100644 --- a/src/Orchard.Web/Modules/Orchard.Widgets/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Widgets/Properties/AssemblyInfo.cs @@ -30,6 +30,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Workflows/Controllers/AdminController.cs b/src/Orchard.Web/Modules/Orchard.Workflows/Controllers/AdminController.cs index 12c8bb6f8..af02427bf 100644 --- a/src/Orchard.Web/Modules/Orchard.Workflows/Controllers/AdminController.cs +++ b/src/Orchard.Web/Modules/Orchard.Workflows/Controllers/AdminController.cs @@ -23,6 +23,7 @@ using Orchard.UI.Notify; using Orchard.Workflows.Models; using Orchard.Workflows.Services; using Orchard.Workflows.ViewModels; +using Orchard.Workflows.Helpers; namespace Orchard.Workflows.Controllers { [ValidateInput(false)] @@ -298,7 +299,7 @@ namespace Orchard.Workflows.Controllers { dynamic activity = new JObject(); activity.Name = x.Name; activity.Id = x.Id; - activity.ClientId = x.Name + "_" + x.Id; + activity.ClientId = x.GetClientId(); activity.Left = x.X; activity.Top = x.Y; activity.Start = x.Start; @@ -323,7 +324,7 @@ namespace Orchard.Workflows.Controllers { [HttpPost, ActionName("Edit")] [FormValueRequired("submit.Save")] - public ActionResult EditPost(int id, string localId, string data) { + public ActionResult EditPost(int id, string localId, string data, bool clearWorkflows) { if (!Services.Authorizer.Authorize(StandardPermissions.SiteOwner, T("Not authorized to edit workflows"))) return new HttpUnauthorizedResult(); @@ -339,7 +340,6 @@ namespace Orchard.Workflows.Controllers { var activitiesIndex = new Dictionary(); workflowDefinitionRecord.ActivityRecords.Clear(); - workflowDefinitionRecord.WorkflowRecords.Clear(); foreach (var activity in state.Activities) { ActivityRecord activityRecord; @@ -367,9 +367,32 @@ namespace Orchard.Workflows.Controllers { }); } + if (clearWorkflows) { + workflowDefinitionRecord.WorkflowRecords.Clear(); + } + else { + foreach (var workflowRecord in workflowDefinitionRecord.WorkflowRecords) { + // Update any awaiting activity records with the new activity record. + foreach (var awaitingActivityRecord in workflowRecord.AwaitingActivities) { + var clientId = awaitingActivityRecord.ActivityRecord.GetClientId(); + if (activitiesIndex.ContainsKey(clientId)) { + awaitingActivityRecord.ActivityRecord = activitiesIndex[clientId]; + } + else { + workflowRecord.AwaitingActivities.Remove(awaitingActivityRecord); + } + } + // Remove any workflows with no awaiting activities. + if (!workflowRecord.AwaitingActivities.Any()) { + workflowDefinitionRecord.WorkflowRecords.Remove(workflowRecord); + } + } + } + Services.Notifier.Information(T("Workflow saved successfully")); - return RedirectToAction("Edit", new { id, localId }); + // Don't pass the localId to force the activites to refresh and use the deterministic clientId. + return RedirectToAction("Edit", new { id }); } [HttpPost, ActionName("Edit")] diff --git a/src/Orchard.Web/Modules/Orchard.Workflows/Models/ActivityRecord.cs b/src/Orchard.Web/Modules/Orchard.Workflows/Models/ActivityRecord.cs index 8c45e5a19..9233d5bcd 100644 --- a/src/Orchard.Web/Modules/Orchard.Workflows/Models/ActivityRecord.cs +++ b/src/Orchard.Web/Modules/Orchard.Workflows/Models/ActivityRecord.cs @@ -38,5 +38,14 @@ namespace Orchard.Workflows.Models { /// containing this activity. ///
        public virtual WorkflowDefinitionRecord WorkflowDefinitionRecord { get; set; } + + + /// + /// Gets the Id which can be used on the client. + /// + /// An unique Id to represent this activity on the client. + public string GetClientId() { + return Name + "_" + Id; + } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Workflows/Module.txt b/src/Orchard.Web/Modules/Orchard.Workflows/Module.txt index 83a7e0640..62fbcc260 100644 --- a/src/Orchard.Web/Modules/Orchard.Workflows/Module.txt +++ b/src/Orchard.Web/Modules/Orchard.Workflows/Module.txt @@ -2,7 +2,7 @@ AntiForgery: enabled Author: The Orchard Team Website: http://orchardproject.net -Version: 1.9.1 +Version: 1.9.2 OrchardVersion: 1.9 Category: Workflows Description: Provides tools to create custom workflows. diff --git a/src/Orchard.Web/Modules/Orchard.Workflows/Properties/AssemblyInfo.cs b/src/Orchard.Web/Modules/Orchard.Workflows/Properties/AssemblyInfo.cs index 1e36590ba..31c882f5f 100644 --- a/src/Orchard.Web/Modules/Orchard.Workflows/Properties/AssemblyInfo.cs +++ b/src/Orchard.Web/Modules/Orchard.Workflows/Properties/AssemblyInfo.cs @@ -31,6 +31,6 @@ using System.Security; // // You can specify all the values or you can default the Revision and Build Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard.Web/Modules/Orchard.Workflows/Views/Admin/Edit.cshtml b/src/Orchard.Web/Modules/Orchard.Workflows/Views/Admin/Edit.cshtml index c1acd41b5..1ab7a4294 100644 --- a/src/Orchard.Web/Modules/Orchard.Workflows/Views/Admin/Edit.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Workflows/Views/Admin/Edit.cshtml @@ -7,6 +7,7 @@ Style.Require("WorkflowsAdmin"); Style.Require("WorkflowsActivities"); Style.Require("jQueryUI_Orchard"); + Script.Require("jQueryUI_Dialog").AtFoot(); Script.Require("jsPlumb").AtFoot(); Script.Include("orchard-workflows-serialize.js").AtFoot(); Script.Include("orchard-workflows.js").AtFoot(); @@ -74,23 +75,41 @@ @Html.Hidden("data", String.Empty) @Html.Hidden("confirm-delete-activity", T("Are you sure you want to remove this activity?")) -@Html.Hidden("confirm-delete-instances", T("Are you sure you want to remove running instances of this workflow?")) +@Html.Hidden("confirm-delete-instances", T("You have running instances of this workflow, do you want to stop them?")) using (Script.Foot()) { } \ No newline at end of file diff --git a/src/Orchard.Web/Themes/TheThemeMachine/Theme.txt b/src/Orchard.Web/Themes/TheThemeMachine/Theme.txt index 938ba92ec..b70d501be 100644 --- a/src/Orchard.Web/Themes/TheThemeMachine/Theme.txt +++ b/src/Orchard.Web/Themes/TheThemeMachine/Theme.txt @@ -1,7 +1,7 @@ Name: The Theme Machine Author: jowall, mibach, loudej, heskew Description: Orchard Theme Machine is a flexible multi-zone theme that provides a solid foundation to build your site. It features 20 collapsible widget zones and is flexible enough to cover a wide range of layouts. -Version: 1.9.1 +Version: 1.9.2 Tags: Awesome Website: http://orchardproject.net Zones: Header, Navigation, Featured, BeforeMain, AsideFirst, Messages, BeforeContent, Content, AfterContent, AsideSecond, AfterMain, TripelFirst, TripelSecond, TripelThird, FooterQuadFirst, FooterQuadSecond, FooterQuadThird, FooterQuadFourth, Footer diff --git a/src/Orchard.Web/Web.config b/src/Orchard.Web/Web.config index dcc846ed7..ea98388f4 100644 --- a/src/Orchard.Web/Web.config +++ b/src/Orchard.Web/Web.config @@ -1,219 +1,219 @@ - - -
        -
        - - + + +
        +
        + + - - - - - - + + + + + + - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + - - - + + + - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Orchard.sln b/src/Orchard.sln index 40a51d6f3..ace52701a 100644 --- a/src/Orchard.sln +++ b/src/Orchard.sln @@ -255,14 +255,14 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Orchard.Layouts", "Orchard. EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Orchard.DynamicForms", "Orchard.Web\Modules\Orchard.DynamicForms\Orchard.DynamicForms.csproj", "{82190F52-2901-46D6-8A4C-34649959483F}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Orchard.Dashboards", "Orchard.Web\Modules\Orchard.Dashboards\Orchard.Dashboards.csproj", "{BAC82DB9-F4C4-4DD1-ABDB-F70E6229E6B0}" +EndProject Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{DF3909B0-1DDD-4D8A-9919-56FC438E25E2}" ProjectSection(SolutionItems) = preProject Rebracer.xml = Rebracer.xml WebEssentials-Settings.json = WebEssentials-Settings.json EndProjectSection EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Orchard.Dashboards", "Orchard.Web\Modules\Orchard.Dashboards\Orchard.Dashboards.csproj", "{BAC82DB9-F4C4-4DD1-ABDB-F70E6229E6B0}" -EndProject Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Gulp", "Gulp", "{90EBEE36-B5CD-42A8-A21B-76270E2C5D24}" ProjectSection(SolutionItems) = preProject Gulpfile.js = Gulpfile.js diff --git a/src/Orchard/BackgroundHttpContextFactory.cs b/src/Orchard/BackgroundHttpContextFactory.cs new file mode 100644 index 000000000..9935e7dfc --- /dev/null +++ b/src/Orchard/BackgroundHttpContextFactory.cs @@ -0,0 +1,33 @@ +using System.IO; +using System.Web; +using Orchard.Settings; + +namespace Orchard { + /// + /// A factory class that creates an HttpContext instance and initializes the HttpContext.Current property with that instance. + /// This is useful when rendering views from a background thread, as some Html Helpers access HttpContext.Current directly, thus preventing a NullReferenceException. + /// + public class BackgroundHttpContextFactory : IBackgroundHttpContextFactory { + public const string IsBackgroundHttpContextKey = "IsBackgroundHttpContext"; + private readonly ISiteService _siteService; + public BackgroundHttpContextFactory(ISiteService siteService) { + _siteService = siteService; + } + + public HttpContext CreateHttpContext() { + var url = _siteService.GetSiteSettings().BaseUrl; + var httpContext = new HttpContext(new HttpRequest("", url, ""), new HttpResponse(new StringWriter())); + + httpContext.Items[IsBackgroundHttpContextKey] = true; + + return httpContext; + } + + public void InitializeHttpContext() { + if (HttpContext.Current != null) + return; + + HttpContext.Current = CreateHttpContext(); + } + } +} \ No newline at end of file diff --git a/src/Orchard/Caching/DefaultAsyncTokenProvider.cs b/src/Orchard/Caching/DefaultAsyncTokenProvider.cs index b6769ec82..9f13571c0 100644 --- a/src/Orchard/Caching/DefaultAsyncTokenProvider.cs +++ b/src/Orchard/Caching/DefaultAsyncTokenProvider.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using System.Linq; using System.Threading; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Caching { public class DefaultAsyncTokenProvider : IAsyncTokenProvider { @@ -37,9 +38,12 @@ namespace Orchard.Caching { try { _task(token => _taskTokens.Add(token)); } - catch (Exception e) { - Logger.Error(e, "Error while monitoring extension files. Assuming extensions are not current."); - _taskException = e; + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error(ex, "Error while monitoring extension files. Assuming extensions are not current."); + _taskException = ex; } finally { _isTaskFinished = true; diff --git a/src/Orchard/Caching/ICacheManager.cs b/src/Orchard/Caching/ICacheManager.cs index c54902eb9..793b29a46 100644 --- a/src/Orchard/Caching/ICacheManager.cs +++ b/src/Orchard/Caching/ICacheManager.cs @@ -5,4 +5,17 @@ namespace Orchard.Caching { TResult Get(TKey key, Func, TResult> acquire); ICache GetCache(); } + + public static class CacheManagerExtensions { + public static TResult Get(this ICacheManager cacheManager, TKey key, bool preventConcurrentCalls, Func, TResult> acquire) { + if (preventConcurrentCalls) { + lock(key) { + return cacheManager.Get(key, acquire); + } + } + else { + return cacheManager.Get(key, acquire); + } + } + } } diff --git a/src/Orchard/Commands/CommandHostAgent.cs b/src/Orchard/Commands/CommandHostAgent.cs index 6c7a955aa..05b1122ef 100644 --- a/src/Orchard/Commands/CommandHostAgent.cs +++ b/src/Orchard/Commands/CommandHostAgent.cs @@ -15,6 +15,7 @@ using Orchard.FileSystems.VirtualPath; using Orchard.Localization; using Orchard.Logging; using Orchard.Tasks; +using Orchard.Exceptions; namespace Orchard.Commands { @@ -94,19 +95,21 @@ namespace Orchard.Commands { return CommandReturnCodes.Ok; } - catch (OrchardCommandHostRetryException e) { + catch (OrchardCommandHostRetryException ex) { // Special "Retry" return code for our host - output.WriteLine(T("{0} (Retrying...)", e.Message)); + output.WriteLine(T("{0} (Retrying...)", ex.Message)); return CommandReturnCodes.Retry; } - catch (Exception e) { - if (e is TargetInvocationException && - e.InnerException != null) { - // If this is an exception coming from reflection and there is an innerexception which is the actual one, redirect - e = e.InnerException; + catch (Exception ex) { + if (ex.IsFatal()) { + throw; } - - OutputException(output, T("Error executing command \"{0}\"", string.Join(" ", args)), e); + if (ex is TargetInvocationException && + ex.InnerException != null) { + // If this is an exception coming from reflection and there is an innerexception which is the actual one, redirect + ex = ex.InnerException; + } + OutputException(output, T("Error executing command \"{0}\"", string.Join(" ", args)), ex); return CommandReturnCodes.Fail; } } @@ -116,13 +119,16 @@ namespace Orchard.Commands { _hostContainer = CreateHostContainer(); return CommandReturnCodes.Ok; } - catch (OrchardCommandHostRetryException e) { + catch (OrchardCommandHostRetryException ex) { // Special "Retry" return code for our host - output.WriteLine(T("{0} (Retrying...)", e.Message)); + output.WriteLine(T("{0} (Retrying...)", ex.Message)); return CommandReturnCodes.Retry; } - catch (Exception e) { - OutputException(output, T("Error starting up Orchard command line host"), e); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + OutputException(output, T("Error starting up Orchard command line host"), ex); return CommandReturnCodes.Fail; } } @@ -135,8 +141,11 @@ namespace Orchard.Commands { } return CommandReturnCodes.Ok; } - catch (Exception e) { - OutputException(output, T("Error shutting down Orchard command line host"), e); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + OutputException(output, T("Error shutting down Orchard command line host"), ex); return CommandReturnCodes.Fail; } } @@ -184,13 +193,12 @@ namespace Orchard.Commands { [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Reliability", "CA2000:Dispose objects before losing scope")] private IContainer CreateHostContainer() { var hostContainer = OrchardStarter.CreateHostContainer(ContainerRegistrations); - var host = hostContainer.Resolve(); + host.Initialize(); return hostContainer; } - private IWorkContextScope CreateStandaloneEnvironment(string tenant) { var host = _hostContainer.Resolve(); var tenantManager = _hostContainer.Resolve(); @@ -207,13 +215,12 @@ namespace Orchard.Commands { return env; } else { - // In case of an unitiliazed site (no default settings yet), we create a default settings instance. + // In case of an uninitialized site (no default settings yet), we create a default settings instance. var settings = new ShellSettings { Name = ShellSettings.DefaultName, State = TenantState.Uninitialized }; return host.CreateStandaloneEnvironment(settings); } } - protected void ContainerRegistrations(ContainerBuilder builder) { MvcSingletons(builder); @@ -225,14 +232,14 @@ namespace Orchard.Commands { private CommandHostShellContainerRegistrations CreateShellRegistrations() { return new CommandHostShellContainerRegistrations { Registrations = shellBuilder => { - shellBuilder.RegisterType() - .As() - .As() - .InstancePerMatchingLifetimeScope("shell"); - shellBuilder.RegisterType() - .As() - .InstancePerLifetimeScope(); - } + shellBuilder.RegisterType() + .As() + .As() + .InstancePerMatchingLifetimeScope("shell"); + shellBuilder.RegisterType() + .As() + .InstancePerLifetimeScope(); + } }; } diff --git a/src/Orchard/Commands/DefaultOrchardCommandHandler.cs b/src/Orchard/Commands/DefaultOrchardCommandHandler.cs index a5e8a3d89..7936ddef1 100644 --- a/src/Orchard/Commands/DefaultOrchardCommandHandler.cs +++ b/src/Orchard/Commands/DefaultOrchardCommandHandler.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using System.Linq; using System.Reflection; using Orchard.Localization; +using Orchard.Exceptions; namespace Orchard.Commands { public abstract class DefaultOrchardCommandHandler : ICommandHandler { @@ -41,12 +42,15 @@ namespace Orchard.Commands { object value = Convert.ChangeType(commandSwitch.Value, propertyInfo.PropertyType); propertyInfo.SetValue(this, value, null/*index*/); } - catch(Exception e) { + catch(Exception ex) { + if (ex.IsFatal()) { + throw; + } string message = T("Error converting value \"{0}\" to \"{1}\" for switch \"{2}\"", LocalizedString.TextOrDefault(commandSwitch.Value, T("(empty)")), propertyInfo.PropertyType.FullName, commandSwitch.Key).Text; - throw new InvalidOperationException(message, e); + throw new InvalidOperationException(message, ex); } } diff --git a/src/Orchard/ContentManagement/ContentPart.cs b/src/Orchard/ContentManagement/ContentPart.cs index cdb4bbfbe..16ed02311 100644 --- a/src/Orchard/ContentManagement/ContentPart.cs +++ b/src/Orchard/ContentManagement/ContentPart.cs @@ -21,14 +21,6 @@ namespace Orchard.ContentManagement { public virtual ContentItem ContentItem { get; set; } - //interesting thought, should/could parts also have zones (would then have zones on the page, content item and parts...)? - private readonly IZoneCollection _zones = new ZoneCollection(); - public virtual IZoneCollection Zones { - get { - return _zones; - } - } - /// /// The ContentItem's identifier. /// diff --git a/src/Orchard/ContentManagement/DefaultContentManager.cs b/src/Orchard/ContentManagement/DefaultContentManager.cs index 048ea4dc0..fc6fd7de6 100644 --- a/src/Orchard/ContentManagement/DefaultContentManager.cs +++ b/src/Orchard/ContentManagement/DefaultContentManager.cs @@ -817,7 +817,7 @@ namespace Orchard.ContentManagement { } private ContentTypeRecord AcquireContentTypeRecord(string contentType) { - var contentTypeId = _cacheManager.Get(contentType + "_Record", ctx => { + var contentTypeId = _cacheManager.Get(contentType + "_Record", true, ctx => { ctx.Monitor(_signals.When(contentType + "_Record")); var contentTypeRecord = _contentTypeRepository.Get(x => x.Name == contentType); diff --git a/src/Orchard/ContentManagement/DefaultContentQuery.cs b/src/Orchard/ContentManagement/DefaultContentQuery.cs index 468d105ef..befcd03aa 100644 --- a/src/Orchard/ContentManagement/DefaultContentQuery.cs +++ b/src/Orchard/ContentManagement/DefaultContentQuery.cs @@ -76,7 +76,7 @@ namespace Orchard.ContentManagement { } private int GetContentTypeRecordId(string contentType) { - return _cacheManager.Get(contentType + "_Record", ctx => { + return _cacheManager.Get(contentType + "_Record", true, ctx => { ctx.Monitor(_signals.When(contentType + "_Record")); var contentTypeRecord = _contentTypeRepository.Get(x => x.Name == contentType); diff --git a/src/Orchard/Data/Migration/AutomaticDataMigrations.cs b/src/Orchard/Data/Migration/AutomaticDataMigrations.cs index 50727d47f..797c77b80 100644 --- a/src/Orchard/Data/Migration/AutomaticDataMigrations.cs +++ b/src/Orchard/Data/Migration/AutomaticDataMigrations.cs @@ -1,9 +1,13 @@ using System; using System.Linq; +using Orchard.Data.Migration.Interpreters; +using Orchard.Data.Migration.Schema; using Orchard.Environment; +using Orchard.Environment.Configuration; using Orchard.Environment.Features; using Orchard.Logging; using Orchard.Tasks.Locking.Services; +using Orchard.Exceptions; namespace Orchard.Data.Migration { /// @@ -13,24 +17,35 @@ namespace Orchard.Data.Migration { private readonly IDataMigrationManager _dataMigrationManager; private readonly IFeatureManager _featureManager; private readonly IDistributedLockService _distributedLockService; + private readonly IDataMigrationInterpreter _dataMigrationInterpreter; + private readonly ShellSettings _shellSettings; + private readonly ITransactionManager _transactionManager; public AutomaticDataMigrations( IDataMigrationManager dataMigrationManager, + IDataMigrationInterpreter dataMigrationInterpreter, IFeatureManager featureManager, - IDistributedLockService distributedLockService) { + IDistributedLockService distributedLockService, + ITransactionManager transactionManager, + ShellSettings shellSettings) { _dataMigrationManager = dataMigrationManager; _featureManager = featureManager; _distributedLockService = distributedLockService; + _shellSettings = shellSettings; + _transactionManager = transactionManager; + _dataMigrationInterpreter = dataMigrationInterpreter; Logger = NullLogger.Instance; } - public ILogger Logger { get; set; } + public ILogger Logger { get; set; } public void Activated() { - DistributedLock @lock; - if(_distributedLockService.TryAcquireLockForThread(GetType().FullName, TimeSpan.FromMinutes(30), TimeSpan.FromMilliseconds(250), out @lock)) { + EnsureDistributedLockSchemaExists(); + + IDistributedLock @lock; + if (_distributedLockService.TryAcquireLock(GetType().FullName, TimeSpan.FromMinutes(30), TimeSpan.FromMilliseconds(250), out @lock)) { using (@lock) { // Let's make sure that the basic set of features is enabled. If there are any that are not enabled, then let's enable them first. var theseFeaturesShouldAlwaysBeActive = new[] { @@ -48,7 +63,10 @@ namespace Orchard.Data.Migration { _dataMigrationManager.Update(feature); } catch (Exception ex) { - Logger.Error(ex, "Could not run migrations automatically on {0}.", feature); + if (ex.IsFatal()) { + throw; + } + Logger.Error("Could not run migrations automatically on " + feature, ex); } } } @@ -58,5 +76,16 @@ namespace Orchard.Data.Migration { public void Terminating() { // No-op. } + + /// + /// This ensures that the framework migrations have run for the distributed locking feature, as existing Orchard installations will not have the required tables when upgrading. + /// + private void EnsureDistributedLockSchemaExists() { + // Ensure the distributed lock record schema exists. + var schemaBuilder = new SchemaBuilder(_dataMigrationInterpreter); + var distributedLockSchemaBuilder = new DistributedLockSchemaBuilder(_shellSettings, schemaBuilder); + if (distributedLockSchemaBuilder.EnsureSchema()) + _transactionManager.RequireNew(); + } } } diff --git a/src/Orchard/Data/Migration/DataMigrationManager.cs b/src/Orchard/Data/Migration/DataMigrationManager.cs index 1ef6e33d7..02c6204f0 100644 --- a/src/Orchard/Data/Migration/DataMigrationManager.cs +++ b/src/Orchard/Data/Migration/DataMigrationManager.cs @@ -9,6 +9,7 @@ using Orchard.Data.Migration.Schema; using Orchard.Environment.Extensions; using Orchard.Localization; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Data.Migration { /// @@ -123,6 +124,9 @@ namespace Orchard.Data.Migration { current = (int)lookupTable[current].Invoke(migration, new object[0]); } catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } Logger.Error(ex, "An unexpected error occurred while applying migration on {0} from version {1}.", feature, current); throw; } @@ -139,10 +143,13 @@ namespace Orchard.Data.Migration { dataMigrationRecord.Version = current; } } - catch (Exception e) { - Logger.Error(e, "Error while running migration version {0} for {1}.", current, feature); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error(ex, "Error while running migration version {0} for {1}.", current, feature); _transactionManager.Cancel(); - throw new OrchardException(T("Error while running migration version {0} for {1}.", current, feature), e); + throw new OrchardException(T("Error while running migration version {0} for {1}.", current, feature), ex); } } diff --git a/src/Orchard/Data/Migration/Interpreters/DefaultDataMigrationInterpreter.cs b/src/Orchard/Data/Migration/Interpreters/DefaultDataMigrationInterpreter.cs index e3f5d958c..fc75c61cb 100644 --- a/src/Orchard/Data/Migration/Interpreters/DefaultDataMigrationInterpreter.cs +++ b/src/Orchard/Data/Migration/Interpreters/DefaultDataMigrationInterpreter.cs @@ -382,7 +382,7 @@ namespace Orchard.Data.Migration.Interpreters { } } finally { - _sqlStatements.Clear(); + _sqlStatements.Clear(); } } diff --git a/src/Orchard/Data/Migration/Schema/SchemaBuilder.cs b/src/Orchard/Data/Migration/Schema/SchemaBuilder.cs index 627836efa..541c0619b 100644 --- a/src/Orchard/Data/Migration/Schema/SchemaBuilder.cs +++ b/src/Orchard/Data/Migration/Schema/SchemaBuilder.cs @@ -1,6 +1,7 @@ using System; using Orchard.Data.Migration.Interpreters; using Orchard.Localization; +using Orchard.Exceptions; namespace Orchard.Data.Migration.Schema { public class SchemaBuilder { @@ -72,6 +73,9 @@ namespace Orchard.Data.Migration.Schema { Run(sqlStatmentCommand); return this; } catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } throw new OrchardException(T("An unexpected error occured while executing the SQL statement: {0}", sql), ex); // Add the sql to the nested exception information } } diff --git a/src/Orchard/Data/SessionConfigurationCache.cs b/src/Orchard/Data/SessionConfigurationCache.cs index 037f80801..30bae0547 100644 --- a/src/Orchard/Data/SessionConfigurationCache.cs +++ b/src/Orchard/Data/SessionConfigurationCache.cs @@ -11,6 +11,7 @@ using Orchard.Environment.ShellBuilders.Models; using Orchard.FileSystems.AppData; using Orchard.Logging; using Orchard.Utility; +using Orchard.Exceptions; namespace Orchard.Data { public class SessionConfigurationCache : ISessionConfigurationCache { @@ -80,11 +81,11 @@ namespace Orchard.Data { formatter.Serialize(stream, cache.Configuration); } } - catch (SerializationException e) { + catch (SerializationException ex) { //Note: This can happen when multiple processes/AppDomains try to save // the cached configuration at the same time. Only one concurrent // writer will win, and it's harmless for the other ones to fail. - for (Exception scan = e; scan != null; scan = scan.InnerException) + for (Exception scan = ex; scan != null; scan = scan.InnerException) Logger.Warning("Error storing new NHibernate cache configuration: {0}", scan.Message); } } @@ -118,8 +119,11 @@ namespace Orchard.Data { }; } } - catch (Exception e) { - for (var scan = e; scan != null; scan = scan.InnerException) + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + for (var scan = ex; scan != null; scan = scan.InnerException) Logger.Warning("Error reading the cached NHibernate configuration: {0}", scan.Message); Logger.Information("A new one will be re-generated."); return null; diff --git a/src/Orchard/DisplayManagement/Descriptors/DefaultShapeTableManager.cs b/src/Orchard/DisplayManagement/Descriptors/DefaultShapeTableManager.cs index bc9eb0667..e1cfd57a7 100644 --- a/src/Orchard/DisplayManagement/Descriptors/DefaultShapeTableManager.cs +++ b/src/Orchard/DisplayManagement/Descriptors/DefaultShapeTableManager.cs @@ -37,7 +37,7 @@ namespace Orchard.DisplayManagement.Descriptors { public ILogger Logger { get; set; } public ShapeTable GetShapeTable(string themeName) { - return _cacheManager.Get(themeName ?? "", x => { + return _cacheManager.Get(themeName ?? "", true, x => { Logger.Information("Start building shape table"); var alterationSets = _parallelCacheContext.RunInParallel(_bindingStrategies, bindingStrategy => { diff --git a/src/Orchard/DisplayManagement/Descriptors/ShapePlacementStrategy/PlacementFileParser.cs b/src/Orchard/DisplayManagement/Descriptors/ShapePlacementStrategy/PlacementFileParser.cs index 8725ded54..9ce3454b9 100644 --- a/src/Orchard/DisplayManagement/Descriptors/ShapePlacementStrategy/PlacementFileParser.cs +++ b/src/Orchard/DisplayManagement/Descriptors/ShapePlacementStrategy/PlacementFileParser.cs @@ -30,7 +30,7 @@ namespace Orchard.DisplayManagement.Descriptors.ShapePlacementStrategy { public bool DisableMonitoring { get; set; } public PlacementFile Parse(string virtualPath) { - return _cacheManager.Get(virtualPath, context => { + return _cacheManager.Get(virtualPath, true, context => { if (!DisableMonitoring) { Logger.Debug("Monitoring virtual path \"{0}\"", virtualPath); diff --git a/src/Orchard/DisplayManagement/Descriptors/ShapeTemplateStrategy/ShapeTemplateBindingStrategy.cs b/src/Orchard/DisplayManagement/Descriptors/ShapeTemplateStrategy/ShapeTemplateBindingStrategy.cs index 2e3e841d0..95b0c6694 100644 --- a/src/Orchard/DisplayManagement/Descriptors/ShapeTemplateStrategy/ShapeTemplateBindingStrategy.cs +++ b/src/Orchard/DisplayManagement/Descriptors/ShapeTemplateStrategy/ShapeTemplateBindingStrategy.cs @@ -77,7 +77,7 @@ namespace Orchard.DisplayManagement.Descriptors.ShapeTemplateStrategy { var pathContexts = harvesterInfos.SelectMany(harvesterInfo => harvesterInfo.subPaths.Select(subPath => { var basePath = Path.Combine(extensionDescriptor.Location, extensionDescriptor.Id).Replace(Path.DirectorySeparatorChar, '/'); var virtualPath = Path.Combine(basePath, subPath).Replace(Path.DirectorySeparatorChar, '/'); - var fileNames = _cacheManager.Get(virtualPath, ctx => { + var fileNames = _cacheManager.Get(virtualPath, true, ctx => { if (!_virtualPathProvider.DirectoryExists(virtualPath)) return new List(); @@ -179,8 +179,8 @@ namespace Orchard.DisplayManagement.Descriptors.ShapeTemplateStrategy { private ControllerContext CreateControllerContext() { var controller = new StubController(); - var httpContext = _workContextAccessor.GetContext().HttpContext; - var requestContext = httpContext.Request.RequestContext; + var httpContext = _workContextAccessor.GetContext().Resolve(); + var requestContext = _workContextAccessor.GetContext().Resolve(); var routeData = requestContext.RouteData; routeData.DataTokens["IWorkContextAccessor"] = _workContextAccessor; diff --git a/src/Orchard/DisplayManagement/IPositioned.cs b/src/Orchard/DisplayManagement/IPositioned.cs new file mode 100644 index 000000000..898cdc76a --- /dev/null +++ b/src/Orchard/DisplayManagement/IPositioned.cs @@ -0,0 +1,5 @@ +namespace Orchard.DisplayManagement { + public interface IPositioned { + string Position { get; } + } +} \ No newline at end of file diff --git a/src/Orchard/DisplayManagement/IShape.cs b/src/Orchard/DisplayManagement/IShape.cs index 27ad21151..5ec1483cb 100644 --- a/src/Orchard/DisplayManagement/IShape.cs +++ b/src/Orchard/DisplayManagement/IShape.cs @@ -1,5 +1,4 @@ -using System.Diagnostics; -using Orchard.DisplayManagement.Shapes; +using Orchard.DisplayManagement.Shapes; namespace Orchard.DisplayManagement { /// diff --git a/src/Orchard/DisplayManagement/PositionWrapper.cs b/src/Orchard/DisplayManagement/PositionWrapper.cs new file mode 100644 index 000000000..e684bfdb5 --- /dev/null +++ b/src/Orchard/DisplayManagement/PositionWrapper.cs @@ -0,0 +1,26 @@ +using System.Web; + +namespace Orchard.DisplayManagement { + public class PositionWrapper : IHtmlString, IPositioned { + + private IHtmlString _value; + public string Position { get; private set; } + + public PositionWrapper(IHtmlString value, string position) { + _value = value; + Position = position; + } + + public PositionWrapper(string value, string position) + : this(new HtmlString(HttpUtility.HtmlEncode(value)), position) { + } + + public string ToHtmlString() { + return _value.ToHtmlString(); + } + + public override string ToString() { + return _value.ToString(); + } + } +} diff --git a/src/Orchard/DisplayManagement/ShapeDisplay.cs b/src/Orchard/DisplayManagement/ShapeDisplay.cs index a432adb51..1313d8a13 100644 --- a/src/Orchard/DisplayManagement/ShapeDisplay.cs +++ b/src/Orchard/DisplayManagement/ShapeDisplay.cs @@ -1,6 +1,8 @@ using System.Collections.Generic; using System.Linq; +using System.Web; using System.Web.Mvc; +using System.Web.Routing; using Orchard.DisplayManagement.Implementation; using Orchard.DisplayManagement.Shapes; @@ -8,24 +10,28 @@ namespace Orchard.DisplayManagement { public class ShapeDisplay : IShapeDisplay { private readonly IDisplayHelperFactory _displayHelperFactory; private readonly IWorkContextAccessor _workContextAccessor; + private readonly HttpContextBase _httpContextBase; + private readonly RequestContext _requestContext; public ShapeDisplay( - IDisplayHelperFactory displayHelperFactory, - IWorkContextAccessor workContextAccessor) { + IDisplayHelperFactory displayHelperFactory, + IWorkContextAccessor workContextAccessor, + HttpContextBase httpContextBase, + RequestContext requestContext) { _displayHelperFactory = displayHelperFactory; _workContextAccessor = workContextAccessor; + _httpContextBase = httpContextBase; + _requestContext = requestContext; } public string Display(Shape shape) { - return Display((object)shape); + return Display((object) shape); } public string Display(object shape) { - var workContext = _workContextAccessor.GetContext(); - var httpContext = workContext.HttpContext; var viewContext = new ViewContext { - HttpContext = httpContext, - RequestContext = httpContext.Request.RequestContext + HttpContext = _httpContextBase, + RequestContext = _requestContext }; viewContext.RouteData.DataTokens["IWorkContextAccessor"] = _workContextAccessor; var display = _displayHelperFactory.CreateHelper(viewContext, new ViewDataContainer()); diff --git a/src/Orchard/DisplayManagement/Shapes/Shape.cs b/src/Orchard/DisplayManagement/Shapes/Shape.cs index d009d1123..a3e30e6d8 100644 --- a/src/Orchard/DisplayManagement/Shapes/Shape.cs +++ b/src/Orchard/DisplayManagement/Shapes/Shape.cs @@ -4,11 +4,11 @@ using System.Dynamic; using System.Linq; using System.Collections; using System.Collections.Generic; -using System.Web.Mvc; +using System.Web; namespace Orchard.DisplayManagement.Shapes { [DebuggerTypeProxy(typeof(ShapeDebugView))] - public class Shape : Composite, IShape, IEnumerable { + public class Shape : Composite, IShape, IPositioned, IEnumerable { private const string DefaultPosition = "5"; private readonly IList _items = new List(); @@ -22,6 +22,12 @@ namespace Orchard.DisplayManagement.Shapes { public virtual IDictionary Attributes { get { return _attributes; } } public virtual IEnumerable Items { get { return _items; } } + public string Position { + get { + return Metadata.Position; + } + } + public Shape() { Metadata = new ShapeMetadata(); } @@ -33,13 +39,14 @@ namespace Orchard.DisplayManagement.Shapes { } try { - // todo: (sebros) this is a temporary implementation to prevent common known scenarios throwing exceptions. The final solution would need to filter based on the fact that it is a Shape instance - if (item is MvcHtmlString || - item is String) { - // need to implement positioned wrapper for non-shape objects + if (position != null && item is IHtmlString) { + item = new PositionWrapper((IHtmlString)item, position); + } + else if (position != null && item is string) { + item = new PositionWrapper((string)item, position); } else if (item is IShape) { - ((dynamic)item).Metadata.Position = position; + ((IShape)item).Metadata.Position = position; } } catch { diff --git a/src/Orchard/Environment/ApplicationEnvironment.cs b/src/Orchard/Environment/ApplicationEnvironment.cs new file mode 100644 index 000000000..ceff22c63 --- /dev/null +++ b/src/Orchard/Environment/ApplicationEnvironment.cs @@ -0,0 +1,12 @@ +using System; +using System.Diagnostics; + +namespace Orchard.Environment { + public class ApplicationEnvironment : IApplicationEnvironment { + public string GetEnvironmentIdentifier() { + // Add process ID to machine name because multiple web servers can + // be running on the same physical machine. + return String.Format("{0}:{1}", System.Environment.MachineName, Process.GetCurrentProcess().Id); + } + } +} \ No newline at end of file diff --git a/src/Orchard/Environment/Configuration/ShellSettingsManager.cs b/src/Orchard/Environment/Configuration/ShellSettingsManager.cs index 3b7422d41..6c391cc74 100644 --- a/src/Orchard/Environment/Configuration/ShellSettingsManager.cs +++ b/src/Orchard/Environment/Configuration/ShellSettingsManager.cs @@ -9,7 +9,7 @@ namespace Orchard.Environment.Configuration { public class ShellSettingsManager : Component, IShellSettingsManager { - private const string _settingsFileName = "Settings.txt"; + private const string SettingsFileName = "Settings.txt"; private readonly IAppDataFolder _appDataFolder; private readonly IShellSettingsManagerEventHandler _events; @@ -37,7 +37,7 @@ namespace Orchard.Environment.Configuration { throw new ArgumentException("The Name property of the supplied ShellSettings object is null or empty; the settings cannot be saved.", "settings"); Logger.Information("Saving ShellSettings for tenant '{0}'...", settings.Name); - var filePath = Path.Combine(Path.Combine("Sites", settings.Name), _settingsFileName); + var filePath = Path.Combine(Path.Combine("Sites", settings.Name), SettingsFileName); _appDataFolder.CreateFile(filePath, ShellSettingsSerializer.ComposeSettings(settings)); Logger.Information("ShellSettings saved successfully; flagging tenant '{0}' for restart.", settings.Name); @@ -48,7 +48,7 @@ namespace Orchard.Environment.Configuration { var filePaths = _appDataFolder .ListDirectories("Sites") .SelectMany(path => _appDataFolder.ListFiles(path)) - .Where(path => String.Equals(Path.GetFileName(path), _settingsFileName, StringComparison.OrdinalIgnoreCase)); + .Where(path => String.Equals(Path.GetFileName(path), SettingsFileName, StringComparison.OrdinalIgnoreCase)); foreach (var filePath in filePaths) { yield return ShellSettingsSerializer.ParseSettings(_appDataFolder.ReadFile(filePath)); diff --git a/src/Orchard/Environment/Configuration/TenantState.cs b/src/Orchard/Environment/Configuration/TenantState.cs index 5f1a79060..6ef9e1815 100644 --- a/src/Orchard/Environment/Configuration/TenantState.cs +++ b/src/Orchard/Environment/Configuration/TenantState.cs @@ -1,6 +1,7 @@ namespace Orchard.Environment.Configuration { public enum TenantState { Uninitialized, + Initializing, Running, Disabled, Invalid diff --git a/src/Orchard/Environment/DefaultOrchardHost.cs b/src/Orchard/Environment/DefaultOrchardHost.cs index 6bd9076a8..948142ee3 100644 --- a/src/Orchard/Environment/DefaultOrchardHost.cs +++ b/src/Orchard/Environment/DefaultOrchardHost.cs @@ -1,7 +1,6 @@ using System; using System.Linq; using System.Collections.Generic; -using System.Security.Cryptography; using System.Threading.Tasks; using Orchard.Caching; using Orchard.Environment.Configuration; @@ -12,7 +11,10 @@ using Orchard.Environment.Descriptor; using Orchard.Environment.Descriptor.Models; using Orchard.Localization; using Orchard.Logging; +using Orchard.Mvc; +using Orchard.Mvc.Extensions; using Orchard.Utility.Extensions; +using Orchard.Exceptions; namespace Orchard.Environment { // All the event handlers that DefaultOrchardHost implements have to be declared in OrchardStarter. @@ -25,12 +27,13 @@ namespace Orchard.Environment { private readonly IExtensionLoaderCoordinator _extensionLoaderCoordinator; private readonly IExtensionMonitoringCoordinator _extensionMonitoringCoordinator; private readonly ICacheManager _cacheManager; + private readonly IHttpContextAccessor _httpContextAccessor; private readonly static object _syncLock = new object(); private readonly static object _shellContextsWriteLock = new object(); private IEnumerable _shellContexts; - private readonly ContextState> _tenantsToRestart; + public DefaultOrchardHost( IShellSettingsManager shellSettingsManager, IShellContextFactory shellContextFactory, @@ -39,7 +42,9 @@ namespace Orchard.Environment { IExtensionLoaderCoordinator extensionLoaderCoordinator, IExtensionMonitoringCoordinator extensionMonitoringCoordinator, ICacheManager cacheManager, - IHostLocalRestart hostLocalRestart) { + IHostLocalRestart hostLocalRestart, + IHttpContextAccessor httpContextAccessor) { + _shellSettingsManager = shellSettingsManager; _shellContextFactory = shellContextFactory; _runningShellTable = runningShellTable; @@ -48,6 +53,7 @@ namespace Orchard.Environment { _extensionMonitoringCoordinator = extensionMonitoringCoordinator; _cacheManager = cacheManager; _hostLocalRestart = hostLocalRestart; + _httpContextAccessor = httpContextAccessor; _tenantsToRestart = new ContextState>("DefaultOrchardHost.TenantsToRestart", () => new List()); @@ -128,20 +134,23 @@ namespace Orchard.Environment { void CreateAndActivateShells() { Logger.Information("Start creation of shells"); - // is there any tenant right now ? + // Is there any tenant right now? var allSettings = _shellSettingsManager.LoadSettings() - .Where(settings => settings.State == TenantState.Running || settings.State == TenantState.Uninitialized) + .Where(settings => settings.State == TenantState.Running || settings.State == TenantState.Uninitialized || settings.State == TenantState.Initializing) .ToArray(); - // load all tenants, and activate their shell + // Load all tenants, and activate their shell. if (allSettings.Any()) { Parallel.ForEach(allSettings, settings => { try { var context = CreateShellContext(settings); ActivateShell(context); } - catch (Exception e) { - Logger.Error(e, "A tenant could not be started: " + settings.Name); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error(ex, "A tenant could not be started: " + settings.Name); } while (_processingEngine.AreTasksPending()) { Logger.Debug("Processing pending task after activate Shell"); @@ -149,7 +158,7 @@ namespace Orchard.Environment { } }); } - // no settings, run the Setup + // No settings, run the Setup. else { var setupContext = CreateSetupContext(); ActivateShell(setupContext); @@ -176,23 +185,23 @@ namespace Orchard.Environment { } /// - /// Creates a transient shell for the default tenant's setup + /// Creates a transient shell for the default tenant's setup. /// private ShellContext CreateSetupContext() { - Logger.Debug("Creating shell context for root setup"); + Logger.Debug("Creating shell context for root setup."); return _shellContextFactory.CreateSetupContext(new ShellSettings { Name = ShellSettings.DefaultName }); } /// - /// Creates a shell context based on shell settings + /// Creates a shell context based on shell settings. /// private ShellContext CreateShellContext(ShellSettings settings) { - if (settings.State == TenantState.Uninitialized) { - Logger.Debug("Creating shell context for tenant {0} setup", settings.Name); + if (settings.State == TenantState.Uninitialized || settings.State == TenantState.Invalid) { + Logger.Debug("Creating shell context for tenant {0} setup.", settings.Name); return _shellContextFactory.CreateSetupContext(settings); } - Logger.Debug("Creating shell context for tenant {0}", settings.Name); + Logger.Debug("Creating shell context for tenant {0}.", settings.Name); return _shellContextFactory.CreateShellContext(settings); } @@ -204,7 +213,7 @@ namespace Orchard.Environment { // This is a "fake" cache entry to allow the extension loader coordinator // notify us (by resetting _current to "null") when an extension has changed // on disk, and we need to reload new/updated extensions. - _cacheManager.Get("OrchardHost_Extensions", + _cacheManager.Get("OrchardHost_Extensions", true, ctx => { _extensionMonitoringCoordinator.MonitorExtensions(ctx.Monitor); _hostLocalRestart.Monitor(ctx.Monitor); @@ -234,6 +243,8 @@ namespace Orchard.Environment { } protected virtual void BeginRequest() { + BlockRequestsDuringSetup(); + // Ensure all shell contexts are loaded, or need to be reloaded if // extensions have changed MonitorExtensions(); @@ -341,6 +352,25 @@ namespace Orchard.Environment { _tenantsToRestart.GetState().Add(context.Settings); } + private void BlockRequestsDuringSetup() { + var httpContext = _httpContextAccessor.Current(); + if (httpContext.IsBackgroundContext()) + return; + + // Get the requested shell. + var runningShell = _runningShellTable.Match(httpContext); + if (runningShell == null) + return; + + // If the requested shell is currently initializing, return a Service Unavailable HTTP status code. + if (runningShell.State == TenantState.Initializing) { + var response = httpContext.Response; + response.StatusCode = 503; + response.StatusDescription = "This tenant is currently initializing. Please try again later."; + response.Write("This tenant is currently initializing. Please try again later."); + } + } + // To be used from CreateStandaloneEnvironment(), also disposes the ShellContext LifetimeScope. private class StandaloneEnvironmentWorkContextScopeWrapper : IWorkContextScope { private readonly ShellContext _shellContext; diff --git a/src/Orchard/Environment/DefaultOrchardShell.cs b/src/Orchard/Environment/DefaultOrchardShell.cs index 400c0399f..8b935499d 100644 --- a/src/Orchard/Environment/DefaultOrchardShell.cs +++ b/src/Orchard/Environment/DefaultOrchardShell.cs @@ -11,11 +11,12 @@ using Orchard.Owin; using Orchard.Tasks; using Orchard.UI; using Orchard.WebApi.Routes; +using Orchard.Exceptions; using IModelBinderProvider = Orchard.Mvc.ModelBinders.IModelBinderProvider; namespace Orchard.Environment { public class DefaultOrchardShell : IOrchardShell { - private readonly Func> _eventsFactory; + private readonly IWorkContextAccessor _workContextAccessor; private readonly IEnumerable _routeProviders; private readonly IEnumerable _httpRouteProviders; private readonly IRoutePublisher _routePublisher; @@ -26,7 +27,7 @@ namespace Orchard.Environment { private readonly ShellSettings _shellSettings; public DefaultOrchardShell( - Func> eventsFactory, + IWorkContextAccessor workContextAccessor, IEnumerable routeProviders, IEnumerable httpRouteProviders, IRoutePublisher routePublisher, @@ -35,8 +36,7 @@ namespace Orchard.Environment { ISweepGenerator sweepGenerator, IEnumerable owinMiddlewareProviders, ShellSettings shellSettings) { - - _eventsFactory = eventsFactory; + _workContextAccessor = workContextAccessor; _routeProviders = routeProviders; _httpRouteProviders = httpRouteProviders; _routePublisher = routePublisher; @@ -74,8 +74,10 @@ namespace Orchard.Environment { _routePublisher.Publish(allRoutes, pipeline); _modelBinderPublisher.Publish(_modelBinderProviders.SelectMany(provider => provider.GetModelBinders())); - using (var events = _eventsFactory()) { - events.Value.Activated(); + using (var scope = _workContextAccessor.CreateWorkContextScope()) { + using (var events = scope.Resolve>()) { + events.Value.Activated(); + } } _sweepGenerator.Activate(); @@ -83,22 +85,26 @@ namespace Orchard.Environment { public void Terminate() { SafelyTerminate(() => { - using (var events = _eventsFactory()) { - var localEvents = events; - SafelyTerminate(() => localEvents.Value.Terminating()); - } + using (var scope = _workContextAccessor.CreateWorkContextScope()) { + using (var events = scope.Resolve>()) { + SafelyTerminate(() => events.Value.Terminating()); + } + } }); SafelyTerminate(() => _sweepGenerator.Terminate()); } - private void SafelyTerminate(Action action) { try { action(); } - catch(Exception e) { - Logger.Error(e, "An unexcepted error occured while terminating the Shell"); + catch(Exception ex) { + if (ex.IsFatal()) { + throw; + } + + Logger.Error(ex, "An unexcepted error occured while terminating the Shell"); } } } diff --git a/src/Orchard/Environment/Extensions/Compilers/DefaultExtensionCompiler.cs b/src/Orchard/Environment/Extensions/Compilers/DefaultExtensionCompiler.cs index 3dfc2d6af..488dcb90a 100644 --- a/src/Orchard/Environment/Extensions/Compilers/DefaultExtensionCompiler.cs +++ b/src/Orchard/Environment/Extensions/Compilers/DefaultExtensionCompiler.cs @@ -8,6 +8,7 @@ using Orchard.FileSystems.Dependencies; using Orchard.FileSystems.VirtualPath; using Orchard.Localization; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Environment.Extensions.Compilers { /// @@ -106,10 +107,13 @@ namespace Orchard.Environment.Extensions.Compilers { } } } - catch (Exception e) { + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } //Note: we need to embed the "e.Message" in the exception text because // ASP.NET build manager "swallows" inner exceptions from this method. - throw new OrchardCoreException(T("Error compiling module \"{0}\" from file \"{1}\":\r\n{2}", moduleName, context.VirtualPath, e.Message), e); + throw new OrchardCoreException(T("Error compiling module \"{0}\" from file \"{1}\":\r\n{2}", moduleName, context.VirtualPath, ex.Message), ex); } } diff --git a/src/Orchard/Environment/Extensions/Compilers/DefaultProjectFileParser.cs b/src/Orchard/Environment/Extensions/Compilers/DefaultProjectFileParser.cs index 01a717b12..08adf43a1 100644 --- a/src/Orchard/Environment/Extensions/Compilers/DefaultProjectFileParser.cs +++ b/src/Orchard/Environment/Extensions/Compilers/DefaultProjectFileParser.cs @@ -23,7 +23,7 @@ namespace Orchard.Environment.Extensions.Compilers { public bool DisableMonitoring { get; set; } public ProjectFileDescriptor Parse(string virtualPath) { - return _cacheManager.Get(virtualPath, + return _cacheManager.Get(virtualPath, true, ctx => { if (!DisableMonitoring) { Logger.Debug("Monitoring virtual path \"{0}\"", virtualPath); diff --git a/src/Orchard/Environment/Extensions/ExtensionManager.cs b/src/Orchard/Environment/Extensions/ExtensionManager.cs index 3701c9764..40092fd21 100644 --- a/src/Orchard/Environment/Extensions/ExtensionManager.cs +++ b/src/Orchard/Environment/Extensions/ExtensionManager.cs @@ -9,6 +9,7 @@ using Orchard.Localization; using Orchard.Logging; using Orchard.Utility; using Orchard.Utility.Extensions; +using Orchard.Exceptions; namespace Orchard.Environment.Extensions { public class ExtensionManager : IExtensionManager { @@ -44,7 +45,7 @@ namespace Orchard.Environment.Extensions { } public IEnumerable AvailableExtensions() { - return _cacheManager.Get("AvailableExtensions", ctx => + return _cacheManager.Get("AvailableExtensions", true, ctx => _parallelCacheContext .RunInParallel(_folders, folder => folder.AvailableExtensions().ToList()) .SelectMany(descriptors => descriptors) @@ -52,7 +53,7 @@ namespace Orchard.Environment.Extensions { } public IEnumerable AvailableFeatures() { - return _cacheManager.Get("AvailableFeatures", ctx => + return _cacheManager.Get("AvailableFeatures", true, ctx => AvailableExtensions() .SelectMany(ext => ext.Features) .OrderByDependenciesAndPriorities(HasDependency, GetPriority) @@ -92,7 +93,7 @@ namespace Orchard.Environment.Extensions { var result = _parallelCacheContext - .RunInParallel(featureDescriptors, descriptor => _cacheManager.Get(descriptor.Id, ctx => LoadFeature(descriptor))) + .RunInParallel(featureDescriptors, descriptor => _cacheManager.Get(descriptor.Id, true, ctx => LoadFeature(descriptor))) .ToArray(); Logger.Information("Done loading features"); @@ -106,7 +107,7 @@ namespace Orchard.Environment.Extensions { ExtensionEntry extensionEntry; try { - extensionEntry = _cacheManager.Get(extensionId, ctx => { + extensionEntry = _cacheManager.Get(extensionId, true, ctx => { var entry = BuildEntry(extensionDescriptor); if (entry != null) { ctx.Monitor(_asyncTokenProvider.GetToken(monitor => { @@ -119,6 +120,9 @@ namespace Orchard.Environment.Extensions { }); } catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } Logger.Error(ex, "Error loading extension '{0}'", extensionId); throw new OrchardException(T("Error while loading extension '{0}'.", extensionId), ex); } diff --git a/src/Orchard/Environment/Extensions/Folders/ExtensionHarvester.cs b/src/Orchard/Environment/Extensions/Folders/ExtensionHarvester.cs index 6dfe8cf80..808aa6636 100644 --- a/src/Orchard/Environment/Extensions/Folders/ExtensionHarvester.cs +++ b/src/Orchard/Environment/Extensions/Folders/ExtensionHarvester.cs @@ -9,6 +9,7 @@ using Orchard.FileSystems.WebSite; using Orchard.Localization; using Orchard.Logging; using Orchard.Utility.Extensions; +using Orchard.Exceptions; namespace Orchard.Environment.Extensions.Folders { public class ExtensionHarvester : IExtensionHarvester { @@ -57,7 +58,7 @@ namespace Orchard.Environment.Extensions.Folders { private IEnumerable HarvestExtensions(string path, string extensionType, string manifestName, bool manifestIsOptional) { string key = string.Format("{0}-{1}-{2}", path, manifestName, extensionType); - return _cacheManager.Get(key, ctx => { + return _cacheManager.Get(key, true, ctx => { if (!DisableMonitoring) { Logger.Debug("Monitoring virtual path \"{0}\"", path); ctx.Monitor(_webSiteFolder.WhenPathChanges(path)); @@ -100,6 +101,9 @@ namespace Orchard.Environment.Extensions.Folders { } catch (Exception ex) { // Ignore invalid module manifests + if (ex.IsFatal()) { + throw; + } Logger.Error(ex, "The module '{0}' could not be loaded. It was ignored.", extensionId); _criticalErrorProvider.RegisterErrorMessage(T("The extension '{0}' manifest could not be loaded. It was ignored.", extensionId)); } @@ -134,7 +138,7 @@ namespace Orchard.Environment.Extensions.Folders { } private ExtensionDescriptor GetExtensionDescriptor(string locationPath, string extensionId, string extensionType, string manifestPath, bool manifestIsOptional) { - return _cacheManager.Get(manifestPath, context => { + return _cacheManager.Get(manifestPath, true, context => { if (!DisableMonitoring) { Logger.Debug("Monitoring virtual path \"{0}\"", manifestPath); context.Monitor(_webSiteFolder.WhenPathChanges(manifestPath)); diff --git a/src/Orchard/Environment/IApplicationEnvironment.cs b/src/Orchard/Environment/IApplicationEnvironment.cs new file mode 100644 index 000000000..cccd6fb53 --- /dev/null +++ b/src/Orchard/Environment/IApplicationEnvironment.cs @@ -0,0 +1,13 @@ +namespace Orchard.Environment { + + /// + /// Describes a service which returns the a machine identifier running the application. + /// + public interface IApplicationEnvironment { + + /// + /// Returns the machine identifier running the application. + /// + string GetEnvironmentIdentifier(); + } +} \ No newline at end of file diff --git a/src/Orchard/Environment/IAssemblyLoader.cs b/src/Orchard/Environment/IAssemblyLoader.cs index 410822a45..32722b5f1 100644 --- a/src/Orchard/Environment/IAssemblyLoader.cs +++ b/src/Orchard/Environment/IAssemblyLoader.cs @@ -4,6 +4,8 @@ using System.Collections.Generic; using System.Linq; using System.Reflection; using Orchard.Logging; +using Orchard.Exceptions; + namespace Orchard.Environment { public interface IAssemblyLoader { @@ -25,8 +27,11 @@ namespace Orchard.Environment { try { return _loadedAssemblies.GetOrAdd(this.ExtractAssemblyShortName(assemblyName), shortName => LoadWorker(shortName, assemblyName)); } - catch (Exception e) { - Logger.Error(e, "Error loading assembly '{0}'", assemblyName); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error(ex, "Error loading assembly '{0}'", assemblyName); return null; } } diff --git a/src/Orchard/Environment/IAssemblyNameResolver.cs b/src/Orchard/Environment/IAssemblyNameResolver.cs index f97631719..78d809ab7 100644 --- a/src/Orchard/Environment/IAssemblyNameResolver.cs +++ b/src/Orchard/Environment/IAssemblyNameResolver.cs @@ -37,7 +37,7 @@ namespace Orchard.Environment { public string Resolve(string shortName) { // A few common .net framework assemblies are referenced by the Orchard.Framework assembly. // Look into those to see if we can find the assembly we are looking for. - var orchardFrameworkReferences = _cacheManager.Get(typeof(IAssemblyLoader), ctx => + var orchardFrameworkReferences = _cacheManager.Get(typeof(IAssemblyLoader), true, ctx => ctx.Key.Assembly .GetReferencedAssemblies() .GroupBy(n => AssemblyLoaderExtensions.ExtractAssemblyShortName(n.FullName), StringComparer.OrdinalIgnoreCase) diff --git a/src/Orchard/Environment/IBuildManager.cs b/src/Orchard/Environment/IBuildManager.cs index d9ac231ab..40b8c9aa8 100644 --- a/src/Orchard/Environment/IBuildManager.cs +++ b/src/Orchard/Environment/IBuildManager.cs @@ -53,8 +53,9 @@ namespace Orchard.Environment { return BuildManager.GetCompiledAssembly(virtualPath); } catch (Exception ex) { - if (ex.IsFatal()) throw; - + if (ex.IsFatal()) { + throw; + } Logger.Warning(ex, "Error when compiling assembly under {0}.", virtualPath); return null; } diff --git a/src/Orchard/Environment/IHostLocalRestart.cs b/src/Orchard/Environment/IHostLocalRestart.cs index 90a2371d2..f801e6efe 100644 --- a/src/Orchard/Environment/IHostLocalRestart.cs +++ b/src/Orchard/Environment/IHostLocalRestart.cs @@ -5,6 +5,7 @@ using Orchard.Environment.Descriptor; using Orchard.Environment.Descriptor.Models; using Orchard.FileSystems.AppData; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Environment { public interface IHostLocalRestart { @@ -45,8 +46,11 @@ namespace Orchard.Environment { try { _appDataFolder.CreateFile(fileName, "Host Restart"); } - catch(Exception e) { - Logger.Warning(e, "Error updating file '{0}'", fileName); + catch(Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Warning(ex, "Error updating file '{0}'", fileName); } } } diff --git a/src/Orchard/Environment/IMachineNameProvider.cs b/src/Orchard/Environment/IMachineNameProvider.cs deleted file mode 100644 index 93391e934..000000000 --- a/src/Orchard/Environment/IMachineNameProvider.cs +++ /dev/null @@ -1,13 +0,0 @@ -namespace Orchard.Environment { - - /// - /// Describes a service which returns the name of the machine running the application. - /// - public interface IMachineNameProvider { - - /// - /// Returns the name of the machine running the application. - /// - string GetMachineName(); - } -} \ No newline at end of file diff --git a/src/Orchard/Environment/IOrchardHost.cs b/src/Orchard/Environment/IOrchardHost.cs index aee8940e2..613176f2e 100644 --- a/src/Orchard/Environment/IOrchardHost.cs +++ b/src/Orchard/Environment/IOrchardHost.cs @@ -1,6 +1,5 @@ using Orchard.Environment.Configuration; using Orchard.Environment.ShellBuilders; -using Orchard.Localization; namespace Orchard.Environment { public interface IOrchardHost { diff --git a/src/Orchard/Environment/IThreadProvider.cs b/src/Orchard/Environment/IThreadProvider.cs deleted file mode 100644 index 29458bf27..000000000 --- a/src/Orchard/Environment/IThreadProvider.cs +++ /dev/null @@ -1,13 +0,0 @@ -namespace Orchard.Environment { - - /// - /// Describes a service which returns the managed thread ID of the current thread. - /// - public interface IThreadProvider { - - /// - /// Returns the managed thread ID of the current thread. - /// - int GetCurrentThreadId(); - } -} \ No newline at end of file diff --git a/src/Orchard/Environment/MachineNameProvider.cs b/src/Orchard/Environment/MachineNameProvider.cs deleted file mode 100644 index 4c7543b0a..000000000 --- a/src/Orchard/Environment/MachineNameProvider.cs +++ /dev/null @@ -1,7 +0,0 @@ -namespace Orchard.Environment { - public class MachineNameProvider : IMachineNameProvider { - public string GetMachineName() { - return System.Environment.MachineName; - } - } -} \ No newline at end of file diff --git a/src/Orchard/Environment/OrchardStarter.cs b/src/Orchard/Environment/OrchardStarter.cs index 1e0db5736..79736c0be 100644 --- a/src/Orchard/Environment/OrchardStarter.cs +++ b/src/Orchard/Environment/OrchardStarter.cs @@ -63,11 +63,10 @@ namespace Orchard.Environment { builder.RegisterType().As().SingleInstance(); builder.RegisterType().As().SingleInstance(); builder.RegisterType().As().SingleInstance(); - builder.RegisterType().As().SingleInstance(); + builder.RegisterType().As().InstancePerDependency(); builder.RegisterType().As().SingleInstance(); builder.RegisterType().As().SingleInstance(); builder.RegisterType().As().SingleInstance(); - builder.RegisterType().As().SingleInstance(); //builder.RegisterType().As().SingleInstance(); RegisterVolatileProvider(builder); @@ -80,7 +79,6 @@ namespace Orchard.Environment { RegisterVolatileProvider(builder); RegisterVolatileProvider(builder); - builder.RegisterType().As().As() .Named(typeof(IShellSettingsManagerEventHandler).Name) .Named(typeof(IShellDescriptorManagerEventHandler).Name) diff --git a/src/Orchard/Environment/ShellBuilders/CompositionStrategy.cs b/src/Orchard/Environment/ShellBuilders/CompositionStrategy.cs index 08988203d..5226f0a09 100644 --- a/src/Orchard/Environment/ShellBuilders/CompositionStrategy.cs +++ b/src/Orchard/Environment/ShellBuilders/CompositionStrategy.cs @@ -9,7 +9,6 @@ using Orchard.ContentManagement.Records; using Orchard.Environment.Configuration; using Orchard.Environment.Descriptor.Models; using Orchard.Environment.Extensions; -using Orchard.Environment.Extensions.Helpers; using Orchard.Environment.Extensions.Models; using Orchard.Environment.ShellBuilders.Models; using Orchard.Localization; @@ -34,7 +33,11 @@ namespace Orchard.Environment.ShellBuilders { var builtinFeatures = BuiltinFeatures().ToList(); var builtinFeatureDescriptors = builtinFeatures.Select(x => x.Descriptor).ToList(); - var availableFeatures = _extensionManager.AvailableFeatures().Concat(builtinFeatureDescriptors).ToDictionary(x => x.Id, StringComparer.OrdinalIgnoreCase); + var availableFeatures = _extensionManager.AvailableFeatures() + .Concat(builtinFeatureDescriptors) + .GroupBy(x => x.Id.ToLowerInvariant()) // prevent duplicates + .Select(x => x.FirstOrDefault()) + .ToDictionary(x => x.Id, StringComparer.OrdinalIgnoreCase); var enabledFeatures = _extensionManager.EnabledFeatures(descriptor).Select(x => x.Id).ToList(); var expandedFeatures = ExpandDependencies(availableFeatures, descriptor.Features.Select(x => x.Name)).ToList(); var autoEnabledDependencyFeatures = expandedFeatures.Except(enabledFeatures).Except(builtinFeatureDescriptors.Select(x => x.Id)).ToList(); @@ -72,19 +75,29 @@ namespace Orchard.Environment.ShellBuilders { } private IEnumerable ExpandDependencies(IDictionary availableFeatures, IEnumerable features) { - return ExpandDependenciesInternal(availableFeatures, features).Distinct(); + return ExpandDependenciesInternal(availableFeatures, features, dependentFeatureDescriptor: null).Distinct(); } - private IEnumerable ExpandDependenciesInternal(IDictionary availableFeatures, IEnumerable features) { + private IEnumerable ExpandDependenciesInternal(IDictionary availableFeatures, IEnumerable features, FeatureDescriptor dependentFeatureDescriptor = null) { foreach (var shellFeature in features) { if (!availableFeatures.ContainsKey(shellFeature)) { - throw new OrchardException(T("The feature {0} is not available", shellFeature)); + // If the feature comes from a list of feature dependencies it indicates a bug, so throw an exception. + if(dependentFeatureDescriptor != null) + throw new OrchardException( + T("The feature '{0}' was listed as a dependency of '{1}' of extension '{2}', but this feature could not be found. Please update your manifest file or install the module providing the missing feature.", + shellFeature, + dependentFeatureDescriptor.Name, + dependentFeatureDescriptor.Extension.Name)); + + // If the feature comes from the shell descriptor it means the feature is simply orphaned, so don't throw an exception. + Logger.Warning("Identified '{0}' as an orphaned feature state record in Settings_ShellFeatureRecord.", shellFeature); + continue; } var feature = availableFeatures[shellFeature]; - foreach (var childDependency in ExpandDependenciesInternal(availableFeatures, feature.Dependencies)) + foreach (var childDependency in ExpandDependenciesInternal(availableFeatures, feature.Dependencies, dependentFeatureDescriptor: feature)) yield return childDependency; foreach (var dependency in feature.Dependencies) diff --git a/src/Orchard/Environment/ShellBuilders/ShellContainerFactory.cs b/src/Orchard/Environment/ShellBuilders/ShellContainerFactory.cs index 145f40f98..e621c6a6b 100644 --- a/src/Orchard/Environment/ShellBuilders/ShellContainerFactory.cs +++ b/src/Orchard/Environment/ShellBuilders/ShellContainerFactory.cs @@ -135,6 +135,10 @@ namespace Orchard.Environment.ShellBuilders { if (File.Exists(optionalShellConfig)) builder.RegisterModule(new ConfigurationSettingsReader(ConfigurationSettingsReaderConstants.DefaultSectionName, optionalShellConfig)); } + + var optionalComponentsConfig = HostingEnvironment.MapPath("~/Config/HostComponents.config"); + if (File.Exists(optionalComponentsConfig)) + builder.RegisterModule(new HostComponentsConfigModule(optionalComponentsConfig)); }); } diff --git a/src/Orchard/Environment/ShellBuilders/ShellContextFactory.cs b/src/Orchard/Environment/ShellBuilders/ShellContextFactory.cs index 7df34f2fe..5646a3b3c 100644 --- a/src/Orchard/Environment/ShellBuilders/ShellContextFactory.cs +++ b/src/Orchard/Environment/ShellBuilders/ShellContextFactory.cs @@ -103,7 +103,7 @@ namespace Orchard.Environment.ShellBuilders { Features = new[] { new ShellFeature { Name = "Orchard.Setup" }, new ShellFeature { Name = "Shapes" }, - new ShellFeature { Name = "Orchard.jQuery" }, + new ShellFeature { Name = "Orchard.jQuery" } }, }; diff --git a/src/Orchard/Environment/ThreadProvider.cs b/src/Orchard/Environment/ThreadProvider.cs deleted file mode 100644 index d3a65d8fb..000000000 --- a/src/Orchard/Environment/ThreadProvider.cs +++ /dev/null @@ -1,9 +0,0 @@ -using System.Threading; - -namespace Orchard.Environment { - public class ThreadProvider : IThreadProvider { - public int GetCurrentThreadId() { - return Thread.CurrentThread.ManagedThreadId; - } - } -} \ No newline at end of file diff --git a/src/Orchard/Environment/ViewsBackgroundCompilation.cs b/src/Orchard/Environment/ViewsBackgroundCompilation.cs index 919fc9bd2..e85526860 100644 --- a/src/Orchard/Environment/ViewsBackgroundCompilation.cs +++ b/src/Orchard/Environment/ViewsBackgroundCompilation.cs @@ -6,6 +6,7 @@ using System.Timers; using System.Web.Compilation; using Orchard.FileSystems.VirtualPath; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Environment { public interface IViewsBackgroundCompilation { @@ -136,10 +137,13 @@ namespace Orchard.Environment { if (firstFile != null) BuildManager.GetCompiledAssembly(firstFile); } - catch(Exception e) { + catch(Exception ex) { + if (ex.IsFatal()) { + throw; + } // Some views might not compile, this is ok and harmless in this // context of pre-compiling views. - Logger.Information(e, "Compilation of directory '{0}' skipped", viewDirectory); + Logger.Information(ex, "Compilation of directory '{0}' skipped", viewDirectory); } stopwatch.Stop(); Logger.Information("Directory '{0}' compiled in {1} msec", viewDirectory, stopwatch.ElapsedMilliseconds); diff --git a/src/Orchard/Environment/WorkContextAccessor.cs b/src/Orchard/Environment/WorkContextAccessor.cs index 7e6bd0ea3..af709d0d5 100644 --- a/src/Orchard/Environment/WorkContextAccessor.cs +++ b/src/Orchard/Environment/WorkContextAccessor.cs @@ -60,14 +60,11 @@ namespace Orchard.Environment { return CreateWorkContextScope(httpContext); var workLifetime = _lifetimeScope.BeginLifetimeScope("work"); - httpContext = _httpContextAccessor.CreateContext(workLifetime); - workLifetime.Resolve>().Value = httpContext; var events = workLifetime.Resolve>(); events.Invoke(e => e.Started(), NullLogger.Instance); return new ThreadStaticScopeImplementation( - httpContext, events, workLifetime, EnsureThreadStaticContexts(), @@ -116,9 +113,8 @@ namespace Orchard.Environment { readonly WorkContext _workContext; readonly Action _disposer; - public ThreadStaticScopeImplementation(HttpContextBase httpContext, IEnumerable events, ILifetimeScope lifetimeScope, ConcurrentDictionary contexts, object workContextKey) { + public ThreadStaticScopeImplementation(IEnumerable events, ILifetimeScope lifetimeScope, ConcurrentDictionary contexts, object workContextKey) { _workContext = lifetimeScope.Resolve(); - httpContext.Items[workContextKey] = _workContext; contexts.AddOrUpdate(workContextKey, _workContext, (a, b) => _workContext); _disposer = () => { @@ -127,11 +123,6 @@ namespace Orchard.Environment { WorkContext removedContext; contexts.TryRemove(workContextKey, out removedContext); lifetimeScope.Dispose(); - - var staticHttpContext = httpContext as IDisposable; - - if(staticHttpContext != null) - staticHttpContext.Dispose(); }; } diff --git a/src/Orchard/Environment/WorkContextImplementation.cs b/src/Orchard/Environment/WorkContextImplementation.cs index 85a5f9fda..63269e079 100644 --- a/src/Orchard/Environment/WorkContextImplementation.cs +++ b/src/Orchard/Environment/WorkContextImplementation.cs @@ -8,28 +8,36 @@ namespace Orchard.Environment { class WorkContextImplementation : WorkContext { readonly IComponentContext _componentContext; readonly ConcurrentDictionary> _stateResolvers = new ConcurrentDictionary>(); - readonly IEnumerable _workContextStateProviders; + readonly IEnumerable> _workContextStateProviders; public WorkContextImplementation(IComponentContext componentContext) { _componentContext = componentContext; - _workContextStateProviders = componentContext.Resolve>(); + _workContextStateProviders = componentContext.Resolve>>(); } public override T Resolve() { return _componentContext.Resolve(); } + public override object Resolve(Type serviceType) { + return _componentContext.Resolve(serviceType); + } + public override bool TryResolve(out T service) { return _componentContext.TryResolve(out service); } + public override bool TryResolve(Type serviceType, out object service) { + return _componentContext.TryResolve(serviceType, out service); + } + public override T GetState(string name) { var resolver = _stateResolvers.GetOrAdd(name, FindResolverForState); return (T)resolver(); } Func FindResolverForState(string name) { - var resolver = _workContextStateProviders.Select(wcsp => wcsp.Get(name)).FirstOrDefault(value => !Equals(value, default(T))); + var resolver = _workContextStateProviders.Select(wcsp => wcsp.Value.Get(name)).FirstOrDefault(value => !Equals(value, default(T))); if (resolver == null) { return () => default(T); diff --git a/src/Orchard/Environment/WorkContextModule.cs b/src/Orchard/Environment/WorkContextModule.cs index e9475bed3..89778a422 100644 --- a/src/Orchard/Environment/WorkContextModule.cs +++ b/src/Orchard/Environment/WorkContextModule.cs @@ -6,7 +6,6 @@ using System.Web; using Autofac; using Autofac.Builder; using Autofac.Core; -using Autofac.Features.Metadata; using Module = Autofac.Module; namespace Orchard.Environment { @@ -24,10 +23,6 @@ namespace Orchard.Environment { .As>() .InstancePerMatchingLifetimeScope("work"); - builder.Register(ctx => ctx.Resolve>().Value) - .As() - .InstancePerDependency(); - builder.RegisterGeneric(typeof(WorkValues<>)) .InstancePerMatchingLifetimeScope("work"); diff --git a/src/Orchard/Events/DefaultOrchardEventBus.cs b/src/Orchard/Events/DefaultOrchardEventBus.cs index d69d84023..2a5faf49e 100644 --- a/src/Orchard/Events/DefaultOrchardEventBus.cs +++ b/src/Orchard/Events/DefaultOrchardEventBus.cs @@ -53,6 +53,9 @@ namespace Orchard.Events { return TryInvoke(eventHandler, messageName, interfaceName, methodName, eventData, out returnValue); } catch (Exception exception) { + if (exception.IsFatal()) { + throw; + } if (!_exceptionPolicy.HandleException(this, exception)) { throw; } diff --git a/src/Orchard/Events/DelegateHelper.cs b/src/Orchard/Events/DelegateHelper.cs index cf3803828..98bc20489 100644 --- a/src/Orchard/Events/DelegateHelper.cs +++ b/src/Orchard/Events/DelegateHelper.cs @@ -100,7 +100,7 @@ namespace Orchard.Events { } static Func BuildFunc(MethodInfo method) { - var func = (Func)Delegate.CreateDelegate(typeof(Func), method); + var func = (Func)Delegate.CreateDelegate(typeof(Func), method); Func ret = (target, p) => func((T)target, (T1)p[0], (T2)p[1], (T3)p[2], (T4)p[3], (T5)p[4], (T6)p[5], (T7)p[6], (T8)p[7], (T9)p[8], (T10)p[9]); return ret; } diff --git a/src/Orchard/FileSystems/AppData/AppDataFolder.cs b/src/Orchard/FileSystems/AppData/AppDataFolder.cs index 02e60e283..092ca9b45 100644 --- a/src/Orchard/FileSystems/AppData/AppDataFolder.cs +++ b/src/Orchard/FileSystems/AppData/AppDataFolder.cs @@ -7,6 +7,7 @@ using Orchard.FileSystems.VirtualPath; using Orchard.Localization; using Orchard.Logging; using Orchard.Validation; +using Orchard.Exceptions; namespace Orchard.FileSystems.AppData { public class AppDataFolder : IAppDataFolder { @@ -78,8 +79,11 @@ namespace Orchard.FileSystems.AppData { try { File.Delete(destinationFileName); } - catch (Exception e) { - throw new OrchardCoreException(T("Unable to make room for file \"{0}\" in \"App_Data\" folder", destinationFileName), e); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + throw new OrchardCoreException(T("Unable to make room for file \"{0}\" in \"App_Data\" folder", destinationFileName), ex); } } diff --git a/src/Orchard/FileSystems/Dependencies/DefaultDependenciesFolder.cs b/src/Orchard/FileSystems/Dependencies/DefaultDependenciesFolder.cs index de9c05869..ace5674fe 100644 --- a/src/Orchard/FileSystems/Dependencies/DefaultDependenciesFolder.cs +++ b/src/Orchard/FileSystems/Dependencies/DefaultDependenciesFolder.cs @@ -34,7 +34,7 @@ namespace Orchard.FileSystems.Dependencies { } public IEnumerable LoadDescriptors() { - return _cacheManager.Get(PersistencePath, + return _cacheManager.Get(PersistencePath, true, ctx => { _appDataFolder.CreateDirectory(BasePath); diff --git a/src/Orchard/FileSystems/Dependencies/DefaultExtensionDependenciesManager.cs b/src/Orchard/FileSystems/Dependencies/DefaultExtensionDependenciesManager.cs index c917c679d..afde10403 100644 --- a/src/Orchard/FileSystems/Dependencies/DefaultExtensionDependenciesManager.cs +++ b/src/Orchard/FileSystems/Dependencies/DefaultExtensionDependenciesManager.cs @@ -5,6 +5,7 @@ using System.Xml.Linq; using Orchard.Caching; using Orchard.FileSystems.AppData; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.FileSystems.Dependencies { /// @@ -64,7 +65,7 @@ namespace Orchard.FileSystems.Dependencies { } public IEnumerable LoadDescriptors() { - return _cacheManager.Get(PersistencePath, ctx => { + return _cacheManager.Get(PersistencePath, true, ctx => { _appDataFolder.CreateDirectory(BasePath); if (!DisableMonitoring) { @@ -135,8 +136,11 @@ namespace Orchard.FileSystems.Dependencies { return XDocument.Load(stream); } } - catch (Exception e) { - Logger.Information(e, "Error reading file '{0}'. Assuming empty.", persistancePath); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Information(ex, "Error reading file '{0}'. Assuming empty.", persistancePath); return new XDocument(); } } diff --git a/src/Orchard/FileSystems/Media/FileSystemStorageProvider.cs b/src/Orchard/FileSystems/Media/FileSystemStorageProvider.cs index e4ad40152..214de9205 100644 --- a/src/Orchard/FileSystems/Media/FileSystemStorageProvider.cs +++ b/src/Orchard/FileSystems/Media/FileSystemStorageProvider.cs @@ -6,6 +6,7 @@ using System.Web.Hosting; using Orchard.Environment.Configuration; using Orchard.Localization; using Orchard.Validation; +using Orchard.Exceptions; namespace Orchard.FileSystems.Media { public class FileSystemStorageProvider : IStorageProvider { @@ -155,6 +156,9 @@ namespace Orchard.FileSystems.Media { directoryInfo.Create(); } catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } throw new ArgumentException(T("The folder could not be created at path: {0}. {1}", path, ex).ToString()); } } diff --git a/src/Orchard/FileSystems/VirtualPath/DefaultVirtualPathProvider.cs b/src/Orchard/FileSystems/VirtualPath/DefaultVirtualPathProvider.cs index 85a295d1b..4c2941e91 100644 --- a/src/Orchard/FileSystems/VirtualPath/DefaultVirtualPathProvider.cs +++ b/src/Orchard/FileSystems/VirtualPath/DefaultVirtualPathProvider.cs @@ -5,6 +5,7 @@ using System.Linq; using System.Web; using System.Web.Hosting; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.FileSystems.VirtualPath { public class DefaultVirtualPathProvider : IVirtualPathProvider { @@ -59,9 +60,12 @@ namespace Orchard.FileSystems.VirtualPath { } return result; } - catch (Exception e) { + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } // The initial path might have been invalid (e.g. path indicates a path outside the application root) - Logger.Information(e, "Path '{0}' cannot be made app relative", virtualPath); + Logger.Information(ex, "Path '{0}' cannot be made app relative", virtualPath); return null; } } @@ -156,8 +160,11 @@ namespace Orchard.FileSystems.VirtualPath { try { return FileExists(virtualPath); } - catch (Exception e) { - Logger.Information(e, "File '{0}' can not be checked for existence. Assuming doesn't exist.", virtualPath); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Information(ex, "File '{0}' can not be checked for existence. Assuming doesn't exist.", virtualPath); return false; } } diff --git a/src/Orchard/IBackgroundHttpContextFactory.cs b/src/Orchard/IBackgroundHttpContextFactory.cs new file mode 100644 index 000000000..1437518f8 --- /dev/null +++ b/src/Orchard/IBackgroundHttpContextFactory.cs @@ -0,0 +1,8 @@ +using System.Web; + +namespace Orchard { + public interface IBackgroundHttpContextFactory : IDependency { + HttpContext CreateHttpContext(); + void InitializeHttpContext(); + } +} diff --git a/src/Orchard/IStaticHttpContextScopeFactory.cs b/src/Orchard/IStaticHttpContextScopeFactory.cs deleted file mode 100644 index bec2219e5..000000000 --- a/src/Orchard/IStaticHttpContextScopeFactory.cs +++ /dev/null @@ -1,15 +0,0 @@ -using System; - -namespace Orchard { - /// - /// A factory class that creates an instance and initializes the HttpContext.Current property with that instance until the scope is disposed of. - /// This is useful when rendering views from a background thread, as some Html Helpers access HttpContext.Current directly, thus preventing a NullReferenceException. - /// - public interface IStaticHttpContextScopeFactory : IDependency { - /// - /// Creates a disposable static HttpContext scope. This is safe to use even if there is an actual HttpContext.Current instance. - /// - /// - IDisposable CreateStaticScope(); - } -} diff --git a/src/Orchard/Localization/Services/DefaultCultureManager.cs b/src/Orchard/Localization/Services/DefaultCultureManager.cs index 9026e4acb..19ffa05b8 100644 --- a/src/Orchard/Localization/Services/DefaultCultureManager.cs +++ b/src/Orchard/Localization/Services/DefaultCultureManager.cs @@ -26,7 +26,7 @@ namespace Orchard.Localization.Services { } public IEnumerable ListCultures() { - return _cacheManager.Get("Cultures", context => { + return _cacheManager.Get("Cultures", true, context => { context.Monitor(_signals.When("culturesChanged")); return _cultureRepository.Table.Select(o => o.Culture).ToList(); diff --git a/src/Orchard/Localization/Services/DefaultLocalizedStringManager.cs b/src/Orchard/Localization/Services/DefaultLocalizedStringManager.cs index cb9516399..e0a2a29de 100644 --- a/src/Orchard/Localization/Services/DefaultLocalizedStringManager.cs +++ b/src/Orchard/Localization/Services/DefaultLocalizedStringManager.cs @@ -90,7 +90,7 @@ namespace Orchard.Localization.Services { // Cache entry will be invalidated any time the directories hosting // the .po files are modified. private CultureDictionary LoadCulture(string culture) { - return _cacheManager.Get(culture, ctx => { + return _cacheManager.Get(culture, true, ctx => { ctx.Monitor(_signals.When("culturesChanged")); return new CultureDictionary { CultureName = culture, diff --git a/src/Orchard/Messaging/Services/DefaultMessageManager.cs b/src/Orchard/Messaging/Services/DefaultMessageManager.cs index 8e2911857..1495f9e37 100644 --- a/src/Orchard/Messaging/Services/DefaultMessageManager.cs +++ b/src/Orchard/Messaging/Services/DefaultMessageManager.cs @@ -5,6 +5,7 @@ using Orchard.Logging; using Orchard.Messaging.Events; using Orchard.Messaging.Models; using Orchard.ContentManagement.Records; +using Orchard.Exceptions; namespace Orchard.Messaging.Services { [Obsolete] @@ -40,8 +41,11 @@ namespace Orchard.Messaging.Services { PrepareAndSend(type, properties, context); } - catch ( Exception e ) { - Logger.Error(e, "An error occured while sending the message {0}", type); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error(ex, "An error occured while sending the message {0}", type); } } @@ -60,8 +64,11 @@ namespace Orchard.Messaging.Services { PrepareAndSend(type, properties, context); } - catch (Exception e) { - Logger.Error(e, "An error occured while sending the message {0}", type); + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error(ex, "An error occured while sending the message {0}", type); } } diff --git a/src/Orchard/Mvc/Extensions/HttpContextBaseExtensions.cs b/src/Orchard/Mvc/Extensions/HttpContextBaseExtensions.cs new file mode 100644 index 000000000..9a7864887 --- /dev/null +++ b/src/Orchard/Mvc/Extensions/HttpContextBaseExtensions.cs @@ -0,0 +1,9 @@ +using System.Web; + +namespace Orchard.Mvc.Extensions { + public static class HttpContextBaseExtensions { + public static bool IsBackgroundContext(this HttpContextBase httpContextBase) { + return httpContextBase == null || httpContextBase is MvcModule.HttpContextPlaceholder; + } + } +} diff --git a/src/Orchard/Mvc/Extensions/HttpContextExtensions.cs b/src/Orchard/Mvc/Extensions/HttpContextExtensions.cs deleted file mode 100644 index f12ebe884..000000000 --- a/src/Orchard/Mvc/Extensions/HttpContextExtensions.cs +++ /dev/null @@ -1,13 +0,0 @@ -using System.Web; - -namespace Orchard.Mvc.Extensions { - public static class HttpContextExtensions { - public static bool IsBackgroundContext(this HttpContextBase httpContext) { - return httpContext == null || httpContext is MvcModule.HttpContextPlaceholder; - } - - public static bool IsBackgroundContext(this HttpContext httpContext) { - return httpContext == null || httpContext.Items.Contains(StaticHttpContextScopeFactory.IsBackgroundHttpContextKey); - } - } -} diff --git a/src/Orchard/Mvc/Html/LinkExtensions.cs b/src/Orchard/Mvc/Html/LinkExtensions.cs new file mode 100644 index 000000000..34db0c758 --- /dev/null +++ b/src/Orchard/Mvc/Html/LinkExtensions.cs @@ -0,0 +1,144 @@ +using System; +using System.Collections.Generic; +using System.Reflection; +using System.Web; +using System.Web.Mvc; +using System.Web.Routing; +using System.Web.WebPages; + +namespace Orchard.Mvc.Html { + public static class LinkExtensions { + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName) { + return ActionLink(htmlHelper, linkText, actionName, null /* controllerName */, new RouteValueDictionary(), new RouteValueDictionary()); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, object routeValues) { + return ActionLink(htmlHelper, linkText, actionName, null /* controllerName */, ObjectToDictionary(routeValues), new RouteValueDictionary()); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, object routeValues, object htmlAttributes) { + return ActionLink(htmlHelper, linkText, actionName, null /* controllerName */, ObjectToDictionary(routeValues), HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, RouteValueDictionary routeValues) { + return ActionLink(htmlHelper, linkText, actionName, null /* controllerName */, routeValues, new RouteValueDictionary()); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + return ActionLink(htmlHelper, linkText, actionName, null /* controllerName */, routeValues, htmlAttributes); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, string controllerName) { + return ActionLink(htmlHelper, linkText, actionName, controllerName, new RouteValueDictionary(), new RouteValueDictionary()); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, string controllerName, object routeValues, object htmlAttributes) { + return ActionLink(htmlHelper, linkText, actionName, controllerName, ObjectToDictionary(routeValues), HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, string controllerName, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + if (String.IsNullOrEmpty(linkText.ToString())) { + throw new ArgumentException("Argument must be a non empty string", "linkText"); + } + return MvcHtmlString.Create(GenerateLink(htmlHelper.ViewContext.RequestContext, htmlHelper.RouteCollection, linkText, null /* routeName */, actionName, controllerName, routeValues, htmlAttributes)); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, string controllerName, string protocol, string hostName, string fragment, object routeValues, object htmlAttributes) { + return ActionLink(htmlHelper, linkText, actionName, controllerName, protocol, hostName, fragment, ObjectToDictionary(routeValues), HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); + } + + public static MvcHtmlString ActionLink(this HtmlHelper htmlHelper, IHtmlString linkText, string actionName, string controllerName, string protocol, string hostName, string fragment, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + if (String.IsNullOrEmpty(linkText.ToString())) { + throw new ArgumentException("Argument must be a non empty string", "linkText"); + } + return MvcHtmlString.Create(GenerateLink(htmlHelper.ViewContext.RequestContext, htmlHelper.RouteCollection, linkText, null /* routeName */, actionName, controllerName, protocol, hostName, fragment, routeValues, htmlAttributes)); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, object routeValues) { + return RouteLink(htmlHelper, linkText, ObjectToDictionary(routeValues)); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, RouteValueDictionary routeValues) { + return RouteLink(htmlHelper, linkText, routeValues, new RouteValueDictionary()); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, string routeName) { + return RouteLink(htmlHelper, linkText, routeName, (object)null /* routeValues */); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, string routeName, object routeValues) { + return RouteLink(htmlHelper, linkText, routeName, ObjectToDictionary(routeValues)); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, string routeName, RouteValueDictionary routeValues) { + return RouteLink(htmlHelper, linkText, routeName, routeValues, new RouteValueDictionary()); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, object routeValues, object htmlAttributes) { + return RouteLink(htmlHelper, linkText, ObjectToDictionary(routeValues), HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + return RouteLink(htmlHelper, linkText, null /* routeName */, routeValues, htmlAttributes); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, string routeName, object routeValues, object htmlAttributes) { + return RouteLink(htmlHelper, linkText, routeName, ObjectToDictionary(routeValues), HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, string routeName, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + if (String.IsNullOrEmpty(linkText.ToString())) { + throw new ArgumentException("Argument must be a non empty string", "linkText"); + } + return MvcHtmlString.Create(GenerateRouteLink(htmlHelper.ViewContext.RequestContext, htmlHelper.RouteCollection, linkText, routeName, routeValues, htmlAttributes)); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, string routeName, string protocol, string hostName, string fragment, object routeValues, object htmlAttributes) { + return RouteLink(htmlHelper, linkText, routeName, protocol, hostName, fragment, ObjectToDictionary(routeValues), HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); + } + + public static MvcHtmlString RouteLink(this HtmlHelper htmlHelper, IHtmlString linkText, string routeName, string protocol, string hostName, string fragment, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + if (String.IsNullOrEmpty(linkText.ToString())) { + throw new ArgumentException("Argument must be a non empty string", "linkText"); + } + return MvcHtmlString.Create(GenerateRouteLink(htmlHelper.ViewContext.RequestContext, htmlHelper.RouteCollection, linkText, routeName, protocol, hostName, fragment, routeValues, htmlAttributes)); + } + + public static string GenerateLink(RequestContext requestContext, RouteCollection routeCollection, IHtmlString linkText, string routeName, string actionName, string controllerName, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + return GenerateLink(requestContext, routeCollection, linkText, routeName, actionName, controllerName, null /* protocol */, null /* hostName */, null /* fragment */, routeValues, htmlAttributes); + } + + public static string GenerateLink(RequestContext requestContext, RouteCollection routeCollection, IHtmlString linkText, string routeName, string actionName, string controllerName, string protocol, string hostName, string fragment, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + return GenerateLinkInternal(requestContext, routeCollection, linkText, routeName, actionName, controllerName, protocol, hostName, fragment, routeValues, htmlAttributes, true /* includeImplicitMvcValues */); + } + + public static string GenerateRouteLink(RequestContext requestContext, RouteCollection routeCollection, IHtmlString linkText, string routeName, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + return GenerateRouteLink(requestContext, routeCollection, linkText, routeName, null /* protocol */, null /* hostName */, null /* fragment */, routeValues, htmlAttributes); + } + + public static string GenerateRouteLink(RequestContext requestContext, RouteCollection routeCollection, IHtmlString linkText, string routeName, string protocol, string hostName, string fragment, RouteValueDictionary routeValues, IDictionary htmlAttributes) { + return GenerateLinkInternal(requestContext, routeCollection, linkText, routeName, null /* actionName */, null /* controllerName */, protocol, hostName, fragment, routeValues, htmlAttributes, false /* includeImplicitMvcValues */); + } + private static string GenerateLinkInternal(RequestContext requestContext, RouteCollection routeCollection, IHtmlString linkText, string routeName, string actionName, string controllerName, string protocol, string hostName, string fragment, RouteValueDictionary routeValues, IDictionary htmlAttributes, bool includeImplicitMvcValues) { + string url = UrlHelper.GenerateUrl(routeName, actionName, controllerName, protocol, hostName, fragment, routeValues, routeCollection, requestContext, includeImplicitMvcValues); + TagBuilder tagBuilder = new TagBuilder("a") { + InnerHtml = linkText.ToString() + }; + tagBuilder.MergeAttributes(htmlAttributes); + tagBuilder.MergeAttribute("href", url); + return tagBuilder.ToString(TagRenderMode.Normal); + } + + private static RouteValueDictionary ObjectToDictionary(object value) { + RouteValueDictionary dictionary = new RouteValueDictionary(); + + if (value != null) { + foreach (var prop in value.GetType().GetProperties(BindingFlags.Public | BindingFlags.Instance)) { + dictionary.Add(prop.Name, prop.GetValue(value)); + } + } + + return dictionary; + } + } +} diff --git a/src/Orchard/Mvc/HttpContextAccessor.cs b/src/Orchard/Mvc/HttpContextAccessor.cs index 91c7c3b0d..b13d6fe25 100644 --- a/src/Orchard/Mvc/HttpContextAccessor.cs +++ b/src/Orchard/Mvc/HttpContextAccessor.cs @@ -1,40 +1,39 @@ using System; -using System.Collections.Concurrent; using System.Web; -using Autofac; -using Orchard.Mvc.Extensions; namespace Orchard.Mvc { public class HttpContextAccessor : IHttpContextAccessor { - readonly object _contextKey = new object(); - - [ThreadStatic] - static ConcurrentDictionary _threadStaticContexts; + private HttpContextBase _httpContext; public HttpContextBase Current() { - if (!HttpContext.Current.IsBackgroundContext()) - return new HttpContextWrapper(HttpContext.Current); - - return GetContext(); + var httpContext = GetStaticProperty(); + return !IsBackgroundHttpContext(httpContext) ? new HttpContextWrapper(httpContext) : _httpContext; } - public HttpContextBase CreateContext(ILifetimeScope lifetimeScope) { - return new MvcModule.HttpContextPlaceholder( - _threadStaticContexts, - _contextKey, - () => "http://localhost" // Use a valid URL always for the fake request. The value itself doesn't matter. - ); + public void Set(HttpContextBase httpContext) { + _httpContext = httpContext; } - private HttpContextBase GetContext() { - HttpContextBase context; - return ThreadStaticContexts.TryGetValue(_contextKey, out context) ? context : null; + private static bool IsBackgroundHttpContext(HttpContext httpContext) { + return httpContext == null || httpContext.Items.Contains(BackgroundHttpContextFactory.IsBackgroundHttpContextKey); } - static ConcurrentDictionary ThreadStaticContexts { - get { - return _threadStaticContexts ?? (_threadStaticContexts = new ConcurrentDictionary()); + private static HttpContext GetStaticProperty() { + var httpContext = HttpContext.Current; + if (httpContext == null) { + return null; } + + try { + // The "Request" property throws at application startup on IIS integrated pipeline mode. + if (httpContext.Request == null) { + return null; + } + } + catch (Exception) { + return null; + } + return httpContext; } } } \ No newline at end of file diff --git a/src/Orchard/Mvc/IHttpContextAccessor.cs b/src/Orchard/Mvc/IHttpContextAccessor.cs index f34462581..462887fee 100644 --- a/src/Orchard/Mvc/IHttpContextAccessor.cs +++ b/src/Orchard/Mvc/IHttpContextAccessor.cs @@ -1,9 +1,8 @@ using System.Web; -using Autofac; namespace Orchard.Mvc { public interface IHttpContextAccessor { HttpContextBase Current(); - HttpContextBase CreateContext(ILifetimeScope lifetimeScope); + void Set(HttpContextBase httpContext); } } diff --git a/src/Orchard/Mvc/MvcModule.cs b/src/Orchard/Mvc/MvcModule.cs index 5b0c3bbf4..f44c5cd4a 100644 --- a/src/Orchard/Mvc/MvcModule.cs +++ b/src/Orchard/Mvc/MvcModule.cs @@ -1,6 +1,5 @@ using System; using System.Collections; -using System.Collections.Concurrent; using System.Collections.Generic; using System.Collections.Specialized; using System.Globalization; @@ -10,9 +9,9 @@ using System.Web.Instrumentation; using System.Web.Mvc; using System.Web.Routing; using Autofac; -using Orchard.Mvc.Extensions; using Orchard.Mvc.Routes; using Orchard.Settings; +using Orchard.Exceptions; namespace Orchard.Mvc { public class MvcModule : Module { @@ -33,7 +32,11 @@ namespace Orchard.Mvc { // The "Request" property throws at application startup on IIS integrated pipeline mode. var req = HttpContext.Current.Request; } - catch (Exception) { + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + return false; } @@ -52,21 +55,15 @@ namespace Orchard.Mvc { // which requires activating the Site content item, which in turn requires a UrlHelper, which in turn requires a RequestContext, // thus preventing a StackOverflowException. var baseUrl = new Func(() => siteService.GetSiteSettings().BaseUrl); - var httpContextBase = context.Resolve().Current(); + var httpContextBase = new HttpContextPlaceholder(baseUrl); - if (httpContextBase == null) { - context.Resolve().CreateWorkContextScope(); - return context.Resolve().Current(); - } - - context.Resolve().CreateWorkContextScope(httpContextBase); return httpContextBase; } static RequestContext RequestContextFactory(IComponentContext context) { var httpContextAccessor = context.Resolve(); var httpContext = httpContextAccessor.Current(); - if (!httpContext.IsBackgroundContext()) { + if (httpContext != null) { var mvcHandler = httpContext.Handler as MvcHandler; if (mvcHandler != null) { @@ -93,23 +90,16 @@ namespace Orchard.Mvc { /// /// Standin context for background tasks. /// - public class HttpContextPlaceholder : HttpContextBase, IDisposable { + public class HttpContextPlaceholder : HttpContextBase { private readonly Lazy _baseUrl; private readonly IDictionary _items = new Dictionary(); - readonly Action _disposer; - public HttpContextPlaceholder(ConcurrentDictionary contexts, object contextKey, Func baseUrl) { + public HttpContextPlaceholder(Func baseUrl) { _baseUrl = new Lazy(baseUrl); - contexts.AddOrUpdate(contextKey, this, (a, b) => this); - - _disposer = () => { - HttpContextBase removedContext; - contexts.TryRemove(contextKey, out removedContext); - }; } public override HttpRequestBase Request { - get { return new HttpRequestPlaceholder(this, new Uri(_baseUrl.Value)); } + get { return new HttpRequestPlaceholder(new Uri(_baseUrl.Value)); } } public override IHttpHandler Handler { get; set; } @@ -118,10 +108,6 @@ namespace Orchard.Mvc { get { return new HttpResponsePlaceholder(); } } - public override HttpSessionStateBase Session { - get { return null; } - } - public override IDictionary Items { get { return _items; } } @@ -141,10 +127,6 @@ namespace Orchard.Mvc { public override object GetService(Type serviceType) { return null; } - - public void Dispose() { - _disposer(); - } } public class HttpResponsePlaceholder : HttpResponseBase { @@ -163,12 +145,9 @@ namespace Orchard.Mvc { /// standin context for background tasks. /// public class HttpRequestPlaceholder : HttpRequestBase { - private readonly HttpContextBase _httpContext; private readonly Uri _uri; - private RequestContext _requestContext; - public HttpRequestPlaceholder(HttpContextBase httpContext, Uri uri) { - _httpContext = httpContext; + public HttpRequestPlaceholder(Uri uri) { _uri = uri; } @@ -227,7 +206,7 @@ namespace Orchard.Mvc { return new NameValueCollection { { "SERVER_PORT", _uri.Port.ToString(CultureInfo.InvariantCulture) }, { "HTTP_HOST", _uri.Authority.ToString(CultureInfo.InvariantCulture) }, - + }; } } @@ -269,16 +248,6 @@ namespace Orchard.Mvc { return new HttpBrowserCapabilitiesPlaceholder(); } } - - public override RequestContext RequestContext { - get { - if (_requestContext == null) { - _requestContext = new RequestContext(_httpContext, new RouteData()); - } - return _requestContext; - } - set { _requestContext = value; } - } } public class HttpBrowserCapabilitiesPlaceholder : HttpBrowserCapabilitiesBase { diff --git a/src/Orchard/Orchard.Framework.csproj b/src/Orchard/Orchard.Framework.csproj index 0cdfdf839..877afb7cf 100644 --- a/src/Orchard/Orchard.Framework.csproj +++ b/src/Orchard/Orchard.Framework.csproj @@ -56,6 +56,7 @@ 4 AllRules.ruleset false + 0436 @@ -149,12 +150,14 @@ + + + - - + @@ -186,12 +189,11 @@ + - - @@ -298,10 +300,10 @@ - + - + @@ -362,7 +364,7 @@ - + @@ -401,7 +403,8 @@ - + + @@ -695,7 +698,6 @@ - diff --git a/src/Orchard/Properties/AssemblyInfo.cs b/src/Orchard/Properties/AssemblyInfo.cs index 9bc9c300a..aef96e848 100644 --- a/src/Orchard/Properties/AssemblyInfo.cs +++ b/src/Orchard/Properties/AssemblyInfo.cs @@ -34,6 +34,6 @@ using System.Security; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Orchard/Recipes/Services/IRecipeHarvester.cs b/src/Orchard/Recipes/Services/IRecipeHarvester.cs index c566bed59..3cea30e98 100644 --- a/src/Orchard/Recipes/Services/IRecipeHarvester.cs +++ b/src/Orchard/Recipes/Services/IRecipeHarvester.cs @@ -1,8 +1,24 @@ -using System.Collections.Generic; +using System; +using System.Collections.Generic; +using System.Linq; using Orchard.Recipes.Models; namespace Orchard.Recipes.Services { public interface IRecipeHarvester : IDependency { + /// + /// Returns a collection of all recipes. + /// + IEnumerable HarvestRecipes(); + + /// + /// Returns a collection of all recipes found in the specified extension. + /// IEnumerable HarvestRecipes(string extensionId); } + + public static class RecipeHarvesterExtensions { + public static Recipe GetRecipeByName(this IEnumerable recipes, string recipeName) { + return recipes.FirstOrDefault(r => r.Name.Equals(recipeName, StringComparison.OrdinalIgnoreCase)); + } + } } diff --git a/src/Orchard/StaticHttpContextScope.cs b/src/Orchard/StaticHttpContextScope.cs deleted file mode 100644 index b22b181db..000000000 --- a/src/Orchard/StaticHttpContextScope.cs +++ /dev/null @@ -1,17 +0,0 @@ -using System; -using System.Web; - -namespace Orchard { - public class StaticHttpContextScope : IDisposable { - private readonly HttpContext _previousHttpContext; - - public StaticHttpContextScope(HttpContext stub) { - _previousHttpContext = HttpContext.Current; - HttpContext.Current = stub; - } - - public void Dispose() { - HttpContext.Current = _previousHttpContext; - } - } -} \ No newline at end of file diff --git a/src/Orchard/StaticHttpContextScopeFactory.cs b/src/Orchard/StaticHttpContextScopeFactory.cs deleted file mode 100644 index 35aafda4d..000000000 --- a/src/Orchard/StaticHttpContextScopeFactory.cs +++ /dev/null @@ -1,31 +0,0 @@ -using System; -using System.IO; -using System.Web; -using Orchard.Settings; - -namespace Orchard { - public class StaticHttpContextScopeFactory : IStaticHttpContextScopeFactory { - private readonly Func _siteService; - public StaticHttpContextScopeFactory(Func siteService) { - _siteService = siteService; - } - - public const string IsBackgroundHttpContextKey = "IsBackgroundHttpContext"; - - public IDisposable CreateStaticScope() { - // If there already is a current HttpContext, use that one as the stub. - if(HttpContext.Current != null) - return new StaticHttpContextScope(HttpContext.Current); - - // We're in a background task (or some other static context like the console), - // so create a stub context so that Html Helpers can still be executed when rendering shapes in background tasks - // (sadly enought some Html Helpers access HttpContext.Current directly). - var url = _siteService().GetSiteSettings().BaseUrl; - var stub = new HttpContext(new HttpRequest("", url, ""), new HttpResponse(new StringWriter())); - - stub.Items[IsBackgroundHttpContextKey] = true; - - return new StaticHttpContextScope(stub); - } - } -} \ No newline at end of file diff --git a/src/Orchard/Tasks/BackgroundService.cs b/src/Orchard/Tasks/BackgroundService.cs index 18d231a3a..6093f917f 100644 --- a/src/Orchard/Tasks/BackgroundService.cs +++ b/src/Orchard/Tasks/BackgroundService.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using Orchard.Data; using Orchard.Environment.Configuration; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Tasks { @@ -18,7 +19,8 @@ namespace Orchard.Tasks { public BackgroundService( IEnumerable tasks, ITransactionManager transactionManager, - ShellSettings shellSettings) { + ShellSettings shellSettings, + IBackgroundHttpContextFactory backgroundHttpContextFactory) { _tasks = tasks; _transactionManager = transactionManager; @@ -38,9 +40,13 @@ namespace Orchard.Tasks { task.Sweep(); Logger.Information("Finished processing background task \"{0}\" on tenant \"{1}\".", taskName, _shellName); } - catch (Exception e) { + catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + _transactionManager.Cancel(); - Logger.Error(e, "Error while processing background task \"{0}\" on tenant \"{1}\".", taskName, _shellName); + Logger.Error(ex, "Error while processing background task \"{0}\" on tenant \"{1}\".", taskName, _shellName); } } } diff --git a/src/Orchard/Tasks/Locking/Migrations/FrameworkMigrations.cs b/src/Orchard/Tasks/Locking/Migrations/FrameworkMigrations.cs deleted file mode 100644 index ce27736a6..000000000 --- a/src/Orchard/Tasks/Locking/Migrations/FrameworkMigrations.cs +++ /dev/null @@ -1,24 +0,0 @@ -using System; -using Orchard.Data.Migration; - -namespace Orchard.Tasks.Locking.Migrations { - public class FrameworkMigrations : DataMigrationImpl { - - public int Create() { - SchemaBuilder.CreateTable("DistributedLockRecord", table => table - .Column("Id", column => column.PrimaryKey().Identity()) - .Column("Name", column => column.NotNull().WithLength(256)) - .Column("MachineName", column => column.WithLength(256)) - .Column("ThreadId", column => column.Nullable()) - .Column("Count") - .Column("CreatedUtc") - .Column("ValidUntilUtc", column => column.Nullable())); - - SchemaBuilder.AlterTable("DistributedLockRecord", table => { - table.CreateIndex("IDX_DistributedLockRecord_Name_ValidUntilUtc_Count", "Name", "ValidUntilUtc", "Count"); - }); - - return 1; - } - } -} \ No newline at end of file diff --git a/src/Orchard/Tasks/Locking/Records/DistributedLockRecord.cs b/src/Orchard/Tasks/Locking/Records/DistributedLockRecord.cs index ec8fbff96..5dee047b0 100644 --- a/src/Orchard/Tasks/Locking/Records/DistributedLockRecord.cs +++ b/src/Orchard/Tasks/Locking/Records/DistributedLockRecord.cs @@ -5,8 +5,6 @@ namespace Orchard.Tasks.Locking.Records { public virtual int Id { get; set; } public virtual string Name { get; set; } public virtual string MachineName { get; set; } - public virtual int? ThreadId { get; set; } - public virtual int Count { get; set; } public virtual DateTime CreatedUtc { get; set; } public virtual DateTime? ValidUntilUtc { get; set; } } diff --git a/src/Orchard/Tasks/Locking/Services/DistributedLock.cs b/src/Orchard/Tasks/Locking/Services/DistributedLock.cs index 05977bea4..2069c2181 100644 --- a/src/Orchard/Tasks/Locking/Services/DistributedLock.cs +++ b/src/Orchard/Tasks/Locking/Services/DistributedLock.cs @@ -1,46 +1,41 @@ using System; -using System.Threading; namespace Orchard.Tasks.Locking.Services { - /// - /// Represents a distributed lock. /> - /// - public class DistributedLock : IDisposable { - public static DistributedLock ForMachine(IDistributedLockService service, string name, string machineName, string lockId) { - return new DistributedLock { - _service = service, - Name = name, - MachineName = machineName, - Id = lockId - }; + public class DistributedLock : IDistributedLock { + + private readonly string _name; + private readonly string _internalName; + private readonly Action _releaseLockAction; + private int _count; + + internal DistributedLock(string name, string internalName, Action releaseLockAction) { + _name = name; + _internalName = internalName; + _releaseLockAction = releaseLockAction; + _count = 1; } - public static DistributedLock ForThread(IDistributedLockService service, string name, string machineName, int threadId, string lockId) { - return new DistributedLock { - _service = service, - Name = name, - MachineName = machineName, - ThreadId = threadId, - Id = lockId - }; + string IDistributedLock.Name { + get { + return _name; + } } - private IDistributedLockService _service; - private int _isDisposed; - - private DistributedLock() { + internal string InternalName { + get { + return _internalName; + } } - public string Id { get; private set; } - public string Name { get; private set; } - public string MachineName { get; private set; } - public int? ThreadId { get; private set; } + internal void Increment() { + _count++; + } - // This will be called at least and at the latest by the IoC container when the request ends. public void Dispose() { - if(Interlocked.CompareExchange(ref _isDisposed, 1, 0) == 0) - _service.ReleaseLock(this); + _count--; + if (_count == 0) + _releaseLockAction(); } } } \ No newline at end of file diff --git a/src/Orchard/Tasks/Locking/Services/DistributedLockSchemaBuilder.cs b/src/Orchard/Tasks/Locking/Services/DistributedLockSchemaBuilder.cs new file mode 100644 index 000000000..2f22f1220 --- /dev/null +++ b/src/Orchard/Tasks/Locking/Services/DistributedLockSchemaBuilder.cs @@ -0,0 +1,48 @@ +using System; +using Orchard.Data.Migration.Schema; +using Orchard.Environment.Configuration; + +namespace Orchard.Tasks.Locking.Services { + public class DistributedLockSchemaBuilder { + private readonly ShellSettings _shellSettings; + private readonly SchemaBuilder _schemaBuilder; + private const string TableName = "Orchard_Framework_DistributedLockRecord"; + + public DistributedLockSchemaBuilder(ShellSettings shellSettings, SchemaBuilder schemaBuilder) { + _shellSettings = shellSettings; + _schemaBuilder = schemaBuilder; + } + + public bool EnsureSchema() { + if (SchemaExists()) + return false; + + CreateSchema(); + return true; + } + + public void CreateSchema() { + _schemaBuilder.CreateTable(TableName, table => table + .Column("Id", column => column.PrimaryKey().Identity()) + .Column("Name", column => column.NotNull().WithLength(512).Unique()) + .Column("MachineName", column => column.WithLength(256)) + .Column("CreatedUtc") + .Column("ValidUntilUtc", column => column.Nullable())); + + _schemaBuilder.AlterTable(TableName, table => { + table.CreateIndex("IDX_DistributedLockRecord_Name", "Name"); + }); + } + + public bool SchemaExists() { + try { + var tablePrefix = String.IsNullOrEmpty(_shellSettings.DataTablePrefix) ? "" : _shellSettings.DataTablePrefix + "_"; + _schemaBuilder.ExecuteSql(String.Format("select * from {0}{1}", tablePrefix, TableName)); + return true; + } + catch { + return false; + } + } + } +} \ No newline at end of file diff --git a/src/Orchard/Tasks/Locking/Services/DistributedLockService.cs b/src/Orchard/Tasks/Locking/Services/DistributedLockService.cs index ed0fb21e0..fcd4a1c18 100644 --- a/src/Orchard/Tasks/Locking/Services/DistributedLockService.cs +++ b/src/Orchard/Tasks/Locking/Services/DistributedLockService.cs @@ -1,10 +1,13 @@ using System; +using System.Collections.Generic; using System.Data; using System.Linq; +using System.Threading; using System.Threading.Tasks; using Autofac; using Orchard.Data; using Orchard.Environment; +using Orchard.Environment.Configuration; using Orchard.Exceptions; using Orchard.Logging; using Orchard.Services; @@ -13,191 +16,199 @@ using Orchard.Tasks.Locking.Records; namespace Orchard.Tasks.Locking.Services { public class DistributedLockService : Component, IDistributedLockService { - private readonly IMachineNameProvider _machineNameProvider; + + private readonly IApplicationEnvironment _applicationEnvironment; private readonly ILifetimeScope _lifetimeScope; private readonly IClock _clock; - private readonly IThreadProvider _threadProvider; + private readonly ShellSettings _shellSettings; + private readonly Dictionary _locks; + private readonly TimeSpan _defaultRepeatInterval; - public DistributedLockService(IMachineNameProvider machineNameProvider, IThreadProvider threadProvider, ILifetimeScope lifetimeScope, IClock clock) { - _machineNameProvider = machineNameProvider; - _lifetimeScope = lifetimeScope; + public DistributedLockService( + IApplicationEnvironment applicationEnvironment, + ILifetimeScope lifetimeScope, + IClock clock, + ShellSettings shellSettings) { _clock = clock; - _threadProvider = threadProvider; + _lifetimeScope = lifetimeScope; + _shellSettings = shellSettings; + _applicationEnvironment = applicationEnvironment; + _locks = new Dictionary(); + _defaultRepeatInterval = TimeSpan.FromMilliseconds(500); } - public bool TryAcquireLockForMachine(string name, TimeSpan? maxValidFor, TimeSpan? timeout, out DistributedLock @lock) { - return TryAcquireLock(name, maxValidFor, timeout, GetMachineName(), null, out @lock); - } - - public DistributedLock AcquireLockForMachine(string name, TimeSpan? maxValidFor, TimeSpan? timeout) { - return AcquireLock(name, maxValidFor, timeout, GetMachineName(), null); - } - - public bool TryAcquireLockForThread(string name, TimeSpan? maxValidFor, TimeSpan? timeout, out DistributedLock @lock) { - return TryAcquireLock(name, maxValidFor, timeout, GetMachineName(), GetThreadId(), out @lock); - } - - public DistributedLock AcquireLockForThread(string name, TimeSpan? maxValidFor, TimeSpan? timeout) { - return AcquireLock(name, maxValidFor, timeout, GetMachineName(), GetThreadId()); - } - - public void ReleaseLock(DistributedLock @lock) { - var childLifetimeScope = CreateChildLifetimeScope(@lock.Name); - + public bool TryAcquireLock(string name, TimeSpan? maxValidFor, TimeSpan? timeout, out IDistributedLock dLock) { try { - var repository = childLifetimeScope.Resolve>(); - var transactionManager = childLifetimeScope.Resolve(); - transactionManager.RequireNew(IsolationLevel.ReadCommitted); - var lockId = Int32.Parse(@lock.Id); - var record = repository.Get(lockId); + dLock = AcquireLockInternal(name, maxValidFor, timeout, throwOnTimeout: false); - if (record == null) - throw new OrchardException(T("No lock record could be found for the specified lock to be released.")); - - if (record.Count <= 0) - throw new OrchardException(T("The specified lock has already been released.")); - - record.Count--; - - if(record.Count == 0) - repository.Delete(record); - } - catch (Exception ex) { - if (ex.IsFatal()) throw; - Logger.Error(ex, "An non-fatal error occurred while trying to dispose a distributed lock with name '{0}' and ID {1}.", @lock.Name, @lock.Id); - } - finally { - childLifetimeScope.Dispose(); - } - } - - private bool TryAcquireLock(string name, TimeSpan? maxValidFor, TimeSpan? timeout, string machineName, int? threadId, out DistributedLock @lock) { - @lock = AcquireLock(name, maxValidFor, machineName, threadId, timeout ?? TimeSpan.Zero); - return @lock != null; - } - - private DistributedLock AcquireLock(string name, TimeSpan? maxValidFor, TimeSpan? timeout, string machineName, int? threadId) { - var @lock = AcquireLock(name, maxValidFor, machineName, threadId, timeout); - if (@lock != null) - return @lock; - - throw new TimeoutException(String.Format("Failed to acquire a lock named '{0}' within the specified timeout ('{1}').", name, timeout)); - } - - private DistributedLock AcquireLock(string name, TimeSpan? maxValidFor, string machineName, int? threadId, TimeSpan? timeout = null) { - try { - DistributedLock @lock = null; - var acquired = Poll(() => (@lock = AcquireLockInternal(name, maxValidFor, machineName, threadId)) != null, timeout); - - if (acquired) { - Logger.Debug("Successfully acquired a lock named '{0}'.", name); - return @lock; + if (dLock != null) { + Logger.Debug("Successfully acquired lock '{0}'.", name); + return true; } + + Logger.Warning("Failed to acquire lock '{0}' within the specified timeout ({1}).", name, timeout); } catch (Exception ex) { - Logger.Error(ex, "Error while trying to acquire a lock named '{0}'.", name); + Logger.Error(ex, "Error while trying to acquire lock '{0}'.", name); + // TODO: Is it correct to not throw here? Should we instead ONLY swallow TimeoutException? + } + + dLock = null; + return false; + } + + public IDistributedLock AcquireLock(string name, TimeSpan? maxValidFor, TimeSpan? timeout) { + try { + DistributedLock result = AcquireLockInternal(name, maxValidFor, timeout, throwOnTimeout: true); + Logger.Debug("Successfully acquired lock '{0}'.", name); + return result; + } + catch (Exception ex) { + Logger.Error(ex, "Error while trying to acquire lock '{0}'.", name); throw; } - - Logger.Debug(timeout == null - ? "Failed to acquire a lock named '{0}'." - : "Failed to acquire a lock named '{0}' within the specified timeout ('{1}')." - , name, timeout); - - return null; } - private DistributedLock AcquireLockInternal(string name, TimeSpan? maxValidFor, string machineName, int? threadId) { - var childLifetimeScope = CreateChildLifetimeScope(name); + private DistributedLock AcquireLockInternal(string name, TimeSpan? maxValidFor, TimeSpan? timeout, bool throwOnTimeout) { + var internalName = GetInternalLockName(name); + var monitorTimeout = timeout.HasValue ? timeout.Value : TimeSpan.FromMilliseconds(-1); // -1 ms is .NET magic number for "infinite". + var monitorObj = String.Intern(String.Format("{0}:{1}", _applicationEnvironment.GetEnvironmentIdentifier(), internalName)); + + if (!Monitor.TryEnter(monitorObj, monitorTimeout)) { + Logger.Debug("Could not enter local monitor for lock '{0}' within the specified timeout ({1}).", internalName, timeout); + + if (throwOnTimeout) + throw new TimeoutException(String.Format("Failed to acquire lock '{0}' within the specified timeout ({1}).", internalName, timeout)); + + return null; + } + + Logger.Debug("Successfully entered local monitor for lock '{0}'.", internalName); try { - var transactionManager = childLifetimeScope.Resolve(); - transactionManager.RequireNew(IsolationLevel.ReadCommitted); + DistributedLock dLock = null; - // This way we can create a nested transaction scope instead of having the unwanted effect - // of manipulating the transaction of the caller. - var repository = childLifetimeScope.Resolve>(); - - // Find an existing, active lock, if any. - var record = repository.Table.FirstOrDefault(x => x.Name == name && (x.ValidUntilUtc == null || x.ValidUntilUtc >= _clock.UtcNow) && x.Count > 0); - - // The current owner name (based on machine name and current thread ID). - var canAcquireLock = false; - - // Check if there's already an active lock. - if (record != null) { - // Check if the machine name assigned to the lock is the one trying to acquire it. - if (record.MachineName == machineName) { - if (record.ThreadId != threadId) - throw new InvalidOperationException( - threadId == null - ? "An attempt to acquire a lock for a machine was detected while the requested lock is already assigned to a specific thread." - : "An attempt to acquire a lock for a thread was detected while the requested lock is already assigned to a machine."); - - record.Count++; - canAcquireLock = true; - } + // If there's already a distributed lock object in our dictionary, that means + // this acquisition is a reentrance. Use the existing lock object from the + // dictionary but increment its count. + if (_locks.TryGetValue(monitorObj, out dLock)) { + Logger.Debug("Current thread is re-entering lock '{0}'; incrementing count.", internalName); + dLock.Increment(); } else { - // No one has an active lock yet, so good to go. - record = new DistributedLockRecord { - Name = name, - MachineName = machineName, - ThreadId = threadId, - Count = 1, - CreatedUtc = _clock.UtcNow, - ValidUntilUtc = maxValidFor != null ? _clock.UtcNow + maxValidFor : null - }; - repository.Create(record); - canAcquireLock = true; + // No distributed lock object existed in our dictionary. Try to take ownership + // of database record until timeout expires, and if successful create a distributed + // lock object and add it to our dictionary. + var success = RepeatUntilTimeout(timeout, _defaultRepeatInterval, () => { + if (EnsureDistributedLockRecord(internalName, maxValidFor)) { + Logger.Debug("Record for lock '{0}' already owned by current machine or was successfully created; creating lock object.", internalName); + + dLock = new DistributedLock(name, internalName, releaseLockAction: () => { + Monitor.Exit(monitorObj); + DeleteDistributedLockRecord(internalName); + }); + + _locks.Add(monitorObj, dLock); + return true; + } + + return false; + }); + + if (!success) { + Logger.Debug("Record for lock '{0}' could not be created for current machine within the specified timeout ({1}).", internalName, timeout); + + if (throwOnTimeout) + throw new TimeoutException(String.Format("Failed to acquire lock '{0}' within the specified timeout ({1}).", internalName, timeout)); + + return null; + } } - if (!canAcquireLock) - return null; - - return threadId != null - ? DistributedLock.ForThread(this, name, machineName, threadId.Value, record.Id.ToString()) - : DistributedLock.ForMachine(this, name, machineName, record.Id.ToString()); + return dLock; } catch (Exception ex) { - Logger.Error(ex, "An error occurred while trying to acquire a lock."); + Monitor.Exit(monitorObj); + + Logger.Error(ex, "An error occurred while trying to acquire lock '{0}'.", internalName); throw; } - finally { - childLifetimeScope.Dispose(); + } + + private bool EnsureDistributedLockRecord(string internalName, TimeSpan? maxValidFor) { + var environmentIdentifier = _applicationEnvironment.GetEnvironmentIdentifier(); + var hasLockRecord = false; + + ExecuteOnSeparateTransaction(repository => { + // Try to find a valid lock record in the database. + var record = repository.Table.FirstOrDefault(x => x.Name == internalName && (x.ValidUntilUtc == null || x.ValidUntilUtc >= _clock.UtcNow)); + if (record == null) { + // No record existed, so we're good to create a new one. + Logger.Debug("No valid record was found for lock '{0}'; creating a new record.", internalName); + + repository.Create(new DistributedLockRecord { + Name = internalName, + MachineName = environmentIdentifier, + CreatedUtc = _clock.UtcNow, + ValidUntilUtc = maxValidFor.HasValue ? _clock.UtcNow + maxValidFor.Value : default(DateTime?) + }); + + hasLockRecord = true; + } + else if (record.MachineName == environmentIdentifier) { + // Existing lock was for correct machine name => lock record exists. + Logger.Debug("Found a valid record for lock '{0}' and current local machine name '{1}'.", internalName, environmentIdentifier); + hasLockRecord = true; + } + }); + + return hasLockRecord; + } + + private void DeleteDistributedLockRecord(string internalName) { + try { + ExecuteOnSeparateTransaction(repository => { + var record = repository.Table.FirstOrDefault(x => x.Name == internalName); + if (record == null) + throw new Exception(String.Format("No record could be found in the database for lock '{0}'.", internalName)); + repository.Delete(record); + Logger.Debug("Successfully deleted record for lock '{0}'.", internalName); + }); + } + catch (Exception ex) { + if (ex.IsFatal()) + throw; + Logger.Warning(ex, "An error occurred while deleting record for lock '{0}'.", internalName); } } - /// - /// Executes the specified function until it returns true, for the specified amount of time, or indefinitely if no timeout was given. - /// - /// The operation to repeatedly execute until it returns true. - /// The amount of time to retry executing the function. If null is specified, the specified function is executed indefinitely until it returns true. - /// Returns true if the specified function returned true within the specified timeout, false otherwise. - private bool Poll(Func operation, TimeSpan? timeout) { + private bool RepeatUntilTimeout(TimeSpan? timeout, TimeSpan repeatInterval, Func action) { + bool success; + var waitedTime = TimeSpan.Zero; - var waitTime = TimeSpan.FromMilliseconds(timeout.GetValueOrDefault().TotalMilliseconds / 10); - bool acquired; - - while (!(acquired = operation()) && (timeout == null || waitedTime < timeout.Value)) { - Task.Delay(waitTime).ContinueWith(t => { - waitedTime += waitTime; - }).Wait(); + while (!(success = action()) && (!timeout.HasValue || waitedTime < timeout.Value)) { + Task.Delay(repeatInterval).Wait(); + waitedTime += repeatInterval; } - return acquired; + return success; } - private string GetMachineName() { - return _machineNameProvider.GetMachineName(); + private void ExecuteOnSeparateTransaction(Action> action) { + if (action == null) + throw new ArgumentNullException(); + + using (var childLifetimeScope = _lifetimeScope.BeginLifetimeScope()) { + var repository = childLifetimeScope.Resolve>(); + var transactionManager = childLifetimeScope.Resolve(); + transactionManager.RequireNew(IsolationLevel.ReadCommitted); + action(repository); + } } - private int GetThreadId() { - return _threadProvider.GetCurrentThreadId(); - } - - private ILifetimeScope CreateChildLifetimeScope(string name) { - return _lifetimeScope.BeginLifetimeScope("Orchard.Tasks.Locking." + name); + private string GetInternalLockName(string name) { + // Prefix the requested lock name by a constant and the tenant name. + return String.Format("DistributedLock:{0}:{1}", _shellSettings.Name, name); } } } \ No newline at end of file diff --git a/src/Orchard/Tasks/Locking/Services/IDistributedLock.cs b/src/Orchard/Tasks/Locking/Services/IDistributedLock.cs new file mode 100644 index 000000000..a687728e1 --- /dev/null +++ b/src/Orchard/Tasks/Locking/Services/IDistributedLock.cs @@ -0,0 +1,13 @@ +using System; + +namespace Orchard.Tasks.Locking.Services { + + /// + /// Represents a distributed lock returned by IDistributedLockService. The owner of the + /// lock should call Dispose() on an instance of this interface to release the distributed + /// lock. + /// + public interface IDistributedLock : IDisposable { + string Name { get; } + } +} \ No newline at end of file diff --git a/src/Orchard/Tasks/Locking/Services/IDistributedLockService.cs b/src/Orchard/Tasks/Locking/Services/IDistributedLockService.cs index 31818f200..7de40e64f 100644 --- a/src/Orchard/Tasks/Locking/Services/IDistributedLockService.cs +++ b/src/Orchard/Tasks/Locking/Services/IDistributedLockService.cs @@ -1,128 +1,83 @@ using System; namespace Orchard.Tasks.Locking.Services { - /// - /// Provides distributed locking functionality. - /// - public interface IDistributedLockService : ISingletonDependency { - /// - /// Tries to acquire a lock on the specified name for the current machine. - /// - /// The name to use for the lock. - /// The maximum amount of time the lock is allowed. This is a safety net in case the caller fails to release the lock. - /// The amount of time to wait for the lock to be acquired before timing out. A null value will cause the method to return immedieately if no lock could be acquired. - /// The acquired lock. - /// Returns true if a lock was successfully acquired, false otherwise. - bool TryAcquireLockForMachine(string name, TimeSpan? maxValidFor, TimeSpan? timeout, out DistributedLock @lock); + + /// + /// Provides functionality to acquire and release tenant-wide locks which are + /// distributed across all instances in a web farm. + /// + /// + /// Distributed locks can be used to protect critical sections that should only ever + /// be executed by a single thread of execution across a whole web farm. The distributed + /// locks returned by this service are reentrant, i.e. the owner of a lock can reacquire it + /// multiple times, and must also release it (dispose of it) as many times as it was + /// acquired for the lock to be released. + /// + public interface IDistributedLockService : IDependency { /// - /// Acquires a lock with the specified parameters for the current machine. + /// Tries to acquire a named distributed lock within the current tenant. /// - /// The name to use for the lock. - /// The maximum amount of time the lock is allowed. This is a safety net in case the caller fails to release the lock. If null is specified, the lock never expires until it's released by the owner. - /// The amount of time to wait for the lock to be acquired before timing out. A null value will cause the method to block indefinitely until a lock can be acquired. - /// Returns a lock. - /// Throws a TimeoutException if no lock could be acquired in time. - DistributedLock AcquireLockForMachine(string name, TimeSpan? maxValidFor, TimeSpan? timeout); + /// The name of the lock to acquire. + /// The maximum amount of time the lock should be held before automatically expiring. This is a safeguard in case the owner fails to release the lock. If null is specified, the lock never automatically expires. + /// The amount of time to wait for the lock to be acquired. Passing TimeSpan.Zero will cause the method to return immediately. Passing null will cause the method to block indefinitely until a lock can be acquired. + /// This out parameter will be assigned the acquired lock if successful. + /// true if the lock was successfully acquired, otherwise false. + bool TryAcquireLock(string name, TimeSpan? maxValidFor, TimeSpan? timeout, out IDistributedLock @lock); - /// - /// Tries to acquire a lock on the specified name for the current thread. - /// - /// The name to use for the lock. - /// The maximum amount of time the lock is allowed. This is a safety net in case the caller fails to release the lock. If null is specified, the lock never expires until it's released by the owner. - /// The amount of time to wait for the lock to be acquired before timing out. A null value will cause the method to return immedieately if no lock could be acquired. - /// The acquired lock. - /// Returns true if a lock was successfully acquired, false otherwise. - bool TryAcquireLockForThread(string name, TimeSpan? maxValidFor, TimeSpan? timeout, out DistributedLock @lock); - - /// - /// Acquires a lock with the specified parameters for the current thread. - /// - /// The name to use for the lock. - /// The maximum amount of time the lock is allowed. This is a safety net in case the caller fails to release the lock. If null is specified, the lock never expires until it's released by the owner. - /// The amount of time to wait for the lock to be acquired before timing out. A null value will cause the method to block indefinitely until a lock can be acquired. - /// Returns a lock. - /// Throws a TimeoutException if no lock could be acquired in time. - DistributedLock AcquireLockForThread(string name, TimeSpan? maxValidFor, TimeSpan? timeout); - - /// - /// Disposes the specified lock. - /// - void ReleaseLock(DistributedLock @lock); + /// + /// Acquires a named distributed lock within the current tenant or throws if the lock cannot be acquired. + /// + /// The name of the lock to acquire. + /// The maximum amount of time the lock should be held before automatically expiring. This is a safeguard in case the owner fails to release the lock. If null is specified, the lock never automatically expires. + /// The amount of time to wait for the lock to be acquired. Passing TimeSpan.Zero will cause the method to return immediately. Passing null will cause the method to block indefinitely until a lock can be acquired. + /// The acquired lock. + /// This method throws a TimeoutException if the lock could not be acquired within the specified timeout period. + IDistributedLock AcquireLock(string name, TimeSpan? maxValidFor, TimeSpan? timeout); } public static class DistributedLockServiceExtensions { - /// - /// Tries to acquire a lock on the specified name for the current machine. - /// - /// The name to use for the lock. - /// The maximum amount of time the lock is allowed. This is a safety net in case the caller fails to release the lock. If null is specified, the lock never expires until it's released by the owner. - /// The acquired lock. - /// Returns true if a lock was successfully acquired, false otherwise. - public static bool TryAcquireLockForMachine(this IDistributedLockService service, string name, TimeSpan? maxValidFor, out DistributedLock @lock) { - return service.TryAcquireLockForMachine(name, maxValidFor, null, out @lock); + + /// + /// Tries to immediately acquire a named distributed lock with a given expiration time within the current tenant. + /// + /// The name of the lock to acquire. + /// The maximum amount of time the lock should be held before automatically expiring. This is a safeguard in case the owner fails to release the lock. If null is specified, the lock never automatically expires. + /// This out parameter will be assigned the acquired lock if successful. + /// true if the lock could be immediately acquired, otherwise false. + public static bool TryAcquireLock(this IDistributedLockService service, string name, TimeSpan? maxValidFor, out IDistributedLock @lock) { + return service.TryAcquireLock(name, maxValidFor, TimeSpan.Zero, out @lock); } - /// - /// Tries to acquire a lock on the specified name for the current machine. - /// - /// The name to use for the lock. - /// The acquired lock. - /// Returns true if a lock was successfully acquired, false otherwise. - public static bool TryAcquireLockForMachine(this IDistributedLockService service, string name, out DistributedLock @lock) { - return service.TryAcquireLockForMachine(name, null, null, out @lock); + /// + /// Tries to immediately acquire a named distributed lock with no expiration time within the current tenant. + /// + /// The name of the lock to acquire. + /// This out parameter will be assigned the acquired lock if successful. + /// true if the lock could be immediately acquired, otherwise false. + public static bool TryAcquireLock(this IDistributedLockService service, string name, out IDistributedLock @lock) { + return service.TryAcquireLock(name, null, TimeSpan.Zero, out @lock); } - /// - /// Acquires a lock with the specified parameters for the current machine. - /// - /// The name to use for the lock. - /// The maximum amount of time the lock is allowed. This is a safety net in case the caller fails to release the lock. If null is specified, the lock never expires until it's released by the owner. - /// Returns a lock. - /// Throws a TimeoutException if no lock could be acquired in time. - public static DistributedLock AcquireLockForMachine(this IDistributedLockService service, string name, TimeSpan? maxValidFor) { - return service.AcquireLockForMachine(name, maxValidFor, null); + /// + /// Acquires a named distributed lock with a given expiration time within the current tenant. + /// + /// The name of the lock to acquire. + /// The maximum amount of time the lock should be held before automatically expiring. This is a safeguard in case the owner fails to release the lock. If null is specified, the lock never automatically expires. + /// The acquired lock. + /// This method blocks indefinitely until the lock can be acquired. + public static IDistributedLock AcquireLock(this IDistributedLockService service, string name, TimeSpan? maxValidFor) { + return service.AcquireLock(name, maxValidFor, null); } - /// - /// Acquires a lock with the specified parameters for the current machine. - /// - /// The name to use for the lock. - /// Returns a lock. - /// Throws a TimeoutException if no lock could be acquired in time. - public static DistributedLock AcquireLockForMachine(this IDistributedLockService service, string name) { - return service.AcquireLockForMachine(name, null, null); - } - - /// - /// Tries to acquire a lock on the specified name for the current thread. - /// - /// The name to use for the lock. - /// The maximum amount of time the lock is allowed. This is a safety net in case the caller fails to release the lock. If null is specified, the lock never expires until it's released by the owner. - /// The acquired lock. - /// Returns true if a lock was successfully acquired, false otherwise. - public static bool TryAcquireLockForThread(this IDistributedLockService service, string name, TimeSpan? maxValidFor, out DistributedLock @lock) { - return service.TryAcquireLockForThread(name, maxValidFor, null, out @lock); - } - - /// - /// Tries to acquire a lock on the specified name for the current thread. - /// - /// The name to use for the lock. - /// The acquired lock. - /// Returns true if a lock was successfully acquired, false otherwise. - public static bool TryAcquireLockForThread(this IDistributedLockService service, string name, out DistributedLock @lock) { - return service.TryAcquireLockForThread(name, null, null, out @lock); - } - - /// - /// Acquires a lock with the specified parameters for the current thread. - /// - /// The name to use for the lock. - /// Returns a lock. - /// Throws a TimeoutException if no lock could be acquired in time. - public static DistributedLock AcquireLockForThread(this IDistributedLockService service, string name) { - return service.AcquireLockForThread(name, null, null); + /// + /// Acquires a named distributed lock with no expiration time within the current tenant. + /// + /// The name to use for the lock. + /// The acquired lock. + /// This method blocks indefinitely until the lock can be acquired. + public static IDistributedLock AcquireLock(this IDistributedLockService service, string name) { + return service.AcquireLock(name, null, null); } } } \ No newline at end of file diff --git a/src/Orchard/Tasks/SweepGenerator.cs b/src/Orchard/Tasks/SweepGenerator.cs index 9c61b6740..44429e50b 100644 --- a/src/Orchard/Tasks/SweepGenerator.cs +++ b/src/Orchard/Tasks/SweepGenerator.cs @@ -1,6 +1,7 @@ using System; using System.Timers; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Tasks { @@ -41,7 +42,7 @@ namespace Orchard.Tasks { } void Elapsed(object sender, ElapsedEventArgs e) { - // Current implementation disallows re-entrancy. + // current implementation disallows re-entrancy if (!System.Threading.Monitor.TryEnter(_timer)) return; @@ -51,6 +52,10 @@ namespace Orchard.Tasks { } } catch (Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Warning(ex, "Problem in background tasks"); } finally { @@ -60,7 +65,7 @@ namespace Orchard.Tasks { public void DoWork() { using (var scope = _workContextAccessor.CreateWorkContextScope()) { - // Resolve the manager and invoke it. + // resolve the manager and invoke it var manager = scope.Resolve(); manager.Sweep(); } diff --git a/src/Orchard/Time/SiteTimeZoneSelector.cs b/src/Orchard/Time/SiteTimeZoneSelector.cs index 28068b213..366718216 100644 --- a/src/Orchard/Time/SiteTimeZoneSelector.cs +++ b/src/Orchard/Time/SiteTimeZoneSelector.cs @@ -1,6 +1,7 @@ using System; using System.Web; using Orchard.Logging; +using Orchard.Exceptions; namespace Orchard.Time { /// @@ -31,8 +32,11 @@ namespace Orchard.Time { TimeZone = TimeZoneInfo.FindSystemTimeZoneById(siteTimeZoneId) }; } - catch(Exception e) { - Logger.Error(e, "TimeZone could not be loaded"); + catch(Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error(ex, "TimeZone could not be loaded"); // if the database could not be updated in time, ignore this provider return null; diff --git a/src/Orchard/UI/Admin/Notification/NotificationManager.cs b/src/Orchard/UI/Admin/Notification/NotificationManager.cs index 327b4f1c1..d0529224c 100644 --- a/src/Orchard/UI/Admin/Notification/NotificationManager.cs +++ b/src/Orchard/UI/Admin/Notification/NotificationManager.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using System.Linq; using Orchard.Logging; using Orchard.UI.Notify; +using Orchard.Exceptions; namespace Orchard.UI.Admin.Notification { public class NotificationManager : INotificationManager { @@ -22,8 +23,11 @@ namespace Orchard.UI.Admin.Notification { try { return n.GetNotifications(); } - catch(Exception e) { - Logger.Error("An unhandled exception was thrown while generating a notification: " + n.GetType(), e); + catch(Exception ex) { + if (ex.IsFatal()) { + throw; + } + Logger.Error("An unhandled exception was thrown while generating a notification: " + n.GetType(), ex); return Enumerable.Empty(); } }).ToList(); diff --git a/src/Orchard/UI/Resources/ResourceDefinition.cs b/src/Orchard/UI/Resources/ResourceDefinition.cs index f900661a1..6676a21eb 100644 --- a/src/Orchard/UI/Resources/ResourceDefinition.cs +++ b/src/Orchard/UI/Resources/ResourceDefinition.cs @@ -155,6 +155,10 @@ namespace Orchard.UI.Resources { return this; } + /// + /// Sets the version of the resource. + /// + /// The version to set, in the form of major.minor[.build[.revision]] public ResourceDefinition SetVersion(string version) { Version = version; return this; diff --git a/src/Orchard/Utility/Extensions/StringExtensions.cs b/src/Orchard/Utility/Extensions/StringExtensions.cs index 1e66a8248..08b43ea9a 100644 --- a/src/Orchard/Utility/Extensions/StringExtensions.cs +++ b/src/Orchard/Utility/Extensions/StringExtensions.cs @@ -16,8 +16,7 @@ namespace Orchard.Utility.Extensions { var sb = new StringBuilder(camel); for (int i = camel.Length-1; i>0; i--) { - var current = sb[i]; - if('A' <= current && current <= 'Z') { + if(char.IsUpper(sb[i])) { sb.Insert(i, ' '); } } @@ -357,4 +356,4 @@ namespace Orchard.Utility.Extensions { return Encoding.UTF8.GetString(Convert.FromBase64String(value)); } } -} \ No newline at end of file +} diff --git a/src/Orchard/WorkContext.cs b/src/Orchard/WorkContext.cs index 412278e2a..767719f3d 100644 --- a/src/Orchard/WorkContext.cs +++ b/src/Orchard/WorkContext.cs @@ -10,20 +10,35 @@ namespace Orchard { /// public abstract class WorkContext { /// - /// Resolves a registered dependency type + /// Resolves a registered dependency type. /// - /// The type of the dependency - /// An instance of the dependency if it could be resolved + /// The type of the dependency. + /// An instance of the dependency if it could be resolved. public abstract T Resolve(); /// - /// Tries to resolve a registered dependency type + /// Resolves a registered dependency type. /// - /// The type of the dependency - /// An instance of the dependency if it could be resolved - /// True if the dependency could be resolved, false otherwise + /// The type of the dependency. + /// An instance of the dependency if it could be resolved. + public abstract object Resolve(Type serviceType); + + /// + /// Tries to resolve a registered dependency type. + /// + /// The type of the dependency. + /// An instance of the dependency if it could be resolved. + /// True if the dependency could be resolved, false otherwise. public abstract bool TryResolve(out T service); + /// + /// Tries to resolve a registered dependency type. + /// + /// The type of the dependency. + /// An instance of the dependency if it could be resolved. + /// True if the dependency could be resolved, false otherwise. + public abstract bool TryResolve(Type serviceType, out object service); + public abstract T GetState(string name); public abstract void SetState(string name, T value); diff --git a/src/Package.json b/src/Package.json index 96a154825..577b26bf5 100644 --- a/src/Package.json +++ b/src/Package.json @@ -1,5 +1,7 @@ { + "private": true, "devDependencies": { + "fs": "^0.0.2", "glob": "^5.0.14", "path-posix": "^1.0.0", "merge-stream": "^0.1.8", diff --git a/src/Tools/MSBuild.Orchard.Tasks/Properties/AssemblyInfo.cs b/src/Tools/MSBuild.Orchard.Tasks/Properties/AssemblyInfo.cs index 76e1c21dc..c036003f8 100644 --- a/src/Tools/MSBuild.Orchard.Tasks/Properties/AssemblyInfo.cs +++ b/src/Tools/MSBuild.Orchard.Tasks/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Tools/Orchard.Tests/Properties/AssemblyInfo.cs b/src/Tools/Orchard.Tests/Properties/AssemblyInfo.cs index 52ea2beb4..b12d217b6 100644 --- a/src/Tools/Orchard.Tests/Properties/AssemblyInfo.cs +++ b/src/Tools/Orchard.Tests/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")] diff --git a/src/Tools/Orchard/Properties/AssemblyInfo.cs b/src/Tools/Orchard/Properties/AssemblyInfo.cs index fc9dad31d..ec5b96ed6 100644 --- a/src/Tools/Orchard/Properties/AssemblyInfo.cs +++ b/src/Tools/Orchard/Properties/AssemblyInfo.cs @@ -30,5 +30,5 @@ using System.Runtime.InteropServices; // You can specify all the values or you can default the Build and Revision Numbers // by using the '*' as shown below: -[assembly: AssemblyVersion("1.9.1")] -[assembly: AssemblyFileVersion("1.9.1")] +[assembly: AssemblyVersion("1.9.2")] +[assembly: AssemblyFileVersion("1.9.2")]