Skip to content

Latest commit

 

History

History
809 lines (584 loc) · 30.7 KB

README.md

File metadata and controls

809 lines (584 loc) · 30.7 KB

watusi/jquery-mobile-iscrollview, Version 1.1+

JQuery Mobile widget plug-in for easy use of the iScroll scroller in JQuery Mobile projects.

This is a full JQuery Widget Factory widget implementation. It follows the widget-factory-mobile Widget Factory Pattern.


Using Patched iScroll

This widget works best with a special version of iScroll that has a small patch. This patch allows an external script (e.g. this one) to change how iScroll gets the dimensions of elements.

Using standard DOM methods, it's not possible to easily get the dimensions of hidden elements. jQuery mobile hides the content of pages when they are not the current page. Thus, the standard iScroll is not able to properly initialize prior to the jQuery Mobile pageshow event.

The special version of iScroll allows this widget to override the dimension-getting functions in iScroll using calls to the jQuery.actual plugin, which is able to get the dimensions of hidden elements.

This widget will work with the standard iScroll distribution, but it will not be optimal. You may see unwanted visual artifacts when the scroller is refreshed on the pageshow event.

The demo directory contains a copy of the special version of iScroll in demo/iscroll-watusi.js. The demo uses this version of iScroll. The standard iScroll is also available in directory in demo/iscroll-cubiq.js. This file may also in the future contain other changes made to iScroll by WatusiWare Corporation.You can obtain a current copy of this version of iScroll from: https://github.com/watusi/iscroll Please use the "watusi" branch (the default branch for this repository).

You can use this special version of iScroll with other projects. If an external script does not override the dimension-getting functions in iScroll, it will act just like the standard iScroll.

Usage

The most basic usage of this widget is simple: just add a data-iscroll attribute to a container. This is the iScroll wrapper. The first child of the wrapper will be scrolled.

It does not use the typical JQuery Mobile data-role="something" attribute, because a common use case would be to use a data-role="content" div as the container, and, of course, you can't have two data-role attributes on the same element.

This widget has only been tested with a single scroller on a page. The widget will (normally) re-size the container to take up all available height within the viewport after fixed headers/footers are taken into account, and will make necessary adjustments to the page CSS. This can be disabled, and should only be enabled for one scroller on a given page.

The widget has been designed to support multiple scrolling regions on a page - for example, you might want a second, gallery-like horizontal scroll region. So, all data related to a scroller is stored in the scroller's container, not the page. Feel free to experiment with multiple scrollers - I just haven't had the need so haven't put the effort into testing and supporting that scenario.

Make sure to use a data-position="inline" attribute for headers and footers, not data-position="fixed". With data-position="fixed", and some versions of JQuery Mobile, the headers/footers will fade in/out. Since this widget resizes the scrolling region, there is no need for fixed positioning of header/footer.

Additional fixed-height elements (which are not headers or footers) outside of the scrolling region should be given the iscroll-foreground class.


Example

<div data-role="header" data-position="inline">
  <h1>INDEX PAGE</h1>
</div>

<div data-role="content" class='example-wrapper' data-iscroll>
  <div>
    some contents.
  </div>
</div>

<div data-role="footer" class="ui-bar" data-position="inline">
  <div data-role="navbar" class="ui-navbar">
    <ul class="ui-grid-b">
      <li class="ui-block-a"><a href="#home">home</a></li>
      <li class="ui-block-a"><a href="#timeline">timeline</a></li>
      <li class="ui-block-a"><a href="#message">message</a></li>
      <li class="ui-block-a"><a href="#bookmark">bookmark</a></li>
      <li class="ui-block-a"><a href="#config">config</a></li>
    </ul>
  </div>
</div>

Scrolling Inset Listviews

When scrolling a listview with data-inset=true, you may have a problem reaching the bottom of the scrolling area. You will be able to scroll to the bottom, but the scroller will then "snap back". This will occur also, if you are scrolling a series of listviews in some wrapper - for example, this is common on Setup pages when emulating iOS native appearance.

This appears to be due to the fact that iScroll either partially or completely ignores scroller margins.

You can correct this by wrapping your scroller with a <div> and adding some padding to the <div>. Because of the way iScroll works, you will only need to add padding to the bottom of this div. You should double the top or bottom margin of the inset listview being scrolled (since you are adding it all to the bottom.)

For example:

<div data-role="header" data-position="inline">
  <h1>INDEX PAGE</h1>
</div>

<div data-role="content" class='example-wrapper' data-iscroll>
  <div style="padding-bottom: 3em;">
    <ul data-inset="true">
      <li>Some content</li>
      <li Some content</li>
    </ul>
  </div>
</div>

<div data-role="footer" class="ui-bar" data-position="inline">
  Some footer text.
</div>

(Rather than using inline styles, it would be better to assign a class, and put the padding in a style sheet. The padding is shown inline above only for clarity.)

When scrolling a listview where data-inset="false" (the default), you do not need any additional padding, unless you wish to show padding around the listview. Generally, you will want padding around a data-inset="true" listview, both to allow it to scroll completely to the bottom, as well as to maintain the visual appearence of the inset. This is generally not needed or desired for data-inset="false" listviews.

Calling methods

The standard way of calling widget methods is by passing a sub-method name as a string parameter to the widget function. Any parameters to the method should follow.

For example, to call the refresh method:

  $(".example-wrapper").iscrollview("refresh")

The widget factory allows you to access widget methods directly, by accessing a data variable stored in the widget's element:

  $(".example-wrapper").jqmData('iscrollview').refresh();

While this is a bit awkward, it is also more conventional. It is handy in case you need to make a series of calls to different widget methods. You can first get the instance into a variable:

  var myView = $(".example-wrapper").jqmData("iscrollview");
  myView.refresh();

This means, as well, you can easily call any underlying iScroll method through the exposed iscroll member variable. For example,

  $(".example-wrapper").jqmData('iscrollview').iscroll.scrollTo(0,10,200,true);

So, if you replace iscroll.js with a newer version that has new methods, or if you need to call iScroll private methods, or access iScroll member variables, you can call them without any need to modify this widget.

This widget wraps all current iScroll public methods, so the above example can also be called like this:

  $(".example-wrapper").iscrollview("scrollTo", 0, 10, 200, true);

The exceptions are the destroy, refresh, enable, and disable methods.

destroy is a standard widget factory method. In this widget, it calls the iScroll destroy method and then calls the base widget destroy. If you need direct access to iScroll's destroy method, you can access it directly using the iscroll member variable.

The widget's refresh method insures that the underlying iScroll refresh method is called with the proper timing. If you need to call the iScroll refresh method directly, do so using the iscroll member variable.

enable and disable are standard widget methods. Each of these calls iScroll's corresponding method and then calls the underlying widget method.


Methods

###Standard Widget Methods

These are methods that are typically implemented for ALL widgets:

####option(key, [value_or_object])

See "Options", below.

####destroy()

Destroys the iScroll instance and removes page modifications. Also calls the underlying widget destroy() code.

###Custom Widget Methods

These are additional methods implemented in this widget which do not have corresponding iScroll methods.

####resizeWrapper()

This will resize the wrapper to use all available viewport space after accounting for headers, footers, and other fixed-height elements outside of the wrapper. This is normally done for you automatically, but the automatic resize can be overriden with an option. Call this if you have change the page structure and so need to resize the wrapper. This is also normally called for you when page orientation or page size changes.

####calculateBarsHeight()

This will re-calculate the height of header/footer etc. bars on the page. Call this prior to calling resizeWrapper(), if you change the height of header/footer etc. after the widget has been created.

####undoResizeWrapper()

Undoes the resize of the wrapper. Note that this can only change the wrapper size to what it was initially, prior to the very first call to resizeWrapper().

iScroll Methods

These are methods that exist in iScroll. They are available on the widget as a convenience. See "calling methods" for information on how to access any iScroll methods that are not implemented in the widget. (For example, because you have updated iScroll to a newer version, and this widget has not been updated yet.)

Please see the iScroll documentation for details on using these methods.

####refresh(timeout, callback, context)

Note that this performs the proper timing for the iScroll refresh() function using setTimeout. If you want to call the iScroll refresh() method directly, please see "calling methods" above.

If the timeout value is present, then the internal call of iScroll refresh() will be delayed by this value. If the value is null or undefined, then the value of the refreshDelay option will be used.

If present, the optional callback function will be called (with the supplied optional context parameter) upon completion of the refresh, which is asynchronous, since it waits for the DOM update to complete first. If you specify a callback function, then you MUST also specify a timeout.

This permits the application to perform some action (such as scrolling the the scroller, say if positioned at the end of a list) after the iScroll refresh function has completed and updated it's internal variables.

While one might use a refresh event for this, the callback eliminates any ambiguity as to which specific refresh() call has completed.

####scrollTo(x, y, time, relative)

####scrollToElement(el, time)

####scrollToPage(pageX, pageY, time)

####disable()

Note that this method also calls the default widget disable() method. If you want to call the iScroll disable() method directly, please see "calling methods" above.

####enable()

Note that this method also calls the default widget enable() method. If you want to call the iScroll enable() method directly, please see "calling methods" above.

####stop()

####zoom(x, y, scale, time)

iScroll Getters

This widget provides getters for some iScroll internal variables that might be useful to an application. These are all read-only.

For example, let's say you are adding elements to the end of a scrolled list. You'd like to scroll up (using scrollToElement) if the new element would be below the visible area. But if the list is intially empty, you'd want to avoid this until the scrolling area is initially full. So you need to compare the scroller height (scrollerH) to the wrapper height (wrapperH).

While wrapper and scroller height can be easily obtained using JQuery functions, these functions can still be useful because they reflect the precise internal state of the scroller.

x()

Current x origin (top) of the scroller.

y()

Current y origin (left) of the scroller.

wrapperW()

The width, in pixels, of the wrapper. This is the visible width of the scrolling area.

wrapperH()

The height, in pixels, of the wrapper. This is the visible height of the scrolling area.

scrollerW()

The width, in pixels, of the scroller. This is the total width of the scroller, including visible and non-visible portions.

scrollerH()

The height, in pixels, of the scroller. This is the total height of the scroller, including visible and non-visible portions.

iScroll Getters/Setters

This widget provides getters with options setter functionality for some iScroll internal variables that might be useful to an application. If a value is provided, then the functions act as setters. In any case, they return the value of the associated internal variable.

minScrollX(val)

The minimum X scroll position. This defines the left-most position of scroll. The user can scroll past the minimum X, but then the scroller will snap-back to the mimimum X position.

minScrollY(val)

The minimum Y scroll position. This defines the top-most position of the scroll. The user can scroll past the minimum Y, but then the scroller will snap-back to the minimum Y position.

Note that this variable is preset to -topOffset option value, and is useful when implementing pull-down-to-refresh.

maxScrollX(val)

The maximum X scroll position. This defines the right-most position of scroll. The user can scroll past the maximum X, but then the scroller will snap-back to the maximum X position.

maxScrollY(val)

The maximum Y scroll position. This defines the bottom-most position of the scroll. The user can drag past the maximum Y, but then the scroller will snap-back to the maximum Y position.

This is useful when implementing pull-up-to-refresh.

Options

Programatic access

This widget supports programmatic access to options using standard widget factory syntax:

.iscrollview("option");                   Returns an object with all options
.iscrollview("option", "hScroll");        Returns value of option
.iscrollview("option", "hScroll", true);  Sets option
.iscrollview("option", {hScroll: true});  Sets option, alternative syntax
.iscrollview("option", {hScroll: true, vScroll:true}; Set multiple options

The widget handles copying widget options to the iScroll object options and vice-versa.

Setting options in the data-iscroll attribute

This widget also supports setting options directly in the data-iscroll attribute. The options need to be in strict JSON format. This means that keys and string values need to be enclosed in double quotes only. Numeric and boolean values should not be enclosed in quotation marks.

Example:

<div data-role="content" data-iscroll='{"hScroll":true,"vScroll":false,"resizeEvents":"orientationchange"}' data-theme="c">

Modifying options after instantiation

If you modify an option after a scroller has been instantiated, the underlying iScroll object will be destroyed and re-created. This is because iScroll does not currently support modifying options after the object has been created.

However, unofficially, some options can be changed without destroying and re-creating the object. It is unclear exactly which options these are, and so this widget does not attempt it. There is skeletal code in the source that is commented-out to do this if you wish to experiment.

###Widget Options

The following options are available which affect the widget itself. These are not iScroll options.

####pageClass A CSS class, or a space-separated list of classes, which will be added to the page containing the wrapper.

Default: "iscroll-page"

####wrapperClass

A CSS class, or a space-separated list of classes, which will be added to the wrapper.

Default: "iscroll-wrapper"

####scrollerClass

A CSS class, or a space-separated list of classes, which will be added to the scroller. (The scroller is the first child of the wrapper, and is the element that will be scrolled.)

Default: "iscroll-scroller"

####adaptPage

If true, necessary adaptations will be made to the page to accommodate iScroll. If false, the adaptations will not be made. If multiple scrollers are used on the same page, only one of them should have adaptPage set to true. You can also set this false if you want to make the adaptations yourself.

Default: true

####fixedHeightSelector

A JQuery selector which selects the fixed-height elements on the page which are outside of the scrolling area. The heights of these elements will be added-up, and subtracted from the total viewport height to arrive at the wrapper height.

Default: "[data-role='header'], [data-role='footer'], .iscroll-foreground"

####resizeWrapper

If true, the wrapper will be resized to the height remaining after accounting for fixed-height elements.

Default: true

####resizeEvents

A space-separated list of events which will cause a resize of the wrapper.

In some mobile environments, it may be desirable to either substitute or add the orientationchange event. For iOS, however, the resize event works better, because the orientationchange event occurs to late too be useful, resulting in undesirable visual artifacts.

Default: "resize"

####refreshOnPageBeforeShow

If true, the scroller will be refreshed on every JQuery Mobile pageshow event. This should be set to true if scroller content might have changed asynchronously while the page was loaded into the DOM but not shown, as might happen in some native application environment. As well, this is necessary if not using a version of iScroll with overridable dimension-fetching functions, because it's not possible to determine the height of fixed-height elements prior to this event.

I've forked iScroll to make it possible to override iScrolls fetching of element dimensions. If you are using my version of iScroll, it's not necessary to refresh on pageshow, since this widget overrides this using jQuery.actual so that iScroll CAN get the dimensions of hidden elements. In this case, if you change page content while a page is hidden, you should be sure to call refresh() to insure that the scroll range and scrollbar(s) are updated.

Default: true if iScroll does not have overridable dimension-fetching functions false if iScroll does have overridable dimension-fetching functions

####fixInput

If true, applies a fix to allow input elements to work within a scroller. This is optional because there is an alternative fix that patches iScroll itself. You should disable this option if you are using the patched verison of iScroll, or in case it causes some sort of trouble.

Default: false

####wrapperAdd

Number of additional pixels to add to the wrapper height. This can be a positive or negative value. This is an "escape hatch" in case the calculation of wrapper height is not correct for some particular scenario.

Default: 0

####refreshDelay

Number of mSec to permit browser to render before performing refresh().

When you call refresh(), the call to iScroll's internal refresh() is delayed by this amount, to allow the browser to complete any rendering, prior to refreshing the scroller. If this value is 0, then a setTimeout() call is still used. This causes the script to relinquish control to the renderer, and may be sufficient for many browsers.

This value may need to be experimentally determined.

Alternately, you can specify a timeout value when calling refresh(). This is useful in case you have done some update which you know will require a lengthy render.

Default: 200 for Android, otherwise 50


Events

There are two ways to be notified when some event occurs in the widget. The widget exposes JQuery events that can be bound like any other event. The names are prepended with the name of the widget. So, the refresh event for this widget is actually iscrollviewrefresh.

Alternately, you can add callback functions to the options object. They key of the option corresponds to the event name without the widget name prefix. So, you can add a callback function for the refresh event with the key refresh.

When an event is triggered it will call the callback if defined in options, and, as well, trigger any bound events.

Bound event callbacks receive two parameters:

  • event - The underlying DOM event (if any) associated with this event
  • data - A map containing data passed to the event by this widget
  • :iscrollview reference to the iscrollview object associated with this event

As well, when a bound event callback is called, this will be the DOM object that triggered the event. (e.g. the wrapper).

In the case of callbacks that are specified in the options hash, this will be the underlying iscroll object. In this case, you will not get the event and iscrollview parameters. In other words, callbacks specified in the options hash work exactly as documented in the iScroll documentation.

In this case, you can still access the iscrollview easily, because the widget injects a reference to it into iScroll. I recommend that you don't use the option hash callbacks, though - it's much easier to use JQuery event binding.

###Example event delegation:

$(document).delegate("div.my-content", "iscrollviewrefresh", function(event, data) {
    var iscrollview = data.iscrollview;  // In case we need to reference the iscrollview
    console.write("iscrollviewrefresh occured");
    }

###Supported Events

  • iscrollviewrefresh
  • iscrollviewbeforescrollstart
  • iscrollviewscrollstart
  • iscrollviewbeforescrollmove
  • iscrollviewscrollmove
  • iscrollviewbeforescrollend
  • iscrollviewtouchend
  • iscrollviewdestroy
  • iscrollviewzoomstart
  • iscrollviewzoom
  • iscrollviewzoomend

Scroll Bars

Note: I am still investinging why scroll bars are sometimes created the full height and at the right hand side of the browser window (rather than the height at at the right hand side of the wrapper) in some browsers. It is not clear if this is a bug in iScroll or in this widget. For example, the scrollbar is correctly created in Safari Mobile, but not in desktop Safari.

In the mean time, this is a useful workaround, and can also be used if you want to customize the position of a scroll bar without fully-customizing it's appearance.

iScroll gives you the ability to customize scroll bars. See the iscroll4 documentation for full details. You can customize the height, width, position, color, etc. etc. of the scrollbar. To do so, you need to set the scrollbarClass option and then provide CSS to customize the scrollbar.

However, in many cases, all that is really desired is to set the position of the scrollbar. In this case, you can add some very minimal CSS.

In this case, do NOT set the scrollbarClass option. Setting this option causes iScroll to omit quite a bit of it's initialization of the scrollbar, and then you are required to supply a considerable amount of CSS.

Instead, you can usually use a CSS rule similar to this:

div.my-iscroll-wrapper > div:last-child {
  top: 46px !important;
  bottom: 22px !important;
}

iScroll appends the scrollbar to the end of your wrapper. Unless you have appended something else yourself, you can target the last child of the wrapper, and so you don't need the scrollbarClass to identify the scrollbar. So, iScroll will still do all of it's usual initialization. By using the !important modifier, your CSS will override the top and bottom locations that iScroll itself sets.

Multiple Scrolling Areas

If you wish to have multiple scrolling areas, please note the following:

  • The adaptPage option should be set to true for no more than one of your scrollers. Since it defaults to true, you will need to set it to false for all but (a maximum of) one of your scrollers.

  • The resizeWrapper option should be set to true for no more than one of your scrollers. If you have multiple scrollers one above the other, then at most one of them can be auto-sized. If you have multiple scrollers side-by-side, then you will probably have to size all them yourself. Since resizeWrapper is true by default, you will need to set the option to false for all but (a maximum) of one of your scrollers.

  • You should set the resizeEvents option to an empty string for those scrollers for which you have set resizeWrapper to false.

  • iScroll will not work correctly if scrollbars from multiple scrollers overlap. It will fail to scroll in all but one of the scrollers that have overlapping scrollbars. Please see the documentation on scrollbar customization, above.

Listviews With List Items That Are Buttons

Listviews that have list items that are buttons (i.e. the items are clickable, because they are wrapped in an <a> tag) can be very slow on touchscreen devices. This is not an iScroll or jquery.mobile.iscrollview problem per-se - it is inherent to JQuery Mobile.

There is a discussion of this issue here:

http://forum.jquery.com/topic/why-jqm-touchscreen-list-scrolling-performance-stinks-and-what-to-do-about-it

The gist of it is that as you scroll a list, your finger slips from one list item to the next, causing a "piano key" visual effect. The effect is both distracting to the user and slow.

A work-around is to insure that up/down/hover states for your scrollable listviews are identical, so that there is no hover effect and no selected effect. If you're developing exclusively for a touch-screen mobile device, there's little to no need for these effects. Users don't expect them: they expect something to happen when they tap, but not a useless effect.

This is an example of CSS overrides that will remove the hover and selected effects. You will need to modify this CSS to match your theme. The important thing is that the up, down, and hover states must have identical CSS, so that there is no transition between states.

/*
  Sane overrides for buttons in lists

  JQM default styling has up/down/hover/select styles for buttons. This is nice for real
  buttons, but slows list scrolling to a crawl. This can be avoided by styling the
  up/down/hover states identically.
*/

ul.ui-listview *.ui-btn-up-c,
ul.ui-listview *.ui-btn-down-c,
ul.ui-listview *.ui-btn-hover-c
   {
   border-color: #ccc;
   background: #eee;
   font-weight: bold;
   color: #444;
   text-shadow: 0 1px 1px #f6f6f6;
   background-image: -webkit-gradient(linear, left top, left bottom, from( #fdfdfd), to( #eee));
   }

ul.ui-listview *.ui-btn-up-c a.ui-link-inherit,
ul.ui-listview *.ui-btn-down-c a.ui-link-inherit,
ul.ui-listview *.ui-btn-hover-c a.ui-link-inherit
  { color: #444; }

Caching List Items

Webkit-based browsers can exhibit a "flicker" effect when initially scrolling. Once you have scrolled down to the bottom of the list, the flicker will typically stop.

This issue is discussed here:

http://cubiq.org/you-shall-not-flicker

A work-around for this issue to to force list items to be pre-cached. See the above link for why this works.

You can implement this fix in your CSS like this:

/* Force list items to be cached
   See: cubiq.org/you-shall-not-flicker */
 .iscroll-scroller,
 .iscroll-scroller * {
   -webkit-transition-duration: 0;
   -webkit-transform: translate3d(0,0,0);
   }

Demo

The demo directory contains a simple example of a JQuery Mobile page using a scrollview to scroll a long list of items. To demo, simply open the index.html file in your browser.

As a convenience, the demo directory is self-contained (except for the widget, which is expected to be found in the parent directory), and contains the following additional components:

  • JQuery 1.6.4
  • JQuery Mobile 1.0.1
  • iscroll4, commit 712640b7de..., Apr. 10, 2012
  • jquery.actual, commit 0530ce5c64..., Feb 22, 2012

Please obtain these components independently for your projects, so that you can be sure of what version you are getting, and avail yourself of any updates/improvements.


Testing

This widget has only undergone ad-hoc testing, primarily with the components included in the demo directory. Contribution of a test suite would be most welcome. :)

Most testing has been done using JQuery 1.6.4/JQuery Mobile 1.0.1. Some rudimentary testing has been done using JQuery 1.7.1/JQuery Mobile 1.1.0, with no obvious problems noted.


Variable names

The source code code follows the following conventions:

  • Upper-case first letter: constant
  • $ first letter: variable contains a JQuery object
  • first letter: Private method

Bugs and Enhancements

Please submit bug and enhancement requests via jquery.mobile.iscrollview gitHub Issues If you have developed code that you would like to have incorporated in a future release of this widget, please submit it for consideration via a gitHub pull request.

License

Copyright (c), 2012 Watusiware Corporation Distributed under the MIT License

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: NO ADDITIONAl CONDITIONS.

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

Derived in part from jquery.mobile.iscroll.js:
Portions Copyright (c) Kazuhiro Osawa
Dual licensed under the MIT or GPL Version 2 licenses.

Derived in part from (jQuery mobile) jQuery UI Widget-factory
plugin boilerplate (for 1.8/9+)
Author: @scottjehl
Further changes: @addyosmani
Licensed under the MIT license