From 0c9b239bd377c4d89774abff7ba9cbeb51134bbd Mon Sep 17 00:00:00 2001 From: "Peter@KASSA01.lan" Date: Sat, 14 Jul 2012 08:04:58 -0200 Subject: [PATCH 1/7] fixing #30477 and #29003 --- .../umbraco/Trees/BaseMediaTree.cs | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/src/umbraco.presentation/umbraco/Trees/BaseMediaTree.cs b/src/umbraco.presentation/umbraco/Trees/BaseMediaTree.cs index 928223af6e..f9a440f219 100644 --- a/src/umbraco.presentation/umbraco/Trees/BaseMediaTree.cs +++ b/src/umbraco.presentation/umbraco/Trees/BaseMediaTree.cs @@ -1,4 +1,4 @@ -using System; +using System; using System.Collections.Generic; using System.Text; using umbraco.BasePages; @@ -87,8 +87,15 @@ function openMedia(id) { } else { - xNode.Action = null; - xNode.DimNode(); + if (dd.ContentType.Alias.ToLower() == "folder") + { + xNode.Action = "javascript:jQuery('.umbTree #" + dd.Id.ToString() + "').click();"; + } + else + { + xNode.Action = null; + xNode.Style.DimNode(); + } } } else From 63d6b05460be35824b983c1f0d682df908ec5c48 Mon Sep 17 00:00:00 2001 From: "Peter@KASSA01.lan" Date: Sun, 15 Jul 2012 10:53:34 -0200 Subject: [PATCH 2/7] modified 3 dialogs to escape language-items containing single quote when inserted in javascript --- .../umbraco/dialogs/insertTable.aspx | 2 +- .../umbraco/dialogs/insertTable.aspx.designer.cs | 9 +-------- src/umbraco.presentation/umbraco/dialogs/publish.aspx | 4 ++-- .../umbraco/dialogs/publish.aspx.designer.cs | 3 +-- src/umbraco.presentation/umbraco/dialogs/sort.aspx | 2 +- 5 files changed, 6 insertions(+), 14 deletions(-) diff --git a/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx b/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx index a00c055e93..3943cae364 100644 --- a/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx +++ b/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx @@ -284,7 +284,7 @@ function insertTable()   - ')) window.close();" value="<%=umbraco.ui.Text("cancel")%>">   + ')) window.close();" value="<%=umbraco.ui.Text("cancel")%>">   "> diff --git a/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx.designer.cs b/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx.designer.cs index 8441a41c3c..240ec98f6f 100644 --- a/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx.designer.cs +++ b/src/umbraco.presentation/umbraco/dialogs/insertTable.aspx.designer.cs @@ -1,22 +1,15 @@ //------------------------------------------------------------------------------ // // This code was generated by a tool. -// Runtime Version:2.0.50727.312 // // Changes to this file may cause incorrect behavior and will be lost if -// the code is regenerated. +// the code is regenerated. // //------------------------------------------------------------------------------ namespace umbraco.dialogs { - /// - /// insertTable class. - /// - /// - /// Auto-generated class. - /// public partial class insertTable { /// diff --git a/src/umbraco.presentation/umbraco/dialogs/publish.aspx b/src/umbraco.presentation/umbraco/dialogs/publish.aspx index bf1cbc2ce9..b402e89381 100644 --- a/src/umbraco.presentation/umbraco/dialogs/publish.aspx +++ b/src/umbraco.presentation/umbraco/dialogs/publish.aspx @@ -20,7 +20,7 @@ } function showPublication() { - var statusStr = '<%=umbraco.ui.Text("inProgressCounter")%>'; + var statusStr = '<%=umbraco.ui.Text("inProgressCounter").Replace("'", "\\'")%>'; document.getElementById("counter").innerHTML = statusStr.replace('%0%', '0').replace('%1%', pubTotal); document.getElementById('formDiv').style.display = 'none'; document.getElementById('animDiv').style.display = 'block'; @@ -37,7 +37,7 @@ } function updatePublicationDo(retVal) { - var statusStr = '<%=umbraco.ui.Text("inProgressCounter")%>'; + var statusStr = '<%=umbraco.ui.Text("inProgressCounter").Replace("'", "\\'")%>'; document.getElementById("counter").innerHTML = statusStr.replace('%0%', retVal).replace('%1%', pubTotal); setTimeout("updatePublication()", 200); } diff --git a/src/umbraco.presentation/umbraco/dialogs/publish.aspx.designer.cs b/src/umbraco.presentation/umbraco/dialogs/publish.aspx.designer.cs index d128614b83..d9ebc3eefe 100644 --- a/src/umbraco.presentation/umbraco/dialogs/publish.aspx.designer.cs +++ b/src/umbraco.presentation/umbraco/dialogs/publish.aspx.designer.cs @@ -1,10 +1,9 @@ //------------------------------------------------------------------------------ // // This code was generated by a tool. -// Runtime Version:2.0.50727.4200 // // Changes to this file may cause incorrect behavior and will be lost if -// the code is regenerated. +// the code is regenerated. // //------------------------------------------------------------------------------ diff --git a/src/umbraco.presentation/umbraco/dialogs/sort.aspx b/src/umbraco.presentation/umbraco/dialogs/sort.aspx index a2e281c90d..824634ee67 100644 --- a/src/umbraco.presentation/umbraco/dialogs/sort.aspx +++ b/src/umbraco.presentation/umbraco/dialogs/sort.aspx @@ -83,7 +83,7 @@

- + " /> or <%=umbraco.ui.Text("general", "cancel", this.getUser())%>

From 34b10f7499156fbfa63d31ac8a37bef858c7d2a0 Mon Sep 17 00:00:00 2001 From: TomvE Date: Tue, 17 Jul 2012 14:23:47 -0100 Subject: [PATCH 3/7] Only call System.Web.Security.Membership.GetUser once and keep result locally to improve performance of request handler for protected pages. Same thing for the currentPage attributes ID and Path. --- src/umbraco.presentation/requestHandler.cs | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/src/umbraco.presentation/requestHandler.cs b/src/umbraco.presentation/requestHandler.cs index 00655e6ac8..af13b54297 100644 --- a/src/umbraco.presentation/requestHandler.cs +++ b/src/umbraco.presentation/requestHandler.cs @@ -371,21 +371,23 @@ namespace umbraco { // Check access HttpContext.Current.Trace.Write("umbracoRequestHandler", "Access checking started"); if (currentPage != null) { - if ( - Access.IsProtected(int.Parse(currentPage.Attributes.GetNamedItem("id").Value), - currentPage.Attributes.GetNamedItem("path").Value)) { + int id = int.Parse(currentPage.Attributes.GetNamedItem("id").Value); + string path = currentPage.Attributes.GetNamedItem("path").Value; + + if (Access.IsProtected(id, path)) { HttpContext.Current.Trace.Write("umbracoRequestHandler", "Page protected"); + var user = System.Web.Security.Membership.GetUser(); - if (System.Web.Security.Membership.GetUser() == null || !library.IsLoggedOn()) { + if (user == null || !library.IsLoggedOn()) { HttpContext.Current.Trace.Write("umbracoRequestHandler", "Not logged in - redirecting to login page..."); - currentPage = umbracoContent.GetElementById(Access.GetLoginPage(currentPage.Attributes.GetNamedItem("path").Value).ToString()); + currentPage = umbracoContent.GetElementById(Access.GetLoginPage(path).ToString()); } else { - if (System.Web.Security.Membership.GetUser() != null && !Access.HasAccces(int.Parse(currentPage.Attributes.GetNamedItem("id").Value), System.Web.Security.Membership.GetUser().ProviderUserKey)) { + if (user != null && !Access.HasAccces(id, user.ProviderUserKey)) { HttpContext.Current.Trace.Write("umbracoRequestHandler", "Member has not access - redirecting to error page..."); - currentPage = content.Instance.XmlContent.GetElementById(Access.GetErrorPage(currentPage.Attributes.GetNamedItem("path").Value).ToString()); + currentPage = content.Instance.XmlContent.GetElementById(Access.GetErrorPage(path).ToString()); } } } else From 9f60e49a32c6a158f9ef36b19083f10b398865b8 Mon Sep 17 00:00:00 2001 From: "Matt@MBP13-PC" Date: Mon, 23 Jul 2012 06:59:26 -0100 Subject: [PATCH 4/7] Added missing project reference --- src/umbraco.editorControls/tags/DataEditor.cs | 4 +--- src/umbraco.editorControls/umbraco.editorControls.csproj | 4 ++++ 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/src/umbraco.editorControls/tags/DataEditor.cs b/src/umbraco.editorControls/tags/DataEditor.cs index 359d256052..93f2415b2c 100644 --- a/src/umbraco.editorControls/tags/DataEditor.cs +++ b/src/umbraco.editorControls/tags/DataEditor.cs @@ -1,13 +1,11 @@ using System; -using System.Configuration; using System.Collections; using System.Web.UI; using System.Web.UI.WebControls; using umbraco.BusinessLogic; -using umbraco.DataLayer; -using umbraco.presentation; using ClientDependency.Core.Controls; using ClientDependency.Core; +using umbraco.presentation; namespace umbraco.editorControls.tags { diff --git a/src/umbraco.editorControls/umbraco.editorControls.csproj b/src/umbraco.editorControls/umbraco.editorControls.csproj index 44579c7d1d..2ba0d8f22e 100644 --- a/src/umbraco.editorControls/umbraco.editorControls.csproj +++ b/src/umbraco.editorControls/umbraco.editorControls.csproj @@ -164,6 +164,10 @@ + + False + ..\umbraco.presentation\Bin\umbraco.dll + From c51267136caa272d5205e308ff11d335b4f2cce3 Mon Sep 17 00:00:00 2001 From: "Matt@MBP13-PC" Date: Mon, 23 Jul 2012 10:05:29 -0100 Subject: [PATCH 5/7] Upgraded tags datatype to use jQuery tags input plugin gor a nicer UI (still works the same under the hood) --- src/Umbraco.Web.UI/Umbraco.Web.UI.csproj | 10 + .../tags/css/jquery.tagsinput.css | 7 + .../tags/js/jquery.tagsinput.js | 354 ++++++++++++++++++ .../tags/js/jquery.tagsinput.min.js | 1 + .../TagsAutoCompleteHandler.ashx.cs | 18 +- src/umbraco.editorControls/tags/DataEditor.cs | 126 ++----- 6 files changed, 414 insertions(+), 102 deletions(-) create mode 100644 src/Umbraco.Web.UI/umbraco_client/tags/css/jquery.tagsinput.css create mode 100644 src/Umbraco.Web.UI/umbraco_client/tags/js/jquery.tagsinput.js create mode 100644 src/Umbraco.Web.UI/umbraco_client/tags/js/jquery.tagsinput.min.js diff --git a/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj b/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj index 6f4ac938d7..b2989adbc6 100644 --- a/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj +++ b/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj @@ -40,6 +40,10 @@ v4.0 true + + + + bin\ @@ -345,6 +349,11 @@ + + + + jquery.tagsinput.js + @@ -1789,6 +1798,7 @@ + diff --git a/src/Umbraco.Web.UI/umbraco_client/tags/css/jquery.tagsinput.css b/src/Umbraco.Web.UI/umbraco_client/tags/css/jquery.tagsinput.css new file mode 100644 index 0000000000..c595e249f9 --- /dev/null +++ b/src/Umbraco.Web.UI/umbraco_client/tags/css/jquery.tagsinput.css @@ -0,0 +1,7 @@ +div.tagsinput { border:1px solid #CCC; background: #FFF; padding:5px; width:300px; height:100px; overflow-y: auto;} +div.tagsinput span.tag { border: 1px solid #a5d24a; -moz-border-radius:2px; -webkit-border-radius:2px; display: block; float: left; padding: 5px; text-decoration:none; background: #cde69c; color: #638421; margin-right: 5px; margin-bottom:5px;font-family: helvetica; font-size:13px;} +div.tagsinput span.tag a { font-weight: bold; color: #82ad2b; text-decoration:none; font-size: 11px; } +div.tagsinput input { width:80px; margin:0px; font-family: helvetica; font-size: 13px; border:1px solid transparent; padding:5px; background: transparent; color: #000; outline:0px; margin-right:5px; margin-bottom:5px; } +div.tagsinput div { display:block; float: left; } +.tags_clear { clear: both; width: 100%; height: 0px; } +.not_valid {background: #FBD8DB !important; color: #90111A !important;} diff --git a/src/Umbraco.Web.UI/umbraco_client/tags/js/jquery.tagsinput.js b/src/Umbraco.Web.UI/umbraco_client/tags/js/jquery.tagsinput.js new file mode 100644 index 0000000000..dd39357e82 --- /dev/null +++ b/src/Umbraco.Web.UI/umbraco_client/tags/js/jquery.tagsinput.js @@ -0,0 +1,354 @@ +/* + + jQuery Tags Input Plugin 1.3.3 + + Copyright (c) 2011 XOXCO, Inc + + Documentation for this plugin lives here: + http://xoxco.com/clickable/jquery-tags-input + + Licensed under the MIT license: + http://www.opensource.org/licenses/mit-license.php + + ben@xoxco.com + +*/ + +(function($) { + + var delimiter = new Array(); + var tags_callbacks = new Array(); + $.fn.doAutosize = function(o){ + var minWidth = $(this).data('minwidth'), + maxWidth = $(this).data('maxwidth'), + val = '', + input = $(this), + testSubject = $('#'+$(this).data('tester_id')); + + if (val === (val = input.val())) {return;} + + // Enter new content into testSubject + var escaped = val.replace(/&/g, '&').replace(/\s/g,' ').replace(//g, '>'); + testSubject.html(escaped); + // Calculate new width + whether to change + var testerWidth = testSubject.width(), + newWidth = (testerWidth + o.comfortZone) >= minWidth ? testerWidth + o.comfortZone : minWidth, + currentWidth = input.width(), + isValidWidthChange = (newWidth < currentWidth && newWidth >= minWidth) + || (newWidth > minWidth && newWidth < maxWidth); + + // Animate width + if (isValidWidthChange) { + input.width(newWidth); + } + + + }; + $.fn.resetAutosize = function(options){ + // alert(JSON.stringify(options)); + var minWidth = $(this).data('minwidth') || options.minInputWidth || $(this).width(), + maxWidth = $(this).data('maxwidth') || options.maxInputWidth || ($(this).closest('.tagsinput').width() - options.inputPadding), + val = '', + input = $(this), + testSubject = $('').css({ + position: 'absolute', + top: -9999, + left: -9999, + width: 'auto', + fontSize: input.css('fontSize'), + fontFamily: input.css('fontFamily'), + fontWeight: input.css('fontWeight'), + letterSpacing: input.css('letterSpacing'), + whiteSpace: 'nowrap' + }), + testerId = $(this).attr('id')+'_autosize_tester'; + if(! $('#'+testerId).length > 0){ + testSubject.attr('id', testerId); + testSubject.appendTo('body'); + } + + input.data('minwidth', minWidth); + input.data('maxwidth', maxWidth); + input.data('tester_id', testerId); + input.css('width', minWidth); + }; + + $.fn.addTag = function(value,options) { + options = jQuery.extend({focus:false,callback:true},options); + this.each(function() { + var id = $(this).attr('id'); + + var tagslist = $(this).val().split(delimiter[id]); + if (tagslist[0] == '') { + tagslist = new Array(); + } + + value = jQuery.trim(value); + + if (options.unique) { + var skipTag = $(this).tagExist(value); + if(skipTag == true) { + //Marks fake input as not_valid to let styling it + $('#'+id+'_tag').addClass('not_valid'); + } + } else { + var skipTag = false; + } + + if (value !='' && skipTag != true) { + $('').addClass('tag').append( + $('').text(value).append('  '), + $('', { + href : '#', + title : 'Removing tag', + text : 'x' + }).click(function () { + return $('#' + id).removeTag(escape(value)); + }) + ).insertBefore('#' + id + '_addTag'); + + tagslist.push(value); + + $('#'+id+'_tag').val(''); + if (options.focus) { + $('#'+id+'_tag').focus(); + } else { + $('#'+id+'_tag').blur(); + } + + $.fn.tagsInput.updateTagsField(this,tagslist); + + if (options.callback && tags_callbacks[id] && tags_callbacks[id]['onAddTag']) { + var f = tags_callbacks[id]['onAddTag']; + f.call(this, value); + } + if(tags_callbacks[id] && tags_callbacks[id]['onChange']) + { + var i = tagslist.length; + var f = tags_callbacks[id]['onChange']; + f.call(this, $(this), tagslist[i-1]); + } + } + + }); + + return false; + }; + + $.fn.removeTag = function(value) { + value = unescape(value); + this.each(function() { + var id = $(this).attr('id'); + + var old = $(this).val().split(delimiter[id]); + + $('#'+id+'_tagsinput .tag').remove(); + str = ''; + for (i=0; i< old.length; i++) { + if (old[i]!=value) { + str = str + delimiter[id] +old[i]; + } + } + + $.fn.tagsInput.importTags(this,str); + + if (tags_callbacks[id] && tags_callbacks[id]['onRemoveTag']) { + var f = tags_callbacks[id]['onRemoveTag']; + f.call(this, value); + } + }); + + return false; + }; + + $.fn.tagExist = function(val) { + var id = $(this).attr('id'); + var tagslist = $(this).val().split(delimiter[id]); + return (jQuery.inArray(val, tagslist) >= 0); //true when tag exists, false when not + }; + + // clear all existing tags and import new ones from a string + $.fn.importTags = function(str) { + id = $(this).attr('id'); + $('#'+id+'_tagsinput .tag').remove(); + $.fn.tagsInput.importTags(this,str); + } + + $.fn.tagsInput = function(options) { + var settings = jQuery.extend({ + interactive:true, + defaultText:'add a tag', + minChars:0, + width:'300px', + height:'100px', + autocomplete: {selectFirst: false }, + 'hide':true, + 'delimiter':',', + 'unique':true, + removeWithBackspace:true, + placeholderColor:'#666666', + autosize: true, + comfortZone: 20, + inputPadding: 6*2 + },options); + + this.each(function() { + if (settings.hide) { + $(this).hide(); + } + var id = $(this).attr('id'); + if (!id || delimiter[$(this).attr('id')]) { + id = $(this).attr('id', 'tags' + new Date().getTime()).attr('id'); + } + + var data = jQuery.extend({ + pid:id, + real_input: '#'+id, + holder: '#'+id+'_tagsinput', + input_wrapper: '#'+id+'_addTag', + fake_input: '#'+id+'_tag' + },settings); + + delimiter[id] = data.delimiter; + + if (settings.onAddTag || settings.onRemoveTag || settings.onChange) { + tags_callbacks[id] = new Array(); + tags_callbacks[id]['onAddTag'] = settings.onAddTag; + tags_callbacks[id]['onRemoveTag'] = settings.onRemoveTag; + tags_callbacks[id]['onChange'] = settings.onChange; + } + + var markup = '
'; + + if (settings.interactive) { + markup = markup + ''; + } + + markup = markup + '
'; + + $(markup).insertAfter(this); + + $(data.holder).css('width',settings.width); + $(data.holder).css('min-height',settings.height); + $(data.holder).css('height','100%'); + + if ($(data.real_input).val()!='') { + $.fn.tagsInput.importTags($(data.real_input),$(data.real_input).val()); + } + if (settings.interactive) { + $(data.fake_input).val($(data.fake_input).attr('data-default')); + $(data.fake_input).css('color',settings.placeholderColor); + $(data.fake_input).resetAutosize(settings); + + $(data.holder).bind('click',data,function(event) { + $(event.data.fake_input).focus(); + }); + + $(data.fake_input).bind('focus',data,function(event) { + if ($(event.data.fake_input).val()==$(event.data.fake_input).attr('data-default')) { + $(event.data.fake_input).val(''); + } + $(event.data.fake_input).css('color','#000000'); + }); + + if (settings.autocomplete_url != undefined) { + autocomplete_options = {source: settings.autocomplete_url}; + for (attrname in settings.autocomplete) { + autocomplete_options[attrname] = settings.autocomplete[attrname]; + } + + if (jQuery.Autocompleter !== undefined) { + $(data.fake_input).autocomplete(settings.autocomplete_url, settings.autocomplete); + $(data.fake_input).bind('result',data,function(event,data,formatted) { + if (data) { + $('#'+id).addTag(data[0] + "",{focus:true,unique:(settings.unique)}); + } + }); + } else if (jQuery.ui.autocomplete !== undefined) { + $(data.fake_input).autocomplete(autocomplete_options); + $(data.fake_input).bind('autocompleteselect',data,function(event,ui) { + $(event.data.real_input).addTag(ui.item.value,{focus:true,unique:(settings.unique)}); + return false; + }); + } + + + } else { + // if a user tabs out of the field, create a new tag + // this is only available if autocomplete is not used. + $(data.fake_input).bind('blur',data,function(event) { + var d = $(this).attr('data-default'); + if ($(event.data.fake_input).val()!='' && $(event.data.fake_input).val()!=d) { + if( (event.data.minChars <= $(event.data.fake_input).val().length) && (!event.data.maxChars || (event.data.maxChars >= $(event.data.fake_input).val().length)) ) + $(event.data.real_input).addTag($(event.data.fake_input).val(),{focus:true,unique:(settings.unique)}); + } else { + $(event.data.fake_input).val($(event.data.fake_input).attr('data-default')); + $(event.data.fake_input).css('color',settings.placeholderColor); + } + return false; + }); + + } + // if user types a comma, create a new tag + $(data.fake_input).bind('keypress',data,function(event) { + if (event.which==event.data.delimiter.charCodeAt(0) || event.which==13 ) { + event.preventDefault(); + if( (event.data.minChars <= $(event.data.fake_input).val().length) && (!event.data.maxChars || (event.data.maxChars >= $(event.data.fake_input).val().length)) ) + $(event.data.real_input).addTag($(event.data.fake_input).val(),{focus:true,unique:(settings.unique)}); + $(event.data.fake_input).resetAutosize(settings); + return false; + } else if (event.data.autosize) { + $(event.data.fake_input).doAutosize(settings); + + } + }); + //Delete last tag on backspace + data.removeWithBackspace && $(data.fake_input).bind('keydown', function(event) + { + if(event.keyCode == 8 && $(this).val() == '') + { + event.preventDefault(); + var last_tag = $(this).closest('.tagsinput').find('.tag:last').text(); + var id = $(this).attr('id').replace(/_tag$/, ''); + last_tag = last_tag.replace(/[\s]+x$/, ''); + $('#' + id).removeTag(escape(last_tag)); + $(this).trigger('focus'); + } + }); + $(data.fake_input).blur(); + + //Removes the not_valid class when user changes the value of the fake input + if(data.unique) { + $(data.fake_input).keydown(function(event){ + if(event.keyCode == 8 || String.fromCharCode(event.which).match(/\w+|[áéíóúÁÉÍÓÚñÑ,/]+/)) { + $(this).removeClass('not_valid'); + } + }); + } + } // if settings.interactive + }); + + return this; + + }; + + $.fn.tagsInput.updateTagsField = function(obj,tagslist) { + var id = $(obj).attr('id'); + $(obj).val(tagslist.join(delimiter[id])); + }; + + $.fn.tagsInput.importTags = function(obj,val) { + $(obj).val(''); + var id = $(obj).attr('id'); + var tags = val.split(delimiter[id]); + for (i=0; i/g,">");g.html(h);var i=g.width(),j=i+b.comfortZone>=c?i+b.comfortZone:c,k=f.width(),l=j=c||j>c&&j").css({position:"absolute",top:-9999,left:-9999,width:"auto",fontSize:f.css("fontSize"),fontFamily:f.css("fontFamily"),fontWeight:f.css("fontWeight"),letterSpacing:f.css("letterSpacing"),whiteSpace:"nowrap"}),h=a(this).attr("id")+"_autosize_tester";if(!a("#"+h).length>0){g.attr("id",h);g.appendTo("body")}f.data("minwidth",c);f.data("maxwidth",d);f.data("tester_id",h);f.css("width",c)};a.fn.addTag=function(d,e){e=jQuery.extend({focus:false,callback:true},e);this.each(function(){var f=a(this).attr("id");var g=a(this).val().split(b[f]);if(g[0]==""){g=new Array}d=jQuery.trim(d);if(e.unique){var h=a(g).tagExist(d);if(h==true){a("#"+f+"_tag").addClass("not_valid")}}else{var h=false}if(d!=""&&h!=true){a("").addClass("tag").append(a("").text(d).append("  "),a("
",{href:"#",title:"Removing tag",text:"x"}).click(function(){return a("#"+f).removeTag(escape(d))})).insertBefore("#"+f+"_addTag");g.push(d);a("#"+f+"_tag").val("");if(e.focus){a("#"+f+"_tag").focus()}else{a("#"+f+"_tag").blur()}a.fn.tagsInput.updateTagsField(this,g);if(e.callback&&c[f]&&c[f]["onAddTag"]){var i=c[f]["onAddTag"];i.call(this,d)}if(c[f]&&c[f]["onChange"]){var j=g.length;var i=c[f]["onChange"];i.call(this,a(this),g[j-1])}}});return false};a.fn.removeTag=function(d){d=unescape(d);this.each(function(){var e=a(this).attr("id");var f=a(this).val().split(b[e]);a("#"+e+"_tagsinput .tag").remove();str="";for(i=0;i=0};a.fn.importTags=function(b){id=a(this).attr("id");a("#"+id+"_tagsinput .tag").remove();a.fn.tagsInput.importTags(this,b)};a.fn.tagsInput=function(d){var e=jQuery.extend({interactive:true,defaultText:"add a tag",minChars:0,width:"300px",height:"100px",autocomplete:{selectFirst:false},hide:true,delimiter:",",unique:true,removeWithBackspace:true,placeholderColor:"#666666",autosize:true,comfortZone:20,inputPadding:6*2},d);this.each(function(){if(e.hide){a(this).hide()}var d=a(this).attr("id");if(!d||b[a(this).attr("id")]){d=a(this).attr("id","tags"+(new Date).getTime()).attr("id")}var f=jQuery.extend({pid:d,real_input:"#"+d,holder:"#"+d+"_tagsinput",input_wrapper:"#"+d+"_addTag",fake_input:"#"+d+"_tag"},e);b[d]=f.delimiter;if(e.onAddTag||e.onRemoveTag||e.onChange){c[d]=new Array;c[d]["onAddTag"]=e.onAddTag;c[d]["onRemoveTag"]=e.onRemoveTag;c[d]["onChange"]=e.onChange}var g='
';if(e.interactive){g=g+''}g=g+'
';a(g).insertAfter(this);a(f.holder).css("width",e.width);a(f.holder).css("height",e.height);if(a(f.real_input).val()!=""){a.fn.tagsInput.importTags(a(f.real_input),a(f.real_input).val())}if(e.interactive){a(f.fake_input).val(a(f.fake_input).attr("data-default"));a(f.fake_input).css("color",e.placeholderColor);a(f.fake_input).resetAutosize(e);a(f.holder).bind("click",f,function(b){a(b.data.fake_input).focus()});a(f.fake_input).bind("focus",f,function(b){if(a(b.data.fake_input).val()==a(b.data.fake_input).attr("data-default")){a(b.data.fake_input).val("")}a(b.data.fake_input).css("color","#000000")});if(e.autocomplete_url!=undefined){autocomplete_options={source:e.autocomplete_url};for(attrname in e.autocomplete){autocomplete_options[attrname]=e.autocomplete[attrname]}if(jQuery.Autocompleter!==undefined){a(f.fake_input).autocomplete(e.autocomplete_url,e.autocomplete);a(f.fake_input).bind("result",f,function(b,c,f){if(c){a("#"+d).addTag(c[0]+"",{focus:true,unique:e.unique})}})}else if(jQuery.ui.autocomplete!==undefined){a(f.fake_input).autocomplete(autocomplete_options);a(f.fake_input).bind("autocompleteselect",f,function(b,c){a(b.data.real_input).addTag(c.item.value,{focus:true,unique:e.unique});return false})}}else{a(f.fake_input).bind("blur",f,function(b){var c=a(this).attr("data-default");if(a(b.data.fake_input).val()!=""&&a(b.data.fake_input).val()!=c){if(b.data.minChars<=a(b.data.fake_input).val().length&&(!b.data.maxChars||b.data.maxChars>=a(b.data.fake_input).val().length))a(b.data.real_input).addTag(a(b.data.fake_input).val(),{focus:true,unique:e.unique})}else{a(b.data.fake_input).val(a(b.data.fake_input).attr("data-default"));a(b.data.fake_input).css("color",e.placeholderColor)}return false})}a(f.fake_input).bind("keypress",f,function(b){if(b.which==b.data.delimiter.charCodeAt(0)||b.which==13){b.preventDefault();if(b.data.minChars<=a(b.data.fake_input).val().length&&(!b.data.maxChars||b.data.maxChars>=a(b.data.fake_input).val().length))a(b.data.real_input).addTag(a(b.data.fake_input).val(),{focus:true,unique:e.unique});a(b.data.fake_input).resetAutosize(e);return false}else if(b.data.autosize){a(b.data.fake_input).doAutosize(e)}});f.removeWithBackspace&&a(f.fake_input).bind("keydown",function(b){if(b.keyCode==8&&a(this).val()==""){b.preventDefault();var c=a(this).closest(".tagsinput").find(".tag:last").text();var d=a(this).attr("id").replace(/_tag$/,"");c=c.replace(/[\s]+x$/,"");a("#"+d).removeTag(escape(c));a(this).trigger("focus")}});a(f.fake_input).blur();if(f.unique){a(f.fake_input).keydown(function(b){if(b.keyCode==8||String.fromCharCode(b.which).match(/\w+|[áéíóúÁÉÍÓÚñÑ,/]+/)){a(this).removeClass("not_valid")}})}}});return this};a.fn.tagsInput.updateTagsField=function(c,d){var e=a(c).attr("id");a(c).val(d.join(b[e]))};a.fn.tagsInput.importTags=function(d,e){a(d).val("");var f=a(d).attr("id");var g=e.split(b[f]);for(i=0;i(); while (rr.Read()) { - context.Response.Write(rr.GetString("tag") + Environment.NewLine); + tagList.Add(rr.GetString("tag")); } - + context.Response.Write(returnJson + ? new JavaScriptSerializer().Serialize(tagList) + : string.Join(Environment.NewLine, tagList)); } catch (Exception ex) { diff --git a/src/umbraco.editorControls/tags/DataEditor.cs b/src/umbraco.editorControls/tags/DataEditor.cs index 93f2415b2c..a6bd91ceaf 100644 --- a/src/umbraco.editorControls/tags/DataEditor.cs +++ b/src/umbraco.editorControls/tags/DataEditor.cs @@ -1,5 +1,6 @@ using System; using System.Collections; +using System.Linq; using System.Web.UI; using System.Web.UI.WebControls; using umbraco.BusinessLogic; @@ -24,7 +25,6 @@ namespace umbraco.editorControls.tags _group = Prevalues["group"].ToString(); } - public CheckBoxList tagCheckList = new CheckBoxList(); public TextBox tagBox = new TextBox(); @@ -32,49 +32,32 @@ namespace umbraco.editorControls.tags public void Save() { - - CheckBoxList items = tagCheckList; int _nodeID; int.TryParse(_data.NodeId.ToString(), out _nodeID); string allTags = ""; - int tagId = 0; - //first clear out all items associated with this ID... umbraco.cms.businesslogic.Tags.Tag.RemoveTagsFromNode(_nodeID, _group); - //and now we add them again... - foreach (ListItem li in items.Items) + var items = tagBox.Text.Trim().Split(','); + foreach (var item in items) { - if (li.Selected) + var tagName = item.Trim(); + if(string.IsNullOrEmpty(tagName)) + continue; + + var tagId = cms.businesslogic.Tags.Tag.GetTagId(tagName, _group); + if(tagId == 0) + tagId = cms.businesslogic.Tags.Tag.AddTag(tagName, _group); + + if (tagId > 0) { - - if (li.Value == "0") - { - // NH 4.7.1 if a tag doesn't have an id associated we'll do a 2nd check in case the XHR request went wrong - // from Codeplex 30151 - tagId = cms.businesslogic.Tags.Tag.GetTagId(li.Text, _group); - if (tagId == 0) - tagId = cms.businesslogic.Tags.Tag.AddTag(li.Text, _group); - li.Value = tagId.ToString(); - } - else - { - int.TryParse(li.Value, out tagId); - } - - if (tagId > 0) - { - - umbraco.cms.businesslogic.Tags.Tag.AssociateTagToNode(_nodeID, tagId); - - tagId = 0; - allTags += "," + li.Text; - } + umbraco.cms.businesslogic.Tags.Tag.AssociateTagToNode(_nodeID, tagId); } } + //and just in case, we'll save the tags as plain text on the node itself... - _data.Value = allTags.Trim().Trim(','); + _data.Value = tagBox.Text; } public bool ShowLabel @@ -87,44 +70,6 @@ namespace umbraco.editorControls.tags get { return false; } } - public void tagBoxTextChange(object sender, EventArgs e) - { - try - { - if (tagBox.Text.Trim().Length > 0) - { - CheckBoxList items = tagCheckList; - - string[] tags = tagBox.Text.Trim().Trim(',').Split(','); - - - for (int i = 0; i < tags.Length; i++) - { - //if not found we'll get zero and handle that onsave instead... - int id = getTagId(tags[i], _group); - - //we don't want 2 of a kind... - if (items.Items.FindByText(tags[i].Trim()) == null) - { - ListItem li = new ListItem(tags[i], id.ToString()); - li.Selected = true; - items.Items.Add(li); - } - } - - //reset the textbox - tagBox.Text = ""; - - ScriptManager.GetCurrent(Page).SetFocus(tagBox); - } - - } - catch (Exception ex) - { - Log.Add(LogTypes.Debug, -1, ex.ToString()); - } - } - [Obsolete("use the umbraco.cms.businesslogic.Tags.Tag class instead")] public int getTagId(string tag, string group) @@ -137,9 +82,14 @@ namespace umbraco.editorControls.tags { base.OnInit(e); - ClientDependencyLoader.Instance.RegisterDependency("Application/JQuery/jquery.autocomplete.js", "UmbracoClient", ClientDependencyType.Javascript); + //ClientDependencyLoader.Instance.RegisterDependency("Application/JQuery/jquery.autocomplete.js", "UmbracoClient", ClientDependencyType.Javascript); + ClientDependencyLoader.Instance.RegisterDependency("ui/ui-lightness/jquery-ui.custom.css", "UmbracoClient", ClientDependencyType.Css); ClientDependencyLoader.Instance.RegisterDependency("css/umbracoGui.css", "UmbracoRoot", ClientDependencyType.Css); + ClientDependencyLoader.Instance.RegisterDependency("tags/css/jquery.tagsinput.css", "UmbracoClient", ClientDependencyType.Css); + ClientDependencyLoader.Instance.RegisterDependency("tags/js/jquery.tagsinput.min.js", "UmbracoClient", ClientDependencyType.Javascript); + + string _alias = ((umbraco.cms.businesslogic.datatype.DefaultData)_data).PropertyId.ToString(); //making sure that we have a ID for context @@ -163,41 +113,23 @@ namespace umbraco.editorControls.tags } } - - tagBox.ID = "tagBox_" + _alias; - tagBox.AutoPostBack = true; - tagBox.AutoCompleteType = AutoCompleteType.Disabled; - tagBox.TextChanged += new System.EventHandler(this.tagBoxTextChange); - tagBox.CssClass = "umbEditorTextField"; - - Button tagButton = new Button(); - tagButton.Click += new System.EventHandler(this.tagBoxTextChange); - tagButton.Text = "tag"; - - - tagCheckList.ID = "tagCheckList_" + _alias; + tagBox.ID = "tagBox2_" + _alias; + tagBox.CssClass = "umbEditorTextField umbTagBox"; if (!String.IsNullOrEmpty(pageId)) { var tags = umbraco.cms.businesslogic.Tags.Tag.GetTags(int.Parse(pageId), _group); - foreach (var t in tags) - { - ListItem li = new ListItem(t.TagCaption, t.Id.ToString()); - li.Selected = true; - tagCheckList.Items.Add(li); - } + tagBox.Text = string.Join(",", tags.Select(x => x.TagCaption)); } this.ContentTemplateContainer.Controls.Add(tagBox); - this.ContentTemplateContainer.Controls.Add(tagButton); - this.ContentTemplateContainer.Controls.Add(tagCheckList); string tagsAutoCompleteScript = - "jQuery(\"#" - + tagBox.ClientID + "\").autocomplete(\"" - + umbraco.IO.IOHelper.ResolveUrl(umbraco.IO.SystemDirectories.Umbraco) - + "/webservices/TagsAutoCompleteHandler.ashx\",{minChars: 2,max: 100, extraParams:{group:\"" + _group + "\",id:\"" + pageId + "\",rnd:\"" + DateTime.Now.Ticks + "\"}}).result(function(e, data){jQuery(\"#" + tagButton.ClientID + "\").trigger('click');});"; + "jQuery('.umbTagBox').tagsInput({ width: '400px', defaultText: 'Add a tag', minChars: 2, autocomplete_url: '" + + umbraco.IO.IOHelper.ResolveUrl(umbraco.IO.SystemDirectories.Umbraco) + + "/webservices/TagsAutoCompleteHandler.ashx?group=" + _group + "&id=" + pageId + "&rnd=" + + DateTime.Now.Ticks + "&format=json' });"; string tagsAutoCompleteInitScript = @@ -210,10 +142,8 @@ namespace umbraco.editorControls.tags if (Page.IsPostBack) { ScriptManager.RegisterClientScriptBlock(this, GetType(), ClientID + "_tags", tagsAutoCompleteScript, true); - } - } #endregion From fdcc81003da4f013fa0f5a49152e57f24211553c Mon Sep 17 00:00:00 2001 From: "Matt@MBP13-PC" Date: Mon, 23 Jul 2012 10:28:53 -0100 Subject: [PATCH 6/7] Added knockout js library --- src/Umbraco.Web.UI/Umbraco.Web.UI.csproj | 1 + .../umbraco_client/ui/knockout.js | 86 +++++++++++++++++++ 2 files changed, 87 insertions(+) create mode 100644 src/Umbraco.Web.UI/umbraco_client/ui/knockout.js diff --git a/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj b/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj index b2989adbc6..720ca29434 100644 --- a/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj +++ b/src/Umbraco.Web.UI/Umbraco.Web.UI.csproj @@ -926,6 +926,7 @@ + diff --git a/src/Umbraco.Web.UI/umbraco_client/ui/knockout.js b/src/Umbraco.Web.UI/umbraco_client/ui/knockout.js new file mode 100644 index 0000000000..107026da16 --- /dev/null +++ b/src/Umbraco.Web.UI/umbraco_client/ui/knockout.js @@ -0,0 +1,86 @@ +// Knockout JavaScript library v2.1.0 +// (c) Steven Sanderson - http://knockoutjs.com/ +// License: MIT (http://www.opensource.org/licenses/mit-license.php) + +(function(window,document,navigator,undefined){ +function m(w){throw w;}var n=void 0,p=!0,s=null,t=!1;function A(w){return function(){return w}};function E(w){function B(b,c,d){d&&c!==a.k.r(b)&&a.k.S(b,c);c!==a.k.r(b)&&a.a.va(b,"change")}var a="undefined"!==typeof w?w:{};a.b=function(b,c){for(var d=b.split("."),f=a,g=0;g",c[0];);return 4a.a.j(c,b[e])&&c.push(b[e]);return c},T:function(a,b){for(var a=a||[],c=[], +e=0,f=a.length;ea.length?t:a.substring(0,b.length)===b},eb:function(a,b){for(var c="return ("+a+")",e=0;e",""]||!d.indexOf("",""]||(!d.indexOf("",""]||[0,"",""];b="ignored
"+d[1]+b+d[2]+"
";for("function"==typeof window.innerShiv?c.appendChild(window.innerShiv(b)):c.innerHTML=b;d[0]--;)c=c.lastChild;c=a.a.L(c.lastChild.childNodes)}return c}; +a.a.Y=function(b,c){a.a.ga(b);if(c!==s&&c!==n)if("string"!=typeof c&&(c=c.toString()),"undefined"!=typeof jQuery)jQuery(b).html(c);else for(var d=a.a.pa(c),f=0;f"},Va:function(a,b){var c=d[a];c===n&&m(Error("Couldn't find any memo with ID "+a+". Perhaps it's already been unmemoized."));try{return c.apply(s,b||[]),p}finally{delete d[a]}},Wa:function(b,d){var e=[];c(b,e);for(var h=0,j=e.length;hc;c++)b=b();return b})};a.toJSON=function(b,c,e){b=a.Ta(b);return a.a.sa(b,c,e)}})();a.b("toJS",a.Ta);a.b("toJSON",a.toJSON);(function(){a.k={r:function(b){switch(a.a.o(b)){case "option":return b.__ko__hasDomDataOptionValue__===p?a.a.f.get(b,a.c.options.oa):b.getAttribute("value");case "select":return 0<=b.selectedIndex?a.k.r(b.options[b.selectedIndex]):n;default:return b.value}},S:function(b,c){switch(a.a.o(b)){case "option":switch(typeof c){case "string":a.a.f.set(b,a.c.options.oa, +n);"__ko__hasDomDataOptionValue__"in b&&delete b.__ko__hasDomDataOptionValue__;b.value=c;break;default:a.a.f.set(b,a.c.options.oa,c),b.__ko__hasDomDataOptionValue__=p,b.value="number"===typeof c?c:""}break;case "select":for(var d=b.options.length-1;0<=d;d--)if(a.k.r(b.options[d])==c){b.selectedIndex=d;break}break;default:if(c===s||c===n)c="";b.value=c}}}})();a.b("selectExtensions",a.k);a.b("selectExtensions.readValue",a.k.r);a.b("selectExtensions.writeValue",a.k.S);a.g=function(){function b(a,b){for(var d= +s;a!=d;)d=a,a=a.replace(c,function(a,c){return b[c]});return a}var c=/\@ko_token_(\d+)\@/g,d=/^[\_$a-z][\_$a-z0-9]*(\[.*?\])*(\.[\_$a-z][\_$a-z0-9]*(\[.*?\])*)*$/i,f=["true","false"];return{D:[],W:function(c){var e=a.a.w(c);if(3>e.length)return[];"{"===e.charAt(0)&&(e=e.substring(1,e.length-1));for(var c=[],d=s,f,k=0;k$/: +/^\s*ko\s+(.*\:.*)\s*$/,h=g?/^<\!--\s*\/ko\s*--\>$/:/^\s*\/ko\s*$/,j={ul:p,ol:p};a.e={C:{},childNodes:function(a){return b(a)?d(a):a.childNodes},ha:function(c){if(b(c))for(var c=a.e.childNodes(c),e=0,d=c.length;e"),t))}};a.c.uniqueName.gb=0;a.c.checked={init:function(b,c,d){a.a.n(b,"click",function(){var f;if("checkbox"==b.type)f=b.checked;else if("radio"==b.type&&b.checked)f=b.value;else return;var g=c();"checkbox"==b.type&&a.a.d(g)instanceof Array?(f=a.a.j(a.a.d(g),b.value), +b.checked&&0>f?g.push(b.value):!b.checked&&0<=f&&g.splice(f,1)):a.g.$(g,d,"checked",f,p)});"radio"==b.type&&!b.name&&a.c.uniqueName.init(b,A(p))},update:function(b,c){var d=a.a.d(c());"checkbox"==b.type?b.checked=d instanceof Array?0<=a.a.j(d,b.value):d:"radio"==b.type&&(b.checked=b.value==d)}};var F={"class":"className","for":"htmlFor"};a.c.attr={update:function(b,c){var d=a.a.d(c())||{},f;for(f in d)if("string"==typeof f){var g=a.a.d(d[f]),e=g===t||g===s||g===n;e&&b.removeAttribute(f);8>=a.a.ja&& +f in F?(f=F[f],e?b.removeAttribute(f):b[f]=g):e||b.setAttribute(f,g.toString())}}};a.c.hasfocus={init:function(b,c,d){function f(b){var e=c();a.g.$(e,d,"hasfocus",b,p)}a.a.n(b,"focus",function(){f(p)});a.a.n(b,"focusin",function(){f(p)});a.a.n(b,"blur",function(){f(t)});a.a.n(b,"focusout",function(){f(t)})},update:function(b,c){var d=a.a.d(c());d?b.focus():b.blur();a.a.va(b,d?"focusin":"focusout")}};a.c["with"]={p:function(b){return function(){var c=b();return{"if":c,data:c,templateEngine:a.q.K}}}, +init:function(b,c){return a.c.template.init(b,a.c["with"].p(c))},update:function(b,c,d,f,g){return a.c.template.update(b,a.c["with"].p(c),d,f,g)}};a.g.D["with"]=t;a.e.C["with"]=p;a.c["if"]={p:function(b){return function(){return{"if":b(),templateEngine:a.q.K}}},init:function(b,c){return a.c.template.init(b,a.c["if"].p(c))},update:function(b,c,d,f,g){return a.c.template.update(b,a.c["if"].p(c),d,f,g)}};a.g.D["if"]=t;a.e.C["if"]=p;a.c.ifnot={p:function(b){return function(){return{ifnot:b(),templateEngine:a.q.K}}}, +init:function(b,c){return a.c.template.init(b,a.c.ifnot.p(c))},update:function(b,c,d,f,g){return a.c.template.update(b,a.c.ifnot.p(c),d,f,g)}};a.g.D.ifnot=t;a.e.C.ifnot=p;a.c.foreach={p:function(b){return function(){var c=a.a.d(b());return!c||"number"==typeof c.length?{foreach:c,templateEngine:a.q.K}:{foreach:c.data,includeDestroyed:c.includeDestroyed,afterAdd:c.afterAdd,beforeRemove:c.beforeRemove,afterRender:c.afterRender,templateEngine:a.q.K}}},init:function(b,c){return a.c.template.init(b,a.c.foreach.p(c))}, +update:function(b,c,d,f,g){return a.c.template.update(b,a.c.foreach.p(c),d,f,g)}};a.g.D.foreach=t;a.e.C.foreach=p;a.t=function(){};a.t.prototype.renderTemplateSource=function(){m(Error("Override renderTemplateSource"))};a.t.prototype.createJavaScriptEvaluatorBlock=function(){m(Error("Override createJavaScriptEvaluatorBlock"))};a.t.prototype.makeTemplateSource=function(b,c){if("string"==typeof b){var c=c||document,d=c.getElementById(b);d||m(Error("Cannot find template with ID "+b));return new a.l.i(d)}if(1== +b.nodeType||8==b.nodeType)return new a.l.M(b);m(Error("Unknown template type: "+b))};a.t.prototype.renderTemplate=function(a,c,d,f){return this.renderTemplateSource(this.makeTemplateSource(a,f),c,d)};a.t.prototype.isTemplateRewritten=function(a,c){return this.allowTemplateRewriting===t||!(c&&c!=document)&&this.V&&this.V[a]?p:this.makeTemplateSource(a,c).data("isRewritten")};a.t.prototype.rewriteTemplate=function(a,c,d){var f=this.makeTemplateSource(a,d),c=c(f.text());f.text(c);f.data("isRewritten", +p);!(d&&d!=document)&&"string"==typeof a&&(this.V=this.V||{},this.V[a]=p)};a.b("templateEngine",a.t);a.Z=function(){function b(b,c,e){for(var b=a.g.W(b),d=a.g.D,j=0;j/g;return{mb:function(b,c,e){c.isTemplateRewritten(b,e)||c.rewriteTemplate(b,function(b){return a.Z.zb(b,c)},e)},zb:function(a,g){return a.replace(c,function(a,c,d,f,i,l,q){return b(q,c,g)}).replace(d,function(a,c){return b(c,"<\!-- ko --\>",g)})},Za:function(b){return a.s.na(function(c, +e){c.nextSibling&&a.ya(c.nextSibling,b,e)})}}}();a.b("templateRewriting",a.Z);a.b("templateRewriting.applyMemoizedBindingsToNextSibling",a.Z.Za);(function(){a.l={};a.l.i=function(a){this.i=a};a.l.i.prototype.text=function(){var b=a.a.o(this.i),b="script"===b?"text":"textarea"===b?"value":"innerHTML";if(0==arguments.length)return this.i[b];var c=arguments[0];"innerHTML"===b?a.a.Y(this.i,c):this.i[b]=c};a.l.i.prototype.data=function(b){if(1===arguments.length)return a.a.f.get(this.i,"templateSourceData_"+ +b);a.a.f.set(this.i,"templateSourceData_"+b,arguments[1])};a.l.M=function(a){this.i=a};a.l.M.prototype=new a.l.i;a.l.M.prototype.text=function(){if(0==arguments.length){var b=a.a.f.get(this.i,"__ko_anon_template__")||{};b.ua===n&&b.da&&(b.ua=b.da.innerHTML);return b.ua}a.a.f.set(this.i,"__ko_anon_template__",{ua:arguments[0]})};a.l.i.prototype.nodes=function(){if(0==arguments.length)return(a.a.f.get(this.i,"__ko_anon_template__")||{}).da;a.a.f.set(this.i,"__ko_anon_template__",{da:arguments[0]})}; +a.b("templateSources",a.l);a.b("templateSources.domElement",a.l.i);a.b("templateSources.anonymousTemplate",a.l.M)})();(function(){function b(b,c,d){for(var f,c=a.e.nextSibling(c);b&&(f=b)!==c;)b=a.e.nextSibling(f),(1===f.nodeType||8===f.nodeType)&&d(f)}function c(c,d){if(c.length){var f=c[0],g=c[c.length-1];b(f,g,function(b){a.xa(d,b)});b(f,g,function(b){a.s.Wa(b,[d])})}}function d(a){return a.nodeType?a:0a.a.ja)&&b.nodes?b.nodes():s; +if(c)return a.a.L(c.cloneNode(p).childNodes);b=b.text();return a.a.pa(b)};a.q.K=new a.q;a.ra(a.q.K);a.b("nativeTemplateEngine",a.q);(function(){a.ma=function(){var a=this.vb=function(){if("undefined"==typeof jQuery||!jQuery.tmpl)return 0;try{if(0<=jQuery.tmpl.tag.tmpl.open.toString().indexOf("__"))return 2}catch(a){}return 1}();this.renderTemplateSource=function(b,f,g){g=g||{};2>a&&m(Error("Your version of jQuery.tmpl is too old. Please upgrade to jQuery.tmpl 1.0.0pre or later."));var e=b.data("precompiled"); +e||(e=b.text()||"",e=jQuery.template(s,"{{ko_with $item.koBindingContext}}"+e+"{{/ko_with}}"),b.data("precompiled",e));b=[f.$data];f=jQuery.extend({koBindingContext:f},g.templateOptions);f=jQuery.tmpl(e,b,f);f.appendTo(document.createElement("div"));jQuery.fragments={};return f};this.createJavaScriptEvaluatorBlock=function(a){return"{{ko_code ((function() { return "+a+" })()) }}"};this.addTemplate=function(a,b){document.write("