Files
ionic-framework/js/angular/directive/slideBox.js
Andrew 7ef9ad74cf fix(slidebox): refactor for performance and stability
Closes #2336. Closes #2317. Closes #2290. Closes #2228. Closes #2067.
Closes #1890. Closes #1865. Closes #1850. Closes #1755. Closes #1688.
Closes #1578. Closes #1501. Closes #1353. Closes #1342. Closes #782.
Closes #416. Closes #2288.

BREAKING CHANGE: The slideBox's API has undergone many changes.

- **`<ion-slide-box>`** attributes have changed (see
  [documentation](http://ionicframework.com/docs/api/directive/ionSlideBox)):

  * `active-slide` has changed to `selected`. Change your code from
  this:

    ```html
    <ion-slide-box active-slide="activeSlideIndex"></ion-slide-box>
    ```

    To this:

    ```html
    <ion-slide-box selected="activeSlideIndex"></ion-slide-box>
    ```

  * `does-continue` has changed to `loop`.  Change your code from this:

    ```html
    <ion-slide-box does-continue="shouldLoop"></ion-slide-box>
    ```

    To this:

    ```html
    <ion-slide-box loop="shouldLoop"></ion-slide-box>
    ```

  * `auto-play` and `slide-interval` have been merged into `auto-play`.
  Change your code from this:

    ```html
    <!-- autoPlay is on -->
    <ion-slide-box auto-play="true" slide-interval="1000">
    </ion-slide-box>
    <!-- autoPlay is off -->
    <ion-slide-box auto-play="false" slide-interval="1000">
    </ion-slide-box>
    ```

    To this:

    ```html
    <!-- autoPlay is on -->
    <ion-slide-box auto-play="1000"></ion-slide-box>
    <!-- autoPlay is off -->
    <ion-slide-box auto-play="false"></ion-slide-box>
    ```

  * `show-pager` and `pager-click` have been removed. Use
  a child `<ion-slide-pager>` element. See the [`ion-slide-pager`
  documentation](http://ionicframework.com/docs/api/directive/ionSlidePager).
  Change your code from this:

  ```html
  <!-- pager using default click action -->
  <ion-slide-box show-pager="true">
  </ion-slide-box>
  <!-- pager with custom click action -->
  <ion-slide-box show-pager="true" pager-click="doSomething(index)">
  </ion-slide-box>
  ```

  To this:

  ```html
  <ion-slide-box>
    <!-- pager using default click action -->
    <ion-slide-pager></ion-slide-pager>
  </ion-slide-box>
  <ion-slide-box>
    <!-- pager with custom click action -->
    <ion-slide-pager ng-click="doSomething(index)"></ion-slide-pager>
  </ion-slide-box>
  ```

- **`$ionicSlideBoxDelegate`** methods have changed (see
  [documentation](http://ionicframework.com/docs/api/service/$ionicSlideBoxDelegate)):

  - `update()` has been removed. slideBox updates on its own now.

  - `stop()` has been removed. See `autoPlay()` below.

  - `start()` hass been removed. See `autoPlay()` below.

  - `slide(newIndex[, speed])` has been renamed to `select(newIndex[,
    speed]);

  - `currentIndex()` has been renamed to `selected()`.

  - `slidesCount()` has been renamed to `count()`.

  - New method `$ionicSlideBoxDelegate.autoPlay()`. Change your code
    from this:

    ```js
    // stop auto sliding
    $ionicSlideBoxDelegate.stop();
    // later... start auto sliding
    $ionicSlideBoxDelegate.start();
    ```

    To this:

    ```js
    var autoPlaySpeed = 3000; //wait 3000 seconds between changing slide
    // stop auto sliding
    $ionicSlideBoxDelegate.autoPlay(false);
    // later... start auto sliding
    $ionicSlideBoxDelegate.autoPlay(autoPlaySpeed);
    ```

  - `previous()` now returns the index of the previous slide and does
    not select. Change your code from this:

    ```js
    // select previous slide
    $ionicSlideBoxDelegate.previous();
    ```

    To this:

    ```js
    // select previous slide
    $ionicSlideBoxDelegate.select( $ionicSlideBoxDelegate.previous() );
    ```
  - `next()` now returns the index of the next slide and does
    not select. Change your code from this:

    ```js
    // select next slide
    $ionicSlideBoxDelegate.next();
    ```

    To this:

    ```js
    // select next slide
    $ionicSlideBoxDelegate.select( $ionicSlideBoxDelegate.next() );
    ```
2014-10-08 11:09:15 -06:00

138 lines
4.4 KiB
JavaScript

/**
* @ngdoc directive
* @name ionSlideBox
* @module ionic
* @delegate ionic.service:$ionicSlideBoxDelegate
* @restrict E
* @description
* The Slide Box is a multi-page container where each page can be swiped or dragged between:
*
* ![SlideBox](http://ionicframework.com.s3.amazonaws.com/docs/controllers/slideBox.gif)
*
* Note: The slideBox will always take up all of the space within its parent scroll
* container. If you wish to have a smaller slidebox, create a custom-sized parent
* <ion-scroll> element.
*
* @usage
* ```html
* <ion-content>
* <ion-slide-box on-slide-changed="slideHasChanged($index)"
* loop="shouldLoop"
* auto-play="3000">
* <ion-slide>
* <div class="box blue"><h1>BLUE</h1></div>
* </ion-slide>
* <ion-slide>
* <div class="box yellow"><h1>YELLOW</h1></div>
* </ion-slide>
* <ion-slide>
* <div class="box pink"><h1>PINK</h1></div>
* </ion-slide>
* </ion-slide-box>
* </ion-content>
* ```
*
* @param {expression=} selected A model bound to the selected slide index.
* with {@link ionic.service:$ionicSlideBoxDelegate}.
* @param {boolean=} loop Whether the slide box should loop. Default false.
* @param {number=} auto-play If a positive number, then every time the given number of milliseconds have passed, slideBox will go to the next slide. Set to a non-positive number to disable. Default: -1.
* @param {expression=} on-slide-changed Expression called whenever the slide is changed. Is passed an '$index' variable.
* @param {string=} delegate-handle The handle used to identify this slideBox with
* {@link ionic.service:$ionicSlideBoxDelegate}.
*/
IonicModule
.directive('ionSlideBox', [
'$ionicSlideBoxDelegate',
'$window',
function($ionicSlideBoxDelegate, $window) {
return {
restrict: 'E',
controller: '$ionSlideBox',
require: ['ionSlideBox', '^$ionicScroll'],
transclude: true,
scope: {
selectedIndex: '=?selected',
onSlideChanged: '&'
},
template: '<div class="slider-slides" ng-transclude></div>',
compile: compile
};
function compile(element, attr) {
// DEPRECATED attr.doesContinue
isDefined(attr.doesContinue) && attr.$set('loop', attr.doesContinue);
return postLink;
}
function postLink(scope, element, attr, ctrls) {
var slideBoxCtrl = ctrls[0];
var scrollCtrl = ctrls[1];
element.addClass('slider');
var deregister = $ionicSlideBoxDelegate._registerInstance(slideBoxCtrl, attr.delegateHandle);
scope.$on('$destroy', deregister);
isDefined(attr.loop) && watchLoop();
isDefined(attr.selected) && watchSelected();
isDefined(attr.autoPlay) && watchAutoPlay();
var throttledReposition = ionic.animationFrameThrottle(repositionSlideBox);
throttledReposition();
var oldScrollingY = scrollCtrl.scrollView.options.scrollingY;
scrollCtrl.scrollView.options.scrollingY = false;
angular.element($window).on('resize', throttledReposition);
scope.$on('$destroy', function() {
angular.element($window).off('resize', throttledReposition);
scrollCtrl.scrollView.options.scrollingY = oldScrollingY;
});
// ***
// Methods
// ***
// There is no way to make the slidebox stretch to a large enough size
// when its children are all position: absolute elements.
// We just make it so the slidebox is *always* as large as its parent scroll
// container.
function repositionSlideBox() {
element.css({
width: scrollCtrl.$element.prop('offsetWidth') + 'px',
height: scrollCtrl.$element.prop('offsetHeight') + 'px'
});
}
function watchLoop() {
var unwatchParent = scope.$parent.$watch(attr.loop, slideBoxCtrl.loop);
scope.$on('$destroy', unwatchParent);
}
function watchSelected() {
scope.$watch('selectedIndex', function selectedAttrWatchAction(newIndex) {
if (slideBoxCtrl.isInRange(newIndex) &&
slideBoxCtrl.selected() !== newIndex) {
slideBoxCtrl.select(newIndex);
}
});
scope.$watch(slideBoxCtrl.selected, function shownWatchAction(newIndex) {
scope.selectedIndex = newIndex;
scope.onSlideChanged({
$index: newIndex
});
});
}
function watchAutoPlay() {
var unwatchParent = scope.$parent.$watch(attr.autoPlay, slideBoxCtrl.autoPlay);
scope.$on('$destroy', unwatchParent);
}
}
}]);