<!DOCTYPE html>
<html ng-app="MyApp">

  <head>
    <link data-require="bootstrap-css@3.1.1" data-semver="3.1.1" rel="stylesheet" href="//netdna.bootstrapcdn.com/bootstrap/3.1.1/css/bootstrap.min.css" />
    <link rel="stylesheet" href="//cdn.kendostatic.com/2014.1.416/styles/kendo.common-bootstrap.core.min.css" />
    <link rel="stylesheet" href="//cdn.kendostatic.com/2014.1.416/styles/kendo.bootstrap.min.css" />
    <link rel="stylesheet" href="//cdn.kendostatic.com/2014.1.416/styles/kendo.bootstrap.mobile.min.css" />
    
    <script data-require="jquery@1.9.1" data-semver="1.9.1" src="http://code.jquery.com/jquery-1.9.1.min.js"></script>
    <script data-require="angular.js@1.2.16" data-semver="1.2.16" src="http://code.angularjs.org/1.2.16/angular.js"></script>
    <script src="//cdn.kendostatic.com/2014.1.416/js/kendo.all.min.js"></script>
    <script src="angular-kendo.js"></script>

    <link rel="stylesheet" href="style.css" />
    <script src="script.js"></script>
    <script type="text/javascript">
      function doIt() {
        $("input[type=checkbox]").each(function() {
            var kSwitch = $(this).getKendoMobileSwitch();
            if (!kSwitch) return;
            console.log("test");

            // calculate labels' width
            var tmpOn  = $("<span>").hide().css({"text-transform": "uppercase"}).html(kSwitch.options.onLabel);
            var tmpOff = $("<span>").hide().css({"text-transform": "uppercase"}).html(kSwitch.options.offLabel);
            tmpOn.insertAfter(kSwitch.wrapper);
            tmpOff.insertAfter(kSwitch.wrapper);
            var wOn     = tmpOn.width() + 2;
            var wOff    = tmpOff.width() + 2;
            var max     = Math.max(wOn, wOff);
            var width   = max + kSwitch.handle[0].offsetWidth + 10;
            tmpOn.remove();
            tmpOff.remove();

            kSwitch.wrapper.find(".km-switch-label-on").css({width: max + "px", left: "-" + (max + 4) + "px"});
            kSwitch.wrapper.find(".km-switch-label-off").css({width: max + "px"});
            kSwitch.wrapper.css({width: width + "px"});
            kSwitch.wrapper.find(".km-switch-background").css({"background-position": width + "px 0"});

            kSwitch.refresh();
        });
      }
    </script>
  </head>

  <body>
    <div ng-controller="MyCtrl">
    
    <div class="panel panel-primary">
      <div class="panel-heading">
        <h3 class="panel-title">Switch Labels</h3>
      </div>
      <div class="panel-body">
          Default Label:
          <input kendo-mobile-switch type="checkbox" />
          <br />
  
          Declaritive Custom Label:
          <input kendo-mobile-switch type="checkbox" k-on-label="'GOOD'" k-off-label="'BADDEREST'" />
          <br />
  
          k-options Custom Label:
          <input kendo-mobile-switch type="checkbox" k-options="options" />
      </div>
    </div>
    
    <div class="panel panel-info">
      <div class="panel-heading">
        <h3 class="panel-title">Switch Bindings</h3>
      </div>
      <div class="panel-body">
          ng-model binding:
          <input kendo-mobile-switch type="checkbox" ng-model="myValue1" />
          value: {{myValue1}}
          <br />
          
          with on-change:
          <input kendo-mobile-switch type="checkbox" ng-model="myValue2" k-on-change="change(kendoEvent)"/>
          value: {{myValue2}}
      </div>
    </div>
    <button type="button" onclick="doIt();">Do it!</button>


    </div>
  </body>

</html>
var app = angular.module("MyApp", [ "kendo.directives" ]);

app.controller('MyCtrl', function($scope) {
  
  $scope.myValue1 = true;
  $scope.myValue2 = true;
  
  $scope.options={
    onLabel:"A",
    offLabel:"B"
  }
  
  $scope.change = function(e) {
    $scope.myValue2 = e.checked;
  };
  
  $scope.$watch('myValue1', function() {
    console.log('Value 1 changed: ' + $scope.myValue1);
  });
  
  $scope.$watch('myValue2', function() {
    console.log('Value 2 changed: ' + $scope.myValue2);
  });
  
});
/* Styles go here */


(function(f, define){
  define([ "jquery", "angular", "kendo" ], f);
})(function($, angular, kendo) {

  "use strict";

  var _UID_ = kendo.attr("uid");

  var module = angular.module('kendo.directives', []);
  var parse, timeout, compile = function compile(){ return compile }, log;

  function immediately(f) {
    var save_timeout = timeout;
    timeout = function(f) { return f() };
    try {
      return f();
    } finally {
      timeout = save_timeout;
    }
  }

  var OPTIONS_NOW;

  var factories = {
    dataSource: (function() {
      var types = {
        TreeView  : 'HierarchicalDataSource',
        Scheduler : 'SchedulerDataSource',
        PanelBar  : '$PLAIN',
        Menu      : "$PLAIN",
      };
      var toDataSource = function(dataSource, type) {
        if (type == '$PLAIN')
          return dataSource;
        return kendo.data[type].create(dataSource);
      };
      return function(scope, element, attrs, role) {
        var type = types[role] || 'DataSource';
        var ds = toDataSource(scope.$eval(attrs.kDataSource), type);

        // not recursive -- this triggers when the whole data source changed
        scope.$watch(attrs.kDataSource, function(mew, old){
          if (mew !== old) {
            var ds = toDataSource(mew, type);
            var widget = kendoWidgetInstance(element);
            if (widget && typeof widget.setDataSource == "function") {
              widget.setDataSource(ds);
            }
          }
        });
        return ds;
      };
    }()),

    widget: (function() {
      var ignoredAttributes = {
        kDataSource : true,
        kOptions    : true,
        kRebind     : true,
        kNgModel    : true,
      };
      return function(scope, element, attrs, widget, origAttr) {
        var role = widget.replace(/^kendo/, '');
        var options = angular.extend({}, scope.$eval(attrs.kOptions));
        $.each(attrs, function(name, value) {
          if (!ignoredAttributes[name]) {
            var match = name.match(/^k(On)?([A-Z].*)/);
            if (match) {
              var optionName = match[2].charAt(0).toLowerCase() + match[2].slice(1);
              if (match[1]
                  && name != "kOnLabel" // XXX: k-on-label can be used on MobileSwitch :-\
                 ) {
                options[optionName] = value;
              } else {
                if (name == "kOnLabel")
                  optionName = "onLabel"; // XXX: that's awful.
                options[optionName] = angular.copy(scope.$eval(value));
                if (options[optionName] === undefined && value.match(/^\w*$/)) {
                  log.warn(widget + '\'s ' + name + ' attribute resolved to undefined. Maybe you meant to use a string literal like: \'' + value + '\'?');
                }
              }
            }
          }
        });

        // parse the datasource attribute
        if (attrs.kDataSource) {
          options.dataSource = factories.dataSource(scope, element, attrs, role);
        }

        options.$angular = true;
        var ctor = $(element)[widget];
        if (!ctor) {
          console.error("Could not find: " + widget);
          return null;
        }
        var object = ctor.call(element, OPTIONS_NOW = options).data(widget);
        exposeWidget(object, scope, attrs, widget, origAttr);
        scope.$emit("kendoWidgetCreated", object);
        return object;
      };
    }())
  };

  function exposeWidget(widget, scope, attrs, kendoWidget, origAttr) {
    if (attrs[origAttr]) {
      // expose the widget object
      var set = parse(attrs[origAttr]).assign;
      if (set) {
        // set the value of the expression to the kendo widget object to expose its api
        set(scope, widget);
      } else {
        throw new Error(origAttr + ' attribute used but expression in it is not assignable: ' + attrs[kendoWidget]);
      }
    }
  }

  function hasKendoTag(element) {
    return /^kendo/i.test(element.prop("tagName"));
  }

  module.factory('directiveFactory', ['$timeout', '$parse', '$compile', '$log', function($timeout, $parse, $compile, $log) {

    timeout = $timeout;
    parse = $parse;
    compile = $compile;
    log = $log;

    var KENDO_COUNT = 0;

    var create = function(role, origAttr) {

      return {
        // Parse the directive for attributes and classes
        restrict: "ACE",
        require: [ "?ngModel", "^?form" ],
        scope: false,

        transclude: true,
        controller: [ '$scope', '$attrs', '$element', '$transclude', function($scope, $attrs, $element, $transclude) {

          if (hasKendoTag($element)) (function(){
            var element = $element[0];
            $attrs.$kendoOrigElement = element.cloneNode(true);
            var attributes = Array.prototype.slice.call(element.attributes); // guess why we need that. :-\
            for (var i = 0; i < attributes.length; ++i) {
              var orig = attributes[i].nodeName;
              if (!/^(k|ng)-/.test(orig) && !/^(style|class|id)$/.test(orig)) {
                var name = ("k-" + orig).replace(/-(.)/g, function(s, p){
                  return p.toUpperCase();
                });
                if (!(name in $attrs)) {
                  $attrs[name] = attributes[i].nodeValue;
                }

                // we must remove the original attribute!  otherwise some widgets (DropDownList at least) will prefer to
                // take options from there instead of the options object we actually pass to constructor, ending up with
                // dataTextField = "'name'" (unevaluated), leading to a SyntaxError: Unexpected string error.
                element.removeAttribute(orig);
              }
            }
          })();

          // Make the element's contents available to the kendo widget to allow creating some widgets from existing elements.
          $transclude($scope, function(clone){
            $element.append(clone);
          });
        }],

        link: function(scope, element, attrs, controllers) {

          var ngModel = controllers[0];
          var ngForm = controllers[1];

          // we must remove data-kendo-widget-name attribute because
          // it breaks kendo.widgetInstance; can generate all kinds
          // of funny issues like
          // https://github.com/kendo-labs/angular-kendo/issues/167

          // $(element).removeData(role);
          // console.log($(element).data(role)); // --> not undefined.  now I'm pissed.
          $(element)[0].removeAttribute("data-" + role.replace(/([A-Z])/g, "-$1"));

          ++KENDO_COUNT;

          timeout(function() {
            // if k-rebind attribute is provided, rebind the kendo widget when
            // the watched value changes
            if (attrs.kRebind) {
              var originalElement = attrs.$kendoOrigElement || $(element)[0].cloneNode(true);
              // watch for changes on the expression passed in the k-rebind attribute
              var unregister = scope.$watch(attrs.kRebind, function(newValue, oldValue) {
                if (newValue !== oldValue) {
                  unregister(); // this watcher will be re-added if we compile again!

                  /****************************************************************
                   // XXX: this is a gross hack that might not even work with all
                   // widgets.  we need to destroy the current widget and get its
                   // wrapper element out of the DOM, then make the original element
                   // visible so we can initialize a new widget on it.
                   //
                   // kRebind is probably impossible to get right at the moment.
                   ****************************************************************/

                  var _wrapper = $(widget.wrapper)[0];
                  var _element = $(widget.element)[0];
                  widget.destroy();
                  widget = null;
                  if (_wrapper && _element) {
                    _wrapper.parentNode.replaceChild(_element, _wrapper);
                    var clone = originalElement.cloneNode(true);
                    $(element).replaceWith(clone);
                    element = $(clone);
                  }
                  compile(element)(scope);
                }
              }, true); // watch for object equality. Use native or simple values.
            }

            var widget = factories.widget(scope, element, attrs, role, origAttr);
            setupBindings();

            var prev_destroy = null;
            function setupBindings() {

              var isFormField = /^(input|select|textarea)$/i.test(element[0].tagName);
              function formValue(el) {
                if (/checkbox|radio/i.test(element.attr("type")))
                  return element.prop("checked");
                return element.val();
              }
              function value() {
                return isFormField ? formValue(element) : widget.value();
              }

              // Cleanup after ourselves
              if (prev_destroy) {
                prev_destroy();
              }
              prev_destroy = scope.$on("$destroy", function() {
                if (widget) {
                  if (widget.element) {
                    widget.destroy();
                  }
                  widget = null;
                }
              });

              // 2 way binding: ngModel <-> widget.value()
              OUT: if (ngModel) {
                if (!widget.value) {
                  break OUT;
                }

                // Angular will invoke $render when the view needs to be updated with the view value.
                ngModel.$render = function() {
                  // Update the widget with the view value.

                  // delaying with setTimout for cases where the datasource is set thereafter.
                  // https://github.com/kendo-labs/angular-kendo/issues/304
                  var val = ngModel.$viewValue;
                  if (val === undefined) val = ngModel.$modelValue;
                  setTimeout(function(){
                    widget.value(val);
                  }, 0);
                };

                // Some widgets trigger "change" on the input field
                // and this would result in two events sent (#135)
                var haveChangeOnElement;
                if (isFormField) {
                  element.on("change", function(){
                    haveChangeOnElement = true;
                  });
                }

                var onChange = function(pristine){
                  return function(){
                    haveChangeOnElement = false;
                    timeout(function(){
                      if (haveChangeOnElement) return;
                      if (pristine && ngForm) {
                        var formPristine = ngForm.$pristine;
                      }
                      ngModel.$setViewValue(value());
                      if (pristine) {
                        ngModel.$setPristine();
                        if (formPristine) {
                          ngForm.$setPristine();
                        }
                      }
                    });
                  };
                };

                bindBefore(widget, "change", onChange(false));
                bindBefore(widget, "dataBound", onChange(true));

                var currentVal = value();

                // if the model value is undefined, then we set the widget value to match ( == null/undefined )
                if (currentVal != ngModel.$viewValue) {
                  if (!ngModel.$isEmpty(ngModel.$viewValue)) {
                    widget.value(ngModel.$viewValue);
                  } else if (currentVal != null && currentVal != "" && currentVal != ngModel.$viewValue) {
                    ngModel.$setViewValue(currentVal);
                  }
                }

                ngModel.$setPristine();
              }

              // kNgModel is used for the "logical" value
              OUT: if (attrs.kNgModel) {
                if (typeof widget.value != "function") {
                  log.warn("k-ng-model specified on a widget that does not have the value() method: " + (widget.options.name));
                  break OUT;
                }
                var getter = parse(attrs.kNgModel);
                var setter = getter.assign;
                var updating = false;
                widget.value(getter(scope));

                // keep in sync
                scope.$watch(attrs.kNgModel, function(newValue, oldValue){
                  if (updating) return;
                  if (newValue === oldValue) return;
                  widget.value(newValue);
                });
                bindBefore(widget, "change", function(){
                  updating = true;
                  setter(scope, widget.value());
                  digest(scope);
                  updating = false;
                });
              }
            }

            // mutation observers — propagate the original
            // element's class to the widget wrapper.
            (function(){

              if (!(window.MutationObserver && widget.wrapper)) {
                return;
              }

              var prevClassList = [].slice.call($(element)[0].classList);

              var mo = new MutationObserver(function(changes, mo){
                suspend();    // make sure we don't trigger a loop
                if (!widget) return;

                changes.forEach(function(chg){
                  var w = $(widget.wrapper)[0];
                  switch (chg.attributeName) {

                   case "class":
                    // sync classes to the wrapper element
                    var currClassList = [].slice.call(chg.target.classList);
                    currClassList.forEach(function(cls){
                      if (prevClassList.indexOf(cls) < 0) {
                        w.classList.add(cls);
                      }
                    });
                    prevClassList.forEach(function(cls){
                      if (currClassList.indexOf(cls) < 0) {
                        w.classList.remove(cls);
                      }
                    });
                    prevClassList = currClassList;
                    break;

                   case "disabled":
                    if (typeof widget.enable == "function") {
                      widget.enable(!$(chg.target).attr("disabled"));
                    }
                    break;

                   case "readonly":
                    if (typeof widget.readonly == "function") {
                      widget.readonly(!!$(chg.target).attr("readonly"));
                    }
                    break;
                  }
                });

                resume();
              });

              function suspend() {
                mo.disconnect();
              }
              function resume() {
                mo.observe($(element)[0], { attributes: true });
              }
              resume();
              bindBefore(widget, "destroy", suspend);
            })();

            --KENDO_COUNT;
            if (KENDO_COUNT == 0) {
              scope.$emit("kendoRendered");
            }

          });
        }
      };
    };

    return {
      create: create
    };
  }]);

  // create directives for every widget.
  (function(){
    function createDirectives(prefix) {
      function make(name, widget) {
        module.directive(name, [
          "directiveFactory",
          function(directiveFactory) {
            return directiveFactory.create(widget, name);
          }
        ]);
      }
      return function(namespace) {
        angular.forEach(namespace, function(value, key) {
          if (key.match(/^[A-Z]/) && key !== 'Widget') {
            var widget = "kendo" + prefix + key;
            var shortcut = (prefix + key).toLowerCase();
            shortcut = "kendo" + shortcut.charAt(0).toUpperCase() + shortcut.substr(1);
            make(widget, widget);
            if (shortcut != widget) {
              make(shortcut, widget);
            }
          }
        });
      }
    }
    angular.forEach([ kendo.ui, kendo.dataviz && kendo.dataviz.ui ], createDirectives(""));
    angular.forEach([ kendo.mobile && kendo.mobile.ui ], createDirectives("Mobile"));
  })();

  /* -----[ utils ]----- */

  function kendoWidgetInstance(el) {
    el = $(el);
    return kendo.widgetInstance(el, kendo.ui) ||
      kendo.widgetInstance(el, kendo.mobile.ui) ||
      kendo.widgetInstance(el, kendo.dataviz.ui);
  }

  // XXX: using internal API (Widget::_events).  Seems to be no way in Kendo to
  // insert a handler to be executed before any existing ones, hence this hack.
  // Use for a single event/handler combination.
  function bindBefore(widget, name, handler, one) {
    widget.bind.call(widget, name, handler, one);
    var a = widget._events[name];
    a.unshift(a.pop());
  }

  function digest(scope) {
    if (!/^\$(digest|apply)$/.test(scope.$root.$$phase)) {
      scope.$digest();
    }
  }

  function destroyScope(scope, el) {
    scope.$destroy();
    if (el) {
      // prevent leaks. https://github.com/kendo-labs/angular-kendo/issues/237
      $(el)
        .removeData("$scope")
        .removeData("$isolateScope")
        .removeData("$isolateScopeNoTemplate")
        .removeClass("ng-scope");
    }
  }

  // defadvice will patch a class' method with another function.  That
  // function will be called in a context containing `next` (to call
  // the next method) and `object` (a reference to the original
  // object).
  function defadvice(klass, methodName, func) {
    if ($.isArray(klass)) {
      return angular.forEach(klass, function(klass){
        defadvice(klass, methodName, func);
      });
    }
    if (typeof klass == "string") {
      var a = klass.split(".");
      var x = kendo;
      while (x && a.length > 0) x = x[a.shift()];
      if (!x) {
        //console.log("Can't advice " + klass + "::" + methodName);
        return;
      }
      klass = x;
    }
    var origMethod = klass.prototype[methodName];
    klass.prototype[methodName] = function() {
      var self = this, args = arguments;
      return func.apply({
        self: self,
        next: function() {
          return origMethod.apply(self, arguments.length > 0 ? arguments : args);
        }
      }, args);
    };
  }

  var BEFORE = "$angular_beforeCreate";
  var AFTER = "$angular_afterCreate";

  /* -----[ Customize widgets for Angular ]----- */

  // XXX: notice we can't override `init` in general for any widget,
  // because kendo.ui.Widget === kendo.ui.Widget.prototype.init.
  // Hence we resort to the beforeCreate/afterCreate hack.
  defadvice("ui.Widget", "init", function(element, options){
    if (!options && OPTIONS_NOW) options = OPTIONS_NOW;
    OPTIONS_NOW = null;
    var self = this.self;
    if (options && options.$angular) {
      // call before/after hooks only for widgets instantiated by angular-kendo
      self.$angular_beforeCreate(element, options);
      this.next();
      self.$angular_afterCreate();
    } else {
      this.next();
    }
  });

  // All event handlers that are strings are compiled the Angular way.
  defadvice("ui.Widget", BEFORE, function(element, options) {
    var self = this.self;
    if (options && !$.isArray(options)) {
      var scope = angular.element(element).scope();
      for (var i = self.events.length; --i >= 0;) {
        var event = self.events[i];
        var handler = options[event];
        if (handler && typeof handler == "string")
          options[event] = self.$angular_makeEventHandler(event, scope, handler);
      }
    }
  });

  defadvice("ui.Widget", AFTER, function(){});

  // most handers will only contain a kendoEvent in the scope.
  defadvice("ui.Widget", "$angular_makeEventHandler", function(event, scope, handler){
    handler = parse(handler);
    return function(e) {
      if (/^\$(apply|digest)$/.test(scope.$root.$$phase)) {
        handler(scope, { kendoEvent: e });
      } else {
        scope.$apply(function() { handler(scope, { kendoEvent: e }) });
      }
    };
  });

  // for the Grid and ListView we add `data` and `selected` too.
  defadvice([ "ui.Grid", "ui.ListView", "ui.TreeView" ], "$angular_makeEventHandler", function(event, scope, handler){
    if (event != "change") return this.next();
    handler = parse(handler);
    return function(ev) {
      var widget = ev.sender;
      var options = widget.options;
      var dataSource = widget.dataSource;
      var cell, multiple, locals = { kendoEvent: ev }, elems, items, columns, colIdx;

      if (angular.isString(options.selectable)) {
        cell = options.selectable.indexOf('cell') !== -1;
        multiple = options.selectable.indexOf('multiple') !== -1;
      }

      elems = locals.selected = this.select();
      items = locals.data = [];
      columns = locals.columns = [];
      for (var i = 0; i < elems.length; i++) {
        var item = cell ? elems[i].parentNode : elems[i];
        var itemUid = $(item).attr(_UID_);
        var dataItem = dataSource.getByUid(itemUid);
        if (cell) {
          if (angular.element.inArray(dataItem, items) < 0) {
            items.push(dataItem);
          }
          colIdx = angular.element(elems[i]).index();
          if (angular.element.inArray(colIdx, columns) < 0 ) {
            columns.push(colIdx);
          }
        } else {
          items.push(dataItem);
        }
      }

      if (!multiple) {
        locals.dataItem = locals.data = items[0];
        locals.selected = elems[0];
      }

      scope.$apply(function() { handler(scope, locals) });
    };
  });

  // for PanelBar, TabStrip and Splitter, hook on `contentLoad` to
  // compile Angular templates.
  defadvice([ "ui.PanelBar", "ui.TabStrip", "ui.Splitter" ], AFTER, function() {
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope) bindBefore(self, "contentLoad", function(ev){
      //                   tabstrip/panelbar    splitter
      var contentElement = ev.contentElement || ev.pane;
      compile(contentElement)(scope);
      digest(scope);
    });
  });

  // on Draggable::_start compile the content as Angular template, if
  // an $angular_scope method is provided.
  defadvice("ui.Draggable", "_start", function(){
    this.next();
    var self = this.self;
    if (self.hint) {
      var scope = angular.element(self.currentTarget).scope();
      if (scope) {
        compile(self.hint)(scope);
        digest(scope);
      }
    }
  });

  // If no `template` is supplied for Grid columns, provide an Angular
  // template.  The reason is that in this way AngularJS will take
  // care to update the view as the data in scope changes.
  defadvice("ui.Grid", BEFORE, function(element, options){
    this.next();
    if (options.columns) {
      var settings = $.extend({}, kendo.Template, options.templateSettings);
      angular.forEach(options.columns, function(col){
        if (col.field && !col.template && !col.format && !col.values) {
          col.template = "<span ng-bind='"
            + kendo.expr(col.field, "dataItem") + "'>#: "
            + kendo.expr(col.field, settings.paramName) + "#</span>";
        }
      });
    }
  });

  defadvice([ "ui.ListView", "mobile.ui.ListView", "ui.TreeView" ], "$angular_itemsToCompile", function(){
    return this.self.items();
  });

  defadvice("ui.TreeView", "$angular_itemsToCompile", function(){
    return this.self.element.find(".k-item div:first-child");
  });

  defadvice([ "ui.Grid" ], "$angular_itemsToCompile", function(){
    var self = this.self;
    if (!self.lockedContent) return self.items();
    return self.items().add("tr[" + _UID_ + "]", self.lockedContent);
  });

  // for Grid, ListView and TreeView, provide a dataBound handler that
  // recompiles Angular templates.  We need to do this before the
  // widget is initialized so that we catch the first dataBound event.
  defadvice([ "ui.Grid", "ui.ListView", "mobile.ui.ListView", "ui.TreeView" ], BEFORE, function(element, options){
    this.next();
    var scope = angular.element(element).scope();
    if (!scope) return;
    var self = this.self;
    var role = self.options.name;
    var prev_dataBound = options.dataBound;
    options.dataBound = function(ev) {
      var widget = ev.sender;
      var dataSource = widget.dataSource;
      var dirty = false;
      widget.$angular_itemsToCompile().each(function(){
        var elementToCompile = $(this);
        if (!elementToCompile.hasClass("ng-scope")) {
          var itemUid = $(this).closest("[" + _UID_ + "]").attr(_UID_);
          var item = dataSource.getByUid(itemUid);
          var itemScope = scope.$new();
          itemScope.dataItem = item;
          compile(elementToCompile)(itemScope);
          dirty = true;
        }
      });
      try {
        if (prev_dataBound) return prev_dataBound.apply(this, arguments);
      } finally {
        if (dirty) digest(scope);
      }
    };
  });

  defadvice([ "ui.Grid", "ui.ListView", "mobile.ui.ListView" ], AFTER, function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (!scope) return;

    // itemChange triggers when a single item is changed through our
    // DataSource mechanism.
    self.bind("itemChange", function(ev) {
      var dataSource = ev.sender.dataSource;
      var itemElement = ev.item[0];
      var item = ev.item;
      if ($.isArray(item)) item = item[0];
      item = $(item);
      var itemScope = angular.element(item).scope();
      if (!itemScope || itemScope === scope) {
        itemScope = scope.$new();
      }
      itemScope.dataItem = dataSource.getByUid(item.attr(_UID_));
      compile(itemElement)(itemScope);
      digest(itemScope);
    });

    // dataBinding triggers when new data is loaded.  We use this to
    // destroy() each item's scope.
    self.bind("dataBinding", function(ev) {
      ev.sender.$angular_itemsToCompile().each(function(){
        var el = $(this);
        if (el.attr(_UID_)) {
          var rowScope = angular.element(this).scope();
          // avoid destroying the widget's own scope
          // no idea why we get it, but we do.... :(
          if (rowScope && rowScope !== scope) {
            destroyScope(rowScope, el);
          }
        }
      });
    });
  });

  // DropDownList
  defadvice("ui.DropDownList", BEFORE, function(element, options){
    this.next();
    var scope = angular.element(element).scope();
    if (!scope) return;
    var self = this.self;

    // compile {{angular}} on dataBound
    var prev_dataBound = options.dataBound;
    options.dataBound = function(ev) {
      var widget = ev.sender;
      widget.ul.find("li").each(function(idx){
        var itemScope = scope.$new();
        itemScope.dataItem = widget.dataItem(idx);
        compile(this)(itemScope);
      });
      try {
        if (prev_dataBound)
          return prev_dataBound.apply(this, arguments);
      } finally {
        digest(scope);
      }
    };

    // destroy scopes on dataBinding
    var prev_dataBinding = options.dataBinding;
    options.dataBinding = function(ev) {
      var widget = ev.sender;
      widget.ul.find("li").each(function(){
        var itemScope = angular.element(this).scope();
        if (itemScope && itemScope !== scope) {
          destroyScope(itemScope);
        }
      });
      if (prev_dataBinding)
        return prev_dataBinding.apply(this, arguments);
    };
  });

  defadvice("ui.DropDownList", "_textAccessor", function(text){
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope && text !== undefined) {
      var itemScope = angular.element(self.span).scope();
      if (itemScope && itemScope !== scope) {
        destroyScope(itemScope);
      }
    }
    var ret = this.next();
    if (scope && text !== undefined) {
      var itemScope = scope.$new();
      itemScope.dataItem = text;
      compile(self.span)(itemScope);
      digest(itemScope);
    }
    return ret;
  });

  // templates for autocomplete and combo box
  defadvice([ "ui.AutoComplete", "ui.ComboBox" ], BEFORE, function(element, options){
    this.next();
    var scope = angular.element(element).scope();
    if (!scope) return;
    var self = this.self;
    var prev_dataBound = options.dataBound;
    options.dataBound = function(ev) {
      var widget = ev.sender;
      var dataSource = widget.dataSource;
      var dirty = false;
      $(widget.items()).each(function(){
        var el = $(this);
        if (!el.hasClass("ng-scope")) {
          var item = widget.dataItem(el.index());
          var itemScope = scope.$new();
          itemScope.dataItem = item;
          compile(el)(itemScope);
          dirty = true;
        }
      });
      try {
        if (prev_dataBound) return prev_dataBound.apply(this, arguments);
      } finally {
        if (dirty) digest(scope);
      }
    };
  });

  defadvice([ "ui.AutoComplete", "ui.ComboBox" ], AFTER, function(){
    this.next();
    this.self.bind("dataBinding", function(ev){
      $(ev.sender.items()).each(function(){
        var scope = angular.element(this).scope();
        if (scope) {
          destroyScope(scope, this);
        }
      });
    });
  });

  defadvice("ui.Grid", "_toolbar", function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope) {
      compile(self.wrapper.find(".k-grid-toolbar").first())(scope);
      digest(scope);
    }
  });

  defadvice("ui.Grid", "_thead", function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope) {
      compile(self.thead)(scope);
      digest(scope);
    }
  });

  defadvice("ui.Grid", "_footer", function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope) {
      compile(self.footer)(scope);
      digest(scope);
    }
  });

  defadvice("ui.editor.Toolbar", "render", function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope) {
      compile(self.element)(scope);
      digest(scope);
    }
  });

  defadvice("ui.Grid", AFTER, function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope) {
      if (self.options.detailTemplate) bindBefore(self, "detailInit", function(ev){
        var detailScope = scope.$new();
        detailScope.dataItem = ev.data;
        compile(ev.detailCell)(detailScope);
        digest(detailScope);
      });
    }
  });

  defadvice("ui.Grid", "cancelRow", function(){
    var self = this.self;
    var scope = angular.element(self.element).scope();
    var cont = self._editContainer;
    if (cont) {
      var model = self._modelForContainer(cont);
      var uid = model.uid;
      var prevScope = angular.element(cont).scope();
      if (prevScope && prevScope !== scope) {
        destroyScope(prevScope, cont);
      }
    }
    this.next();
    if (uid) {
      var row = self.items().filter("[" + _UID_ + "=" + uid + "]");
      var rowScope = scope.$new();
      rowScope.dataItem = model;
      compile(row)(rowScope);
      digest(scope);
    }
  });

  defadvice("ui.Editable", "refresh", function(){
    this.next();
    var self = this.self;
    var model = self.options.model;
    var scope = angular.element(self.element).scope();
    if (!scope || !model) return;

    if (self.$angular_scope) {
      destroyScope(self.$angular_scope, self.element);
    }

    scope = self.$angular_scope = scope.$new();
    scope.dataItem = model;

    // XXX: we need to disable the timeout here, or else the widget is
    // created but immediately destroyed (focus lost).
    immediately(function(){
      compile(self.element)(scope);
      digest(scope);
    });

    // and we still need to focus it.
    self.element.find(":kendoFocusable").eq(0).focus();
  });

  defadvice("ui.Editable", "destroy", function(){
    var self = this.self;
    if (self.$angular_scope) {
      destroyScope(self.$angular_scope, self.element);
      self.$angular_scope = null;
    }
    this.next();
  });

  defadvice("ui.Window", AFTER, function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (!scope) return;
    bindBefore(self, "refresh", function(){
      var content = self.wrapper.children(".k-window-content");
      var scrollContainer = content.children(".km-scroll-container");
      content = scrollContainer[0] ? scrollContainer : content;
      compile(content.children())(scope);
      digest(scope);
    });
  });

  defadvice("mobile.ui.ListView", "destroy", function(){
    var self = this.self;
    if (self._itemBinder && self._itemBinder.dataSource) {
      this.self._itemBinder._unbindDataSource();
    }
    this.next();
  });

  defadvice("ui.Tooltip", "_appendContent", function(){
    this.next();
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (scope) {
      compile(self.content)(scope);
      digest(scope);
    }
  });

  defadvice("ui.Menu", "_initData", function(options){
    var self = this.self;
    var scope = angular.element(self.element).scope();
    if (!scope) {
      return this.next();
    }
    if (options.dataSource) {
      if (self.$angular_scope)
        destroyScope(self.$angular_scope);
    }
    this.next();
    if (options.dataSource) {
      self.$angular_scope = scope.$new();
      compile(self.element.children())(self.$angular_scope);
    }
  });

  // scheduler
  {
    defadvice("ui.Scheduler", AFTER, function(){
      this.next();
      var self = this.self;
      var scope = angular.element(self.element).scope();
      if (scope) {
        self.$eventsScope = scope.$new();
        bindBefore(self, "edit", function(ev){
          var editScope = self.$editScope = scope.$new();
          editScope.dataItem = ev.model;
          compile(ev.container)(editScope);
        });
        var destroy = function(ev){
          var editScope = self.$editScope;
          if (editScope !== scope) {
            destroyScope(editScope, ev.container);
            self.$editScope = null;
          }
        };
        bindBefore(self, "cancel", destroy);
        bindBefore(self, "save", destroy);
        bindBefore(self, "remove", destroy);
        bindBefore(self, "navigate", function(){
          self.$eventsScope.$destroy();
          self.$eventsScope = scope.$new();
        });
      }
    });

    defadvice("ui.Scheduler", "destroy", function(){
      if (this.$eventsScope) {
        this.$eventsScope.$destroy();
      }
      this.next();
    });

    defadvice([ "ui.MultiDayView", "ui.MonthView" ], "_createEventElement", function(event){
      var element = this.next();
      var self = this.self;
      var scope = angular.element(self.element).scope();
      var itemScope = scope.$new();
      itemScope.dataItem = event;
      compile(element)(itemScope);
      digest(itemScope);
      return element;
    });
  }

  {
    // mobile/ButtonGroup does not have a "value" method, but looks
    // like it would be useful.  We provide it here.

    defadvice("mobile.ui.ButtonGroup", "value", function(mew){
      var self = this.self;
      if (mew != null) {
        self.select(self.element.children("li.km-button").eq(mew));
        self.trigger("change");
        self.trigger("select", { index: self.selectedIndex });
      }
      return self.selectedIndex;
    });

    defadvice("mobile.ui.ButtonGroup", "_select", function(){
      this.next();
      this.self.trigger("change");
    });
  }

  {
    // same for mobile/ScrollView, except that we don't need the
    // "change" event.

    defadvice("mobile.ui.ScrollView", "value", function(mew){
      var self = this.self;
      if (mew != null) {
        self.scrollTo(mew);
      }
      return self.page;
    });

    defadvice("mobile.ui.VirtualScrollViewContent", "setPageContent", function(page){
      var self = this.self;
      var scope = angular.element(self.element).scope();
      if (scope) {
        var itemScope = angular.element(page.element).scope();
        if (itemScope && itemScope !== scope) {
          destroyScope(itemScope);
        }
      }
      this.next();
    });

    defadvice("mobile.ui.ScrollView", AFTER, function(){
      this.next();
      var self = this.self;
      var scope = angular.element(self.element).scope();
      if (!scope) return;
      bindBefore(self, "itemChange", function(ev){
        var itemScope = scope.$new();
        itemScope.dataItem = ev.data;
        compile(ev.item)(itemScope);
        digest(itemScope);
      });
    });
  }

  defadvice("mobile.ui.Switch", "value", function(){
    var self = this.self;
    var ret = self.check.apply(self, arguments);
    return ret;
  });

}, typeof define == 'function' && define.amd ? define : function(_, f){ f(jQuery, angular, kendo); });

// Local Variables:
// js-indent-level: 2
// js2-basic-offset: 2
// End: