├── .eslintignore
├── .eslintrc
├── .gitignore
├── .jscsrc
├── .npmignore
├── .travis.yml
├── ControlledRefreshableListView.js
├── README.md
├── RefreshableListView.js
├── RefreshingIndicator.js
├── index.js
├── lib
├── ControlledRefreshableListView.js
├── ListView.js
├── RefreshableListView.js
├── RefreshingIndicator.js
├── __mocks__
│ ├── ListView.js
│ ├── createElementFrom.js
│ └── react-native.js
├── __tests__
│ ├── ControlledRefreshableListView-test.js
│ ├── RefreshableListView-test.js
│ ├── RefreshingIndicator-test.js
│ └── delay-test.js
├── createElementFrom.js
├── delay.js
└── scrollToCompat.js
├── package.json
└── test
├── RefreshableListViewTest
├── .flowconfig
├── .gitignore
├── .watchmanconfig
├── android
│ ├── app
│ │ ├── build.gradle
│ │ ├── proguard-rules.pro
│ │ ├── react.gradle
│ │ └── src
│ │ │ └── main
│ │ │ ├── AndroidManifest.xml
│ │ │ ├── java
│ │ │ └── com
│ │ │ │ └── refreshablelistviewtest
│ │ │ │ └── MainActivity.java
│ │ │ └── res
│ │ │ ├── mipmap-hdpi
│ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-mdpi
│ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xhdpi
│ │ │ └── ic_launcher.png
│ │ │ ├── mipmap-xxhdpi
│ │ │ └── ic_launcher.png
│ │ │ └── values
│ │ │ ├── strings.xml
│ │ │ └── styles.xml
│ ├── build.gradle
│ ├── gradle.properties
│ ├── gradle
│ │ └── wrapper
│ │ │ ├── gradle-wrapper.jar
│ │ │ └── gradle-wrapper.properties
│ ├── gradlew
│ ├── gradlew.bat
│ └── settings.gradle
├── index.android.js
├── index.ios.js
├── ios
│ ├── RefreshableListViewTest.xcodeproj
│ │ ├── project.pbxproj
│ │ └── xcshareddata
│ │ │ └── xcschemes
│ │ │ └── RefreshableListViewTest.xcscheme
│ ├── RefreshableListViewTest
│ │ ├── AppDelegate.h
│ │ ├── AppDelegate.m
│ │ ├── Base.lproj
│ │ │ └── LaunchScreen.xib
│ │ ├── Images.xcassets
│ │ │ └── AppIcon.appiconset
│ │ │ │ └── Contents.json
│ │ ├── Info.plist
│ │ └── main.m
│ └── RefreshableListViewTestTests
│ │ ├── Info.plist
│ │ └── RefreshableListViewTestTests.m
└── package.json
└── support
├── preprocessor.js
├── setupTestFramework.js
├── shallowRender.js
└── sourceMapPath.js
/.eslintignore:
--------------------------------------------------------------------------------
1 | example/
2 | test/
3 | **/__tests__/**/*.js
4 | **/__mocks__/**/*.js
5 |
--------------------------------------------------------------------------------
/.eslintrc:
--------------------------------------------------------------------------------
1 | {
2 | "ecmaFeatures": {
3 | "globalReturn": true,
4 | "jsx": true,
5 | "modules": true
6 | },
7 |
8 | "env": {
9 | "browser": true,
10 | "es6": true,
11 | "node": true
12 | },
13 |
14 | "globals": {
15 | "document": false,
16 | "escape": false,
17 | "navigator": false,
18 | "unescape": false,
19 | "window": true
20 | },
21 |
22 | "plugins": [
23 | "react"
24 | ],
25 |
26 | "rules": {
27 | "block-scoped-var": 0,
28 | "brace-style": [2, "1tbs", { "allowSingleLine": true }],
29 | "camelcase": 0,
30 | "comma-dangle": [2, "always-multiline"],
31 | "comma-spacing": [2, { "before": false, "after": true }],
32 | "comma-style": [2, "last"],
33 | "complexity": 0,
34 | "consistent-return": 0,
35 | "consistent-this": 0,
36 | "curly": [2, "multi-line"],
37 | "default-case": 0,
38 | "dot-notation": 0,
39 | "eol-last": 2,
40 | "eqeqeq": [2, "allow-null"],
41 | "func-names": 0,
42 | "func-style": [0, "declaration"],
43 | "generator-star": [2, "middle"],
44 | "guard-for-in": 0,
45 | "handle-callback-err": [2, "^(err|error|anySpecificError)$" ],
46 | "indent": [2, 2],
47 | "key-spacing": [2, { "beforeColon": false, "afterColon": true }],
48 | "max-depth": 0,
49 | "max-len": 0,
50 | "max-nested-callbacks": 0,
51 | "max-params": 0,
52 | "max-statements": 0,
53 | "new-cap": [2, { "newIsCap": true, "capIsNew": false }],
54 | "new-parens": 2,
55 | "no-alert": 2,
56 | "no-array-constructor": 2,
57 | "no-bitwise": 0,
58 | "no-caller": 2,
59 | "no-catch-shadow": 0,
60 | "no-cond-assign": 2,
61 | "no-console": 0,
62 | "no-constant-condition": 0,
63 | "no-control-regex": 2,
64 | "no-debugger": 2,
65 | "no-delete-var": 2,
66 | "no-div-regex": 0,
67 | "no-dupe-args": 2,
68 | "no-dupe-keys": 2,
69 | "no-duplicate-case": 2,
70 | "no-else-return": 0,
71 | "no-empty": 0,
72 | "no-empty-class": 2,
73 | "no-empty-label": 2,
74 | "no-eq-null": 0,
75 | "no-eval": 2,
76 | "no-ex-assign": 2,
77 | "no-extend-native": 2,
78 | "no-extra-bind": 2,
79 | "no-extra-boolean-cast": 2,
80 | "no-extra-parens": 0,
81 | "no-extra-semi": 0,
82 | "no-extra-strict": 0,
83 | "no-fallthrough": 2,
84 | "no-floating-decimal": 2,
85 | "no-func-assign": 2,
86 | "no-implied-eval": 2,
87 | "no-inline-comments": 0,
88 | "no-inner-declarations": [2, "functions"],
89 | "no-invalid-regexp": 2,
90 | "no-irregular-whitespace": 2,
91 | "no-iterator": 2,
92 | "no-label-var": 2,
93 | "no-labels": 2,
94 | "no-lone-blocks": 2,
95 | "no-lonely-if": 0,
96 | "no-loop-func": 0,
97 | "no-mixed-requires": [0, false],
98 | "no-mixed-spaces-and-tabs": [2, false],
99 | "no-multi-spaces": 2,
100 | "no-multi-str": 2,
101 | "no-multiple-empty-lines": [2, { "max": 1 }],
102 | "no-native-reassign": 2,
103 | "no-negated-in-lhs": 2,
104 | "no-nested-ternary": 0,
105 | "no-new": 2,
106 | "no-new-func": 2,
107 | "no-new-object": 2,
108 | "no-new-require": 2,
109 | "no-new-wrappers": 2,
110 | "no-obj-calls": 2,
111 | "no-octal": 2,
112 | "no-octal-escape": 2,
113 | "no-path-concat": 0,
114 | "no-plusplus": 0,
115 | "no-process-env": 0,
116 | "no-process-exit": 0,
117 | "no-proto": 2,
118 | "no-redeclare": 2,
119 | "no-regex-spaces": 2,
120 | "no-reserved-keys": 0,
121 | "no-restricted-modules": 0,
122 | "no-return-assign": 2,
123 | "no-script-url": 0,
124 | "no-self-compare": 2,
125 | "no-sequences": 2,
126 | "no-shadow": 0,
127 | "no-shadow-restricted-names": 2,
128 | "no-spaced-func": 2,
129 | "no-sparse-arrays": 2,
130 | "no-sync": 0,
131 | "no-ternary": 0,
132 | "no-throw-literal": 2,
133 | "no-trailing-spaces": 2,
134 | "no-undef": 2,
135 | "no-undef-init": 2,
136 | "no-undefined": 0,
137 | "no-underscore-dangle": 0,
138 | "no-unreachable": 2,
139 | "no-unused-expressions": 0,
140 | "no-unused-vars": [2, { "vars": "all", "args": "none" }],
141 | "no-use-before-define": 0,
142 | "no-var": 0,
143 | "no-void": 0,
144 | "no-warning-comments": [0, { "terms": ["todo", "fixme", "xxx"], "location": "start" }],
145 | "no-with": 2,
146 | "no-wrap-func": 2,
147 | "one-var": 0,
148 | "operator-assignment": [0, "always"],
149 | "padded-blocks": [2, "never"],
150 | "quote-props": 0,
151 | "quotes": [2, "single", "avoid-escape"],
152 | "radix": 2,
153 | "react/jsx-boolean-value": 2,
154 | "react/jsx-quotes": [2, "double", "avoid-escape"],
155 | "react/jsx-no-undef": 2,
156 | "react/jsx-sort-props": 0,
157 | "react/jsx-uses-react": 2,
158 | "react/jsx-uses-vars": 2,
159 | "react/no-did-mount-set-state": 2,
160 | "react/no-did-update-set-state": 2,
161 | "react/no-multi-comp": 2,
162 | "react/no-unknown-property": 2,
163 | "react/prop-types": 2,
164 | "react/react-in-jsx-scope": 2,
165 | "react/self-closing-comp": 2,
166 | "react/wrap-multilines": 2,
167 | "semi": [2, "never"],
168 | "semi-spacing": 0,
169 | "sort-vars": 0,
170 | "space-after-keywords": [2, "always"],
171 | "space-before-blocks": [2, "always"],
172 | "space-before-function-paren": [2, "never"],
173 | "space-in-brackets": 0,
174 | "space-in-parens": [2, "never"],
175 | "space-infix-ops": 2,
176 | "space-return-throw-case": 2,
177 | "space-unary-ops": [2, { "words": true, "nonwords": false }],
178 | "spaced-line-comment": [2, "always"],
179 | "strict": 0,
180 | "use-isnan": 2,
181 | "valid-jsdoc": 0,
182 | "valid-typeof": 2,
183 | "vars-on-top": 0,
184 | "wrap-iife": [2, "any"],
185 | "wrap-regex": 0,
186 | "yoda": [2, "never"]
187 | }
188 | }
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Ignore docs files
2 | _gh_pages
3 | _site
4 | .ruby-version
5 |
6 | # Numerous always-ignore extensions
7 | *.diff
8 | *.err
9 | *.orig
10 | *.log
11 | *.rej
12 | *.swo
13 | *.swp
14 | *.zip
15 | *.vi
16 | *~
17 |
18 | # OS or Editor folders
19 | .DS_Store
20 | ._*
21 | Thumbs.db
22 | .cache
23 | .project
24 | .settings
25 | .tmproj
26 | *.esproj
27 | nbproject
28 | *.sublime-project
29 | *.sublime-workspace
30 | .idea
31 |
32 | # Komodo
33 | *.komodoproject
34 | .komodotools
35 |
36 | # Folders to ignore
37 | node_modules
38 | bower_components
39 |
40 | example/output/
41 |
42 | main.jsbundle
43 |
44 | images/generated/
45 |
46 | .gradle
47 |
--------------------------------------------------------------------------------
/.jscsrc:
--------------------------------------------------------------------------------
1 | {
2 | "esnext": true,
3 | "preset": "node-style-guide",
4 | "maximumLineLength": null,
5 | "requireCapitalizedComments": null,
6 | "requireEarlyReturn": null,
7 | "disallowSemicolons": true,
8 | "disallowMultipleSpaces": true,
9 | "disallowPaddingNewlinesInBlocks": true,
10 | "requireCurlyBraces": null,
11 | "requireLineFeedAtFileEnd": true,
12 | "requireSpaceBeforeKeywords": [
13 | "else",
14 | "while",
15 | "catch"
16 | ]
17 | }
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | test
2 | **/__tests__/**/*.js
3 | **/__mocks__/**/*.js
4 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | language: node_js
2 | node_js:
3 | - "4.2"
--------------------------------------------------------------------------------
/ControlledRefreshableListView.js:
--------------------------------------------------------------------------------
1 | var ControlledRefreshableListView = require('./lib/ControlledRefreshableListView')
2 | var {DataSource} = require('./lib/ListView')
3 | var RefreshingIndicator = require('./lib/RefreshingIndicator')
4 |
5 | Object.assign(ControlledRefreshableListView, {
6 | DataSource,
7 | RefreshingIndicator,
8 | })
9 |
10 | module.exports = ControlledRefreshableListView
11 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # React Native RefreshableListView
2 | A pull-to-refresh ListView which shows a loading spinner while your data reloads
3 |
4 | *Deprecated*: now you can use the built-in [RefreshControl](https://facebook.github.io/react-native/docs/refreshcontrol.html) instead.
5 |
6 | 
7 |
8 | In action (from [ReactNativeHackerNews](https://github.com/jsdf/ReactNativeHackerNews)):
9 |
10 | 
11 |
12 | ## Usage
13 |
14 | **Note:** these are the docs for the 2.x branch, currently in beta. If you are looking for the docs for a 1.x version, see the [1.x branch](https://github.com/jsdf/react-native-refreshable-listview/tree/1.x).
15 |
16 | You can install the latest beta with `npm install react-native-refreshable-listview@next`
17 |
18 | ### RefreshableListView
19 | Replace a ListView with a RefreshableListView to add pulldown-to-refresh
20 | functionality. Accepts the same props as ListView, plus a few extras (see the **props** definitions below).
21 |
22 | ```js
23 | var React = require('react-native')
24 | var {Text, View, ListView} = React
25 | var RefreshableListView = require('react-native-refreshable-listview')
26 |
27 | var ArticleStore = require('../stores/ArticleStore')
28 | var StoreWatchMixin = require('./StoreWatchMixin')
29 | var ArticleView = require('./ArticleView')
30 |
31 | var ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2}) // assumes immutable objects
32 |
33 | var ArticlesScreen = React.createClass({
34 | mixins: [StoreWatchMixin],
35 | getInitialState() {
36 | return {dataSource: ds.cloneWithRows(ArticleStore.all())}
37 | },
38 | getStoreWatches() {
39 | this.watchStore(ArticleStore, () => {
40 | this.setState({dataSource: ds.cloneWithRows(ArticleStore.all())})
41 | })
42 | },
43 | reloadArticles() {
44 | // returns a Promise of reload completion
45 | // for a Promise-free version see ControlledRefreshableListView below
46 | return ArticleStore.reload()
47 | },
48 | renderArticle(article) {
49 | return
50 | },
51 | render() {
52 | return (
53 |
59 | )
60 | }
61 | })
62 | ```
63 |
64 | #### Props
65 |
66 | - `loadData: func.isRequired`
67 | A function returning a Promise or taking a callback, invoked upon pulldown.
68 | The refreshing indicator (spinner) will show until the Promise resolves or the callback
69 | is called.
70 | - `refreshDescription: oneOfType([string, element])`
71 | Text/element to show alongside spinner. If a custom
72 | `refreshingIndicatorComponent` is used this value will be passed as its
73 | `description` prop.
74 | - `refreshingIndicatorComponent: oneOfType([func, element])`
75 | Content to show in list header when refreshing. Can be a component class or
76 | instantiated element. Defaults to `RefreshableListView.RefreshingIndicator`.
77 | You can easily customise the appearance of the indicator by passing in a
78 | customised ``, or provide your own
79 | entirely custom content to be displayed.
80 | - `minDisplayTime: number`
81 | Minimum time the spinner will show for.
82 | - `minBetweenTime: number`
83 | Minimum time after a refresh before another refresh can be performed.
84 | - `minPulldownDistance: number`
85 | Minimum distance (in px) which the list has to be scrolled off the top to
86 | trigger a refresh.
87 | - `ignoreInertialScroll: bool`
88 | Require the user to be actually touching the screen when the pulldown distance
89 | exceeds `minPulldownDistance` to trigger a refresh (eg. not just inertially
90 | scrolling off the top). Defaults to `true`.
91 | - `onScroll: func`
92 | An event handler for the `onScroll` event which will be chained after the one
93 | defined by the `RefreshableListView`.
94 | - `scrollEventThrottle: number`
95 | How often `ListView` produces scroll events, in ms. Defaults to a fairly low
96 | value, try setting it higher if you encounter performance issues. Keep in mind
97 | that a higher value will make the pulldown-to-refresh behaviour less responsive.
98 | - `colors: array of strings`
99 | Colors to be used for pull to refresh indicator in Android
100 | - `progressBackgroundColor: string`
101 | Color to be used for pull to refresh indicator background in Android
102 |
103 |
104 | ### ControlledRefreshableListView
105 | Low level component used by `RefreshableListView`. Use this directly if you want
106 | to manually control the refreshing status (rather than using a Promise).
107 |
108 | This component is more suitable for use in a [Redux](https://github.com/rackt/redux)-style connected component.
109 |
110 | ```js
111 | var React = require('react-native')
112 | var {Text, View, ListView} = React
113 | var {ControlledRefreshableListView} = require('react-native-refreshable-listview')
114 |
115 | var ArticleView = require('./ArticleView')
116 |
117 | var ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2}) // assumes immutable objects
118 |
119 | var ArticlesScreen = React.createClass({
120 | propTypes: {
121 | // eg. props mapped from store state
122 | articles: React.PropTypes.array.isRequired,
123 | isRefreshingArticles: React.PropTypes.bool.isRequired,
124 | // eg. a bound action creator
125 | refreshArticles: React.PropTypes.func.isRequired,
126 | },
127 | getInitialState() {
128 | return {dataSource: ds.cloneWithRows(this.props.articles)}
129 | },
130 | componentWillReceiveProps(nextProps) {
131 | if (this.props.articles !== nextProps.articles) {
132 | this.setState({dataSource: ds.cloneWithRows(nextProps.articles)})
133 | }
134 | },
135 | renderArticle(article) {
136 | return
137 | },
138 | render() {
139 | return (
140 |
147 | )
148 | }
149 | })
150 | ```
151 |
152 | #### Props
153 | - `onRefresh: func.isRequired`
154 | Called when user pulls listview down to refresh.
155 | - `isRefreshing: bool.isRequired`
156 | Whether or not to show the refreshing indicator.
157 | - `refreshDescription: oneOfType([string, element])`
158 | *See `RefreshableListView`*
159 | - `refreshingIndicatorComponent: oneOfType([func, element])`
160 | *See `RefreshableListView`*
161 | - `minPulldownDistance: number`
162 | *See `RefreshableListView`*
163 | - `ignoreInertialScroll: bool`
164 | *See `RefreshableListView`*
165 | - `onScroll: func`
166 | *See `RefreshableListView`*
167 | - `scrollEventThrottle: number`
168 | *See `RefreshableListView`*
169 |
170 | ### RefreshableListView.RefreshingIndicator
171 | Component with activity indicator to be displayed in list header when refreshing.
172 | (also exposed as `ControlledRefreshableListView.RefreshingIndicator`)
173 |
174 | #### Props
175 |
176 | - `description: oneOfType([string, element])`
177 | Text/element to show alongside spinner.
178 | - `stylesheet: object`
179 | A stylesheet object which overrides one or more of the styles defined in the
180 | [RefreshingIndicator stylesheet](lib/RefreshingIndicator.js).
181 | - `activityIndicatorComponent: oneOfType([func, element])`
182 | The spinner to display. Defaults to ``.
183 |
184 | ### RefreshableListView.DataSource, ControlledRefreshableListView.DataSource
185 | Aliases of `ListView.DataSource`, for convenience.
186 |
187 | ## Howto
188 |
189 | ### Customise the refresh indicator (spinner)
190 |
191 | Your first option is to style the default RefreshingIndicator:
192 |
193 | ```js
194 | var indicatorStylesheet = StyleSheet.create({
195 | wrapper: {
196 | backgroundColor: 'red',
197 | height: 60,
198 | marginTop: 10,
199 | },
200 | content: {
201 | backgroundColor: 'blue',
202 | marginTop: 10,
203 | height: 60,
204 | },
205 | })
206 |
207 |
210 | }
211 | />
212 | ```
213 |
214 | Alternatively, you can provide a custom RefreshingIndicator:
215 |
216 | ```js
217 | var MyRefreshingIndicator = React.createClass({
218 | render() {
219 | return (
220 |
221 |
222 | {this.props.description}
223 |
224 | )
225 | },
226 | })
227 |
228 |
229 | // or
230 | } />
231 | ```
232 |
233 | ## Changelog
234 |
235 | - **2.0.0-beta4**
236 | - added Android support! (@maraujop)
237 |
238 | - **2.0.0-beta3**
239 | - fixed proptype warnings from internal component
240 | - adjusted default styling of refreshing indicator
241 |
242 | - **2.0.0-beta2**
243 | - pulling down now reveals refreshing indicator behind current view, rather
244 | than the refreshing indicator moving down from the top of the screen
245 | - `renderHeaderWrapper` is no longer used
246 | - fixed infinite refreshing when pulled down
247 |
248 | - **1.2.0**
249 | - deprecated `renderHeader` in favour of `renderHeaderWrapper` as some
250 | developers seemed to be confused by the fact that a `renderHeader` handler
251 | for a standard `ListView` will not automatically *just work* with this component,
252 | but rather needs to be modified as described in the documentation. The new prop
253 | `renderHeaderWrapper` works identically to the previous one, however hopefully
254 | now it is named differently it will be more apparent that its behaviour is not
255 | the same as with `ListView`. The `renderHeader` prop will be removed in 2.0.
256 | - **1.1.0**
257 | - added behaviour to ignore inertial scrolling (@dhrrgn)
258 | - exposed props: ignoreInertialScroll, scrollEventThrottle
259 | - **1.0.0**
260 | - Split RefreshableListView into 3 parts:
261 | - RefreshableListView handles 'refreshing' state by invoking 'loadData'
262 | callback and waiting for resolution.
263 | - ControlledRefreshableListView handles rendering of ListView header,
264 | depending on isRefreshing prop. Calls onRefresh handler when
265 | pulldown-to-refresh scroll motion occurs.
266 | - RefreshingIndicator is the component rendered in the header of the
267 | ListView when refreshing. Pass in a customised version of this (or a
268 | completely different component) to RefreshableListView or
269 | ControlledRefreshableListView if you want to customise refresh indicator
270 | appearance.
271 | - Added Jest unit tests
272 | - **0.3.0** added minPulldownTime & minBetweenTime props, fixed bug where
273 | refresh could happen twice
274 | - **0.2.0** added support for ListView props setNativeProps and
275 | getScrollResponder (@almost & @sscotth)
276 |
--------------------------------------------------------------------------------
/RefreshableListView.js:
--------------------------------------------------------------------------------
1 | var RefreshableListView = require('./lib/RefreshableListView')
2 | var ControlledRefreshableListView = require('./ControlledRefreshableListView')
3 | var {DataSource} = require('./lib/ListView')
4 | var RefreshingIndicator = require('./lib/RefreshingIndicator')
5 |
6 | Object.assign(RefreshableListView, {
7 | DataSource,
8 | RefreshingIndicator,
9 | ControlledRefreshableListView,
10 | })
11 |
12 | module.exports = RefreshableListView
13 |
--------------------------------------------------------------------------------
/RefreshingIndicator.js:
--------------------------------------------------------------------------------
1 | module.exports = require('./lib/RefreshingIndicator')
2 |
--------------------------------------------------------------------------------
/index.js:
--------------------------------------------------------------------------------
1 | module.exports = require('./RefreshableListView')
2 |
--------------------------------------------------------------------------------
/lib/ControlledRefreshableListView.js:
--------------------------------------------------------------------------------
1 | var React = require('react-native')
2 | var {
3 | PropTypes,
4 | StyleSheet,
5 | View,
6 | Platform,
7 | PullToRefreshViewAndroid,
8 | } = React
9 | var ListView = require('./ListView')
10 | var createElementFrom = require('./createElementFrom')
11 | var RefreshingIndicator = require('./RefreshingIndicator')
12 | var scrollToCompat = require('./scrollToCompat')
13 |
14 | const SCROLL_EVENT_THROTTLE = 32
15 | const MIN_PULLDOWN_DISTANCE = 40
16 | const REFRESHING_INDICATOR_HEIGHT = 60
17 | const LISTVIEW_REF = 'listview'
18 |
19 | /*
20 | * state transitions:
21 | * {isRefreshing: false}
22 | * v - show loading spinner
23 | * {isRefreshing: true, waitingForRelease: true}
24 | * v - reset scroll position, offset scroll top
25 | * {isRefreshing: true, waitingForRelease: false}
26 | * v - hide loading spinner
27 | * {isRefreshing: false}
28 | */
29 |
30 | var ControlledRefreshableListView = React.createClass({
31 | propTypes: {
32 | colors: PropTypes.array,
33 | progressBackgroundColor: PropTypes.string,
34 | onRefresh: PropTypes.func.isRequired,
35 | isRefreshing: PropTypes.bool.isRequired,
36 | waitingForRelease: PropTypes.bool,
37 | onHold: PropTypes.func,
38 | onPull: PropTypes.func,
39 | pullingPrompt: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
40 | pullingIndicator: PropTypes.oneOfType([PropTypes.func, PropTypes.element]),
41 | holdingPrompt: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
42 | holdingIndicator: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
43 | refreshDescription: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
44 | refreshingPrompt: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
45 | refreshingIndicator: PropTypes.oneOfType([PropTypes.func, PropTypes.element]),
46 | refreshingIndicatorComponent: PropTypes.oneOfType([PropTypes.func, PropTypes.element]),
47 | minPulldownDistance: PropTypes.number,
48 | ignoreInertialScroll: PropTypes.bool,
49 | scrollEventThrottle: PropTypes.number,
50 | onScroll: PropTypes.func,
51 | onResponderGrant: PropTypes.func,
52 | onResponderRelease: PropTypes.func,
53 | renderHeaderWrapper: (props, propName, componentName) => {
54 | if (props[propName]) {
55 | return new Error('The \'renderHeaderWrapper\' prop is no longer used')
56 | }
57 | },
58 | refreshingIndictatorComponent: (props, propName, componentName) => {
59 | if (props[propName]) {
60 | return new Error('The \'refreshingIndictatorComponent\' prop has been renamed to \'refreshingIndicatorComponent\'')
61 | }
62 | },
63 | },
64 | getDefaultProps() {
65 | return {
66 | minPulldownDistance: MIN_PULLDOWN_DISTANCE,
67 | scrollEventThrottle: SCROLL_EVENT_THROTTLE,
68 | ignoreInertialScroll: true,
69 | refreshingIndicatorComponent: RefreshingIndicator,
70 | pullingPrompt: 'Pull to refresh',
71 | holdingPrompt: 'Release to refresh',
72 | }
73 | },
74 | getInitialState() {
75 | return {
76 | waitingForRelease: false,
77 | }
78 | },
79 | componentWillReceiveProps(nextProps) {
80 | if (!this.props.isRefreshing && nextProps.isRefreshing && this.isTouching) {
81 | this.waitingForRelease = true
82 | this.setState({waitingForRelease: true})
83 | }
84 | },
85 | componentWillUpdate(nextProps, nextState) {
86 | if (Platform.OS === 'ios') {
87 | if (
88 | this.isReleaseUpdate(this.props, this.state, nextProps, nextState)
89 | ) {
90 | scrollToCompat(
91 | this.getScrollResponder(),
92 | -(this.lastContentInsetTop + REFRESHING_INDICATOR_HEIGHT),
93 | this.lastContentOffsetX,
94 | false
95 | )
96 | }
97 | }
98 | },
99 | componentDidUpdate(prevProps, prevState) {
100 | if (Platform.OS === 'ios') {
101 | if (
102 | this.isReleaseUpdate(prevProps, prevState, this.props, this.state)
103 | ) {
104 | scrollToCompat(
105 | this.getScrollResponder(),
106 | -(this.lastContentInsetTop),
107 | this.lastContentOffsetX,
108 | false
109 | )
110 | }
111 | }
112 | },
113 | handlePullToRefreshViewAndroidRef(swipeRefreshLayout) {
114 | this.swipeRefreshLayout = swipeRefreshLayout
115 | },
116 | handleScroll(e) {
117 | var scrollY = e.nativeEvent.contentInset.top + e.nativeEvent.contentOffset.y
118 | this.lastScrollY = scrollY
119 | this.lastContentInsetTop = e.nativeEvent.contentInset.top
120 | this.lastContentOffsetX = e.nativeEvent.contentOffset.x
121 |
122 | if (!this.props.isRefreshing) {
123 | if ((this.isTouching && scrollY < 0) || (!this.isTouching && !this.props.ignoreInertialScroll)) {
124 | if (scrollY < -this.props.minPulldownDistance) {
125 | if (!this.isWaitingForRelease()) {
126 | this.waitingForRelease = true
127 | this.setState({waitingForRelease: true})
128 | this.props.onHold()
129 | }
130 | } else {
131 | if (this.isWaitingForRelease()) {
132 | this.waitingForRelease = false
133 | this.setState({waitingForRelease: false})
134 | }
135 | this.props.onPull()
136 | }
137 | }
138 | }
139 |
140 | this.props.onScroll && this.props.onScroll(e)
141 | },
142 | handleResponderGrant() {
143 | this.isTouching = true
144 | if (this.props.onResponderGrant) {
145 | this.props.onResponderGrant.apply(null, arguments)
146 | }
147 | },
148 | handleResponderRelease() {
149 | this.isTouching = false
150 | if (this.props.onResponderRelease) {
151 | this.props.onResponderRelease.apply(null, arguments)
152 | }
153 | if (this.isWaitingForRelease()) {
154 | this.waitingForRelease = false
155 | this.setState({waitingForRelease: false})
156 | if (!this.props.isRefreshing) {
157 | if (this.props.onRefresh) {
158 | this.props.onRefresh()
159 | }
160 | }
161 | }
162 | this.props.onPull()
163 | },
164 | getContentContainerStyle() {
165 | if (!this.props.isRefreshing || this.isWaitingForRelease()) return null
166 |
167 | return {marginTop: REFRESHING_INDICATOR_HEIGHT}
168 | },
169 | getScrollResponder() {
170 | return this.refs[LISTVIEW_REF].getScrollResponder()
171 | },
172 | setNativeProps(props) {
173 | this.refs[LISTVIEW_REF].setNativeProps(props)
174 | },
175 | isWaitingForRelease() {
176 | return this.waitingForRelease || this.props.waitingForRelease
177 | },
178 | isReleaseUpdate(oldProps, oldState, newProps, newState) {
179 | return (
180 | (!oldProps.isRefreshing && newProps.isRefreshing && !this.waitingForRelease) ||
181 | (oldProps.isRefreshing && oldState.waitingForRelease && !newState.waitingForRelease)
182 | )
183 | },
184 | renderRefreshingIndicator() {
185 | var {
186 | isRefreshing,
187 | pullingPrompt,
188 | holdingPrompt,
189 | refreshingPrompt,
190 | refreshDescription,
191 | pullingIndicator,
192 | holdingIndicator,
193 | refreshingIndicator,
194 | } = this.props
195 | var refreshingIndicatorProps = {
196 | isRefreshing,
197 | pullingIndicator,
198 | holdingIndicator,
199 | refreshingIndicator,
200 | pullingPrompt: pullingPrompt,
201 | holdingPrompt: holdingPrompt,
202 | refreshingPrompt: refreshingPrompt || refreshDescription,
203 | isTouching: this.isTouching,
204 | isWaitingForRelease: this.isWaitingForRelease(),
205 | }
206 | return createElementFrom(this.props.refreshingIndicatorComponent, refreshingIndicatorProps)
207 | },
208 | render() {
209 | if (Platform.OS === 'android') {
210 | return (
211 |
218 |
222 |
223 | )
224 | } else {
225 | return (
226 |
227 |
228 | {this.renderRefreshingIndicator()}
229 |
230 |
231 |
240 |
241 |
242 | )
243 | }
244 | },
245 | })
246 |
247 | var stylesheet = StyleSheet.create({
248 | container: {
249 | flex: 1,
250 | },
251 | fillParent: {
252 | backgroundColor: 'transparent',
253 | position: 'absolute',
254 | top: 0,
255 | left: 0,
256 | right: 0,
257 | bottom: 0,
258 | },
259 | })
260 |
261 | module.exports = ControlledRefreshableListView
262 |
--------------------------------------------------------------------------------
/lib/ListView.js:
--------------------------------------------------------------------------------
1 | var {ListView} = require('react-native')
2 |
3 | module.exports = ListView
4 |
--------------------------------------------------------------------------------
/lib/RefreshableListView.js:
--------------------------------------------------------------------------------
1 | var React = require('react-native')
2 | var {
3 | PropTypes,
4 | } = React
5 | var isPromise = require('is-promise')
6 | var delay = require('./delay')
7 | var RefreshingIndicator = require('./RefreshingIndicator')
8 | var ControlledRefreshableListView = require('./ControlledRefreshableListView')
9 |
10 | const LISTVIEW_REF = 'listview'
11 |
12 | var RefreshableListView = React.createClass({
13 | propTypes: {
14 | loadData: PropTypes.func.isRequired,
15 | minDisplayTime: PropTypes.number,
16 | minBetweenTime: PropTypes.number,
17 | // props passed to child
18 | refreshPrompt: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
19 | refreshDescription: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
20 | refreshingIndicatorComponent: PropTypes.oneOfType([PropTypes.func, PropTypes.element]),
21 | minPulldownDistance: PropTypes.number,
22 | },
23 | getDefaultProps() {
24 | return {
25 | minDisplayTime: 300,
26 | minBetweenTime: 300,
27 | minPulldownDistance: 40,
28 | refreshingIndicatorComponent: RefreshingIndicator,
29 | }
30 | },
31 | getInitialState() {
32 | return {
33 | isRefreshing: false,
34 | }
35 | },
36 | handlePull() {
37 | this.setState({waitingForRelease: false})
38 | },
39 | handleHold() {
40 | this.setState({waitingForRelease: true})
41 | },
42 | handleRefresh() {
43 | if (this.willRefresh) return
44 |
45 | this.willRefresh = true
46 |
47 | var loadingDataPromise = new Promise((resolve) => {
48 | var loadDataReturnValue = this.props.loadData(resolve)
49 |
50 | if (isPromise(loadDataReturnValue)) {
51 | loadingDataPromise = loadDataReturnValue
52 | }
53 |
54 | Promise.all([
55 | loadingDataPromise,
56 | new Promise((resolve) => this.setState({isRefreshing: true}, resolve)),
57 | delay(this.props.minDisplayTime),
58 | ])
59 | .then(() => delay(this.props.minBetweenTime))
60 | .then(() => {
61 | this.willRefresh = false
62 | this.setState({isRefreshing: false})
63 | })
64 | })
65 |
66 | this.setState({waitingForRelease: false})
67 | },
68 | getScrollResponder() {
69 | return this.refs[LISTVIEW_REF].getScrollResponder()
70 | },
71 | setNativeProps(props) {
72 | this.refs[LISTVIEW_REF].setNativeProps(props)
73 | },
74 | render() {
75 | return (
76 |
85 | )
86 | },
87 | })
88 |
89 | module.exports = RefreshableListView
90 |
--------------------------------------------------------------------------------
/lib/RefreshingIndicator.js:
--------------------------------------------------------------------------------
1 | var React = require('react-native')
2 | var {
3 | View,
4 | Text,
5 | ActivityIndicatorIOS,
6 | PropTypes,
7 | StyleSheet,
8 | isValidElement,
9 | createElement,
10 | } = React
11 |
12 | var RefreshingIndicator = React.createClass({
13 | propTypes: {
14 | activityIndicatorComponent: PropTypes.oneOfType([PropTypes.func, PropTypes.element]),
15 | stylesheet: PropTypes.object,
16 | pullingIndicator: PropTypes.oneOfType([PropTypes.string, PropTypes.func, PropTypes.element]),
17 | holdingIndicator: PropTypes.oneOfType([PropTypes.string, PropTypes.func, PropTypes.element]),
18 | refreshingIndicator: PropTypes.oneOfType([PropTypes.string, PropTypes.func, PropTypes.element]),
19 | pullingPrompt: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
20 | holdingPrompt: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
21 | refreshingPrompt: PropTypes.oneOfType([PropTypes.string, PropTypes.element]),
22 | isTouching: PropTypes.bool,
23 | isRefreshing: PropTypes.bool,
24 | isWaitingForRelease: PropTypes.bool,
25 | },
26 | getDefaultProps() {
27 | return {
28 | activityIndicatorComponent: ActivityIndicatorIOS,
29 | isTouching: false,
30 | isRefreshing: false,
31 | isWaitingForRelease: false,
32 | }
33 | },
34 | renderPrompt() {
35 | if (this.props.isTouching && this.props.isWaitingForRelease) {
36 | return this.props.holdingPrompt
37 | } else if (this.props.isTouching && !this.props.isWaitingForRelease) {
38 | return this.props.pullingPrompt
39 | } else {
40 | if (this.props.isRefreshing) {
41 | return this.props.refreshingPrompt
42 | } else {
43 | return null
44 | }
45 | }
46 | },
47 | renderDescription(styles) {
48 | return (
49 |
50 | {this.renderPrompt()}
51 |
52 | )
53 | },
54 | renderActivityIndicator(styles) {
55 | var activityIndicator
56 | if (this.props.isTouching && this.props.isWaitingForRelease) {
57 | activityIndicator = this.props.holdingIndicator
58 | } else if (this.props.isTouching && !this.props.isWaitingForRelease) {
59 | activityIndicator = this.props.pullingIndicator
60 | } else if (this.props.isRefreshing) {
61 | activityIndicator = this.props.refreshingIndicator || this.props.activityIndicatorComponent
62 | }
63 |
64 | if (activityIndicator) {
65 | if (isValidElement(activityIndicator)) return activityIndicator
66 | // is a component class, not an element
67 | return createElement(activityIndicator, {style: styles.activityIndicator})
68 | }
69 |
70 | return null
71 | },
72 | render() {
73 | var styles = Object.assign({}, stylesheet, this.props.stylesheet)
74 |
75 | return (
76 |
77 |
78 |
79 | {this.renderDescription(styles)}
80 | {this.renderActivityIndicator(styles)}
81 |
82 |
83 |
84 | )
85 | },
86 | })
87 |
88 | var stylesheet = StyleSheet.create({
89 | container: {
90 | flex: 1,
91 | justifyContent: 'center',
92 | alignItems: 'flex-start',
93 | flexDirection: 'row',
94 | },
95 | stack: {
96 | flex: 1,
97 | justifyContent: 'center',
98 | alignItems: 'center',
99 | flexDirection: 'column',
100 | },
101 | wrapper: {
102 | height: 60,
103 | },
104 | content: {
105 | marginTop: 10,
106 | height: 40,
107 | },
108 | })
109 |
110 | module.exports = RefreshingIndicator
111 |
--------------------------------------------------------------------------------
/lib/__mocks__/ListView.js:
--------------------------------------------------------------------------------
1 | module.exports = require.requireActual('./react-native').ListView;
2 |
--------------------------------------------------------------------------------
/lib/__mocks__/createElementFrom.js:
--------------------------------------------------------------------------------
1 | let createElementFrom = jest.genMockFn().mockImpl(require.requireActual('../createElementFrom'))
2 |
3 | module.exports = createElementFrom
4 |
--------------------------------------------------------------------------------
/lib/__mocks__/react-native.js:
--------------------------------------------------------------------------------
1 | var React = require.requireActual('react/dist/react-with-addons')
2 |
3 | function makeMockComponentClass(displayName) {
4 | return React.createClass({
5 | displayName,
6 | render() {
7 | return this.props.children
8 | },
9 | })
10 | }
11 |
12 | var ReactNative = {
13 | ...React,
14 | ActivityIndicatorIOS: makeMockComponentClass('ActivityIndicatorIOS'),
15 | ListView: makeMockComponentClass('ListView'),
16 | View: makeMockComponentClass('View'),
17 | Text: makeMockComponentClass('Text'),
18 | StyleSheet: {
19 | create: (ss) => ss,
20 | },
21 | PropTypes: React.PropTypes,
22 | Platform: {OS: 'ios'},
23 | }
24 |
25 | module.exports = ReactNative
26 |
--------------------------------------------------------------------------------
/lib/__tests__/ControlledRefreshableListView-test.js:
--------------------------------------------------------------------------------
1 | jest.dontMock('../ControlledRefreshableListView')
2 |
3 | describe('ControlledRefreshableListView', function() {
4 | let React
5 | let ControlledRefreshableListView
6 | let ListView
7 | let RefreshingIndicator
8 | let onRefresh
9 | let description
10 | let makeScrollEvent
11 |
12 | beforeEach(function() {
13 | React = require.requireActual('react/dist/react-with-addons')
14 |
15 | ControlledRefreshableListView = require('../ControlledRefreshableListView')
16 | ListView = require('../ListView')
17 | RefreshingIndicator = require('../RefreshingIndicator')
18 |
19 | onRefresh = jest.genMockFn()
20 | description = 'REFRESHING'
21 |
22 | makeScrollEvent = () => ({
23 | nativeEvent: {
24 | contentInset: {top: 64, bottom: 0, left: 0, right: 0},
25 | contentOffset: {x: 0, y: -120},
26 | },
27 | })
28 | })
29 |
30 | it('triggers onRefresh handler when pulldown occurs', function() {
31 | let renderer = this.shallowRender(
32 |
36 | )
37 | let component = renderer.getComponent()
38 | let expectedChildProps = {onScroll: component.handleScroll}
39 |
40 | expect(onRefresh).not.toBeCalled()
41 |
42 | let firstRender = renderer.getRenderOutput()
43 | expect(firstRender.type).toBe(ListView)
44 | expect(firstRender.props).toContainValues(expectedChildProps)
45 |
46 | // simulate scroll
47 | component.handleScroll(makeScrollEvent())
48 | expect(onRefresh).toBeCalled()
49 | })
50 |
51 | iit('renders indicator in header when refreshing', function() {
52 | let renderer = this.shallowRender(
53 |
58 | )
59 | let component = renderer.getComponent()
60 |
61 | let expectedChildProps = {
62 | onScroll: component.handleScroll,
63 | renderHeader: component.renderHeader,
64 | }
65 |
66 | // TODO
67 |
68 | // let firstRender = renderer.getRenderOutput()
69 | // expect(firstRender.type).toBe(ListView)
70 | // expect(firstRender.props).toContainValues(expectedChildProps)
71 |
72 | // let headerElement = component.renderHeader()
73 | // expect(headerElement).toBeReactElement()
74 | // expect(headerElement.type).toBe(RefreshingIndicator)
75 | // expect(headerElement.props).toContainValues({description})
76 | })
77 |
78 | it('does not render indicator when not refreshing', function() {
79 | let renderer = this.shallowRender(
80 |
84 | )
85 |
86 | let expectedChildProps = {
87 | onScroll: component.handleScroll,
88 | renderHeader: component.renderHeader,
89 | }
90 |
91 | let firstRender = renderer.getRenderOutput()
92 | expect(firstRender.type).toBe(ListView)
93 | expect(firstRender.props).toContainValues(expectedChildProps)
94 |
95 | let headerElement = component.renderHeader()
96 | expect(headerElement).toBeNull()
97 | })
98 | })
99 |
--------------------------------------------------------------------------------
/lib/__tests__/RefreshableListView-test.js:
--------------------------------------------------------------------------------
1 | jest.dontMock('../RefreshableListView')
2 |
3 | describe('RefreshableListView', function() {
4 | let React
5 | let merge
6 | let ControlledRefreshableListView
7 | let RefreshableListView
8 | let loadDataResolve
9 | let loadData
10 |
11 | beforeEach(function() {
12 | React = require.requireActual('react/dist/react-with-addons')
13 | merge = require.requireActual('xtend')
14 |
15 | ControlledRefreshableListView = require('../ControlledRefreshableListView')
16 | RefreshableListView = require('../RefreshableListView')
17 |
18 |
19 | loadData = jest.genMockFunction().mockImplementation(() => {
20 | return new Promise((resolve) => {
21 | loadDataResolve = resolve
22 | })
23 | })
24 | })
25 |
26 | it('renders refreshing sequence when child view is refreshed', function() {
27 | let renderer = this.shallowRender(
28 |
29 | )
30 | let component = renderer.getComponent()
31 | let expectedChildProps = {onRefresh: component.handleRefresh}
32 |
33 | let firstRender = renderer.getRenderOutput()
34 | expect(firstRender.type).toBe(ControlledRefreshableListView)
35 | expect(firstRender.props).toContainValues(merge(expectedChildProps, {isRefreshing: false}))
36 |
37 | // simulate refresh
38 | component.handleRefresh()
39 | jest.runAllTimers()
40 |
41 | let secondRender = renderer.getRenderOutput()
42 | expect(secondRender.type).toBe(ControlledRefreshableListView)
43 | expect(secondRender.props).toContainValues(merge(expectedChildProps, {isRefreshing: true}))
44 |
45 | // simulate data reloading completion
46 | loadDataResolve()
47 | jest.runAllTimers()
48 |
49 | // TODO
50 |
51 | // let thirdRender = renderer.getRenderOutput()
52 | // expect(thirdRender.type).toBe(ControlledRefreshableListView)
53 | // expect(thirdRender.props).toContainValues(merge(expectedChildProps, {isRefreshing: false}))
54 | })
55 | })
56 |
--------------------------------------------------------------------------------
/lib/__tests__/RefreshingIndicator-test.js:
--------------------------------------------------------------------------------
1 | jest.dontMock('../RefreshingIndicator')
2 |
3 | describe('RefreshingIndicator', function() {
4 | let React
5 | let ReactNative
6 | let ShallowTestUtils
7 | let RefreshingIndicator
8 | let description
9 |
10 | beforeEach(function() {
11 | React = require.requireActual('react/dist/react-with-addons')
12 | ReactNative = require('react-native')
13 | ShallowTestUtils = require('react-shallow-testutils')
14 |
15 | RefreshingIndicator = require('../RefreshingIndicator')
16 |
17 | description = 'REFRESHING'
18 | })
19 |
20 | it('renders an activity indicator', function() {
21 | let renderer = this.shallowRender(
22 |
26 | )
27 |
28 | let rootEl = renderer.getRenderOutput()
29 | expect(rootEl).toBeReactElementOfType(ReactNative.View)
30 |
31 | let textEl = ShallowTestUtils.findWithType(rootEl, ReactNative.Text)
32 | expect(textEl.props.children).toBe(description)
33 |
34 | let indicatorEl = ShallowTestUtils.findWithType(rootEl, ReactNative.ActivityIndicatorIOS)
35 | expect(indicatorEl).not.toBeUndefined()
36 | })
37 | })
38 |
--------------------------------------------------------------------------------
/lib/__tests__/delay-test.js:
--------------------------------------------------------------------------------
1 | jest.dontMock('../delay')
2 |
3 | describe('delay', function() {
4 | pit('resolves after the specified time', function() {
5 | var delay = require('../delay')
6 |
7 | const DELAY_TIME = 1000
8 |
9 | var resolved = false
10 | var rejected = false
11 |
12 | var delayPromise = delay(DELAY_TIME)
13 | .then(() => resolved = true)
14 | .catch(() => rejected = true)
15 | .then(() => {
16 | expect(setTimeout.mock.calls.length).toBe(1)
17 | expect(setTimeout.mock.calls[0][1]).toBe(DELAY_TIME)
18 | expect(resolved).toBe(true)
19 | expect(rejected).toBe(false)
20 | })
21 |
22 | expect(resolved).toBe(false)
23 | expect(rejected).toBe(false)
24 |
25 | jest.runAllTimers()
26 |
27 | return delayPromise
28 | })
29 | })
30 |
--------------------------------------------------------------------------------
/lib/createElementFrom.js:
--------------------------------------------------------------------------------
1 | var React = require('react-native')
2 | var {
3 | cloneElement,
4 | createElement,
5 | isValidElement,
6 | } = React
7 |
8 | function createElementFrom(elementOrClass, props) {
9 | if (isValidElement(elementOrClass)) {
10 | return cloneElement(elementOrClass, props)
11 | } else { // is a component class, not an element
12 | return createElement(elementOrClass, props)
13 | }
14 | }
15 |
16 | module.exports = createElementFrom
17 |
--------------------------------------------------------------------------------
/lib/delay.js:
--------------------------------------------------------------------------------
1 | function delay(time) {
2 | return new Promise((resolve) => setTimeout(resolve, time))
3 | }
4 |
5 | module.exports = delay
6 |
--------------------------------------------------------------------------------
/lib/scrollToCompat.js:
--------------------------------------------------------------------------------
1 | let cachedResult = null
2 |
3 | function isLegacyReactNative() {
4 | if (cachedResult == null) {
5 | try {
6 | const PACKAGE = require('../../react-native/package.json')
7 | const version = PACKAGE.version
8 | cachedResult = (
9 | version[0] === '0' &&
10 | parseInt(version.split('.')[1], 10) < 20
11 | )
12 | } catch (err) {
13 | cachedResult = true
14 | }
15 | }
16 | return cachedResult
17 | }
18 |
19 | // handle API change in react native 0.20
20 | function scrollToCompat(scrollResponder, y, x, animate = false) {
21 | if (isLegacyReactNative()) {
22 | scrollResponder.scrollTo(y, x, animate)
23 | } else {
24 | scrollResponder.scrollTo({x, y, animate})
25 | }
26 | }
27 |
28 | module.exports = scrollToCompat
29 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "react-native-refreshable-listview",
3 | "version": "2.0.0-beta4",
4 | "description": "A pull-to-refresh ListView which shows a loading spinner while your data reloads",
5 | "main": "index.js",
6 | "scripts": {
7 | "lint": "eslint .",
8 | "watch": "watchy -w lib -- bash -c 'cp -a lib/ test/RefreshableListViewTest/node_modules/react-native-refreshable-listview/lib/'",
9 | "jest": "BABEL_JEST_STAGE=0 jest $JEST_ARGS",
10 | "test": "npm run lint && npm run jest"
11 | },
12 | "author": "James Friend (http://jsdf.co/)",
13 | "license": "ISC",
14 | "keywords": [
15 | "react",
16 | "react-native",
17 | "react-component",
18 | "react-native-component",
19 | "mobile",
20 | "ui"
21 | ],
22 | "homepage": "https://github.com/jsdf/react-native-refreshable-listview",
23 | "bugs": "https://github.com/jsdf/react-native-refreshable-listview/issues",
24 | "repository": {
25 | "type": "git",
26 | "url": "git://github.com/jsdf/react-native-refreshable-listview.git"
27 | },
28 | "jest": {
29 | "scriptPreprocessor": "/test/support/preprocessor.js",
30 | "setupEnvScriptFile": "/node_modules/babel-core/browser-polyfill.js",
31 | "setupTestFrameworkScriptFile": "/test/support/setupTestFramework.js",
32 | "unmockedModulePathPatterns": [
33 | "/test/support",
34 | "/node_modules/react",
35 | "/node_modules/babel-core",
36 | "/node_modules/source-map-support",
37 | "/node_modules/inspect-react-element",
38 | "/node_modules/react-shallow-testutils",
39 | "/node_modules/react-addons-test-utils",
40 | "/node_modules/absolute-path",
41 | "/node_modules/xtend",
42 | "/node_modules/jasmine-object-matchers-jest",
43 | "/node_modules/is-promise"
44 | ],
45 | "testFileExtensions": [
46 | "es6",
47 | "js"
48 | ],
49 | "moduleFileExtensions": [
50 | "js",
51 | "json",
52 | "es6"
53 | ]
54 | },
55 | "devDependencies": {
56 | "absolute-path": "0.0.0",
57 | "babel-core": "^5.8.34",
58 | "eslint": "^0.19.0",
59 | "eslint-plugin-react": "^2.1.1",
60 | "inspect-react-element": "^1.1.1",
61 | "jasmine-object-matchers-jest": "^0.1.0",
62 | "jest-cli": "^0.9.0-fb2",
63 | "jscs": "^2.7.0",
64 | "react": "^0.14.7",
65 | "react-addons-test-utils": "^0.14.7",
66 | "react-shallow-testutils": "^1.0.0",
67 | "source-map-support": "^0.2.10",
68 | "watchy": "^0.6.4",
69 | "xtend": "^4.0.0"
70 | },
71 | "dependencies": {
72 | "is-promise": "^2.0.0"
73 | }
74 | }
75 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/.flowconfig:
--------------------------------------------------------------------------------
1 | [ignore]
2 |
3 | # We fork some components by platform.
4 | .*/*.web.js
5 | .*/*.android.js
6 |
7 | # Some modules have their own node_modules with overlap
8 | .*/node_modules/node-haste/.*
9 |
10 | # Ugh
11 | .*/node_modules/babel.*
12 | .*/node_modules/babylon.*
13 | .*/node_modules/invariant.*
14 |
15 | # Ignore react and fbjs where there are overlaps, but don't ignore
16 | # anything that react-native relies on
17 | .*/node_modules/fbjs-haste/.*/__tests__/.*
18 | .*/node_modules/fbjs-haste/__forks__/Map.js
19 | .*/node_modules/fbjs-haste/__forks__/Promise.js
20 | .*/node_modules/fbjs-haste/__forks__/fetch.js
21 | .*/node_modules/fbjs-haste/core/ExecutionEnvironment.js
22 | .*/node_modules/fbjs-haste/core/isEmpty.js
23 | .*/node_modules/fbjs-haste/crypto/crc32.js
24 | .*/node_modules/fbjs-haste/stubs/ErrorUtils.js
25 | .*/node_modules/react-haste/React.js
26 | .*/node_modules/react-haste/renderers/dom/ReactDOM.js
27 | .*/node_modules/react-haste/renderers/shared/event/eventPlugins/ResponderEventPlugin.js
28 |
29 | # Ignore commoner tests
30 | .*/node_modules/commoner/test/.*
31 |
32 | # See https://github.com/facebook/flow/issues/442
33 | .*/react-tools/node_modules/commoner/lib/reader.js
34 |
35 | # Ignore jest
36 | .*/node_modules/jest-cli/.*
37 |
38 | # Ignore Website
39 | .*/website/.*
40 |
41 | [include]
42 |
43 | [libs]
44 | node_modules/react-native/Libraries/react-native/react-native-interface.js
45 |
46 | [options]
47 | module.system=haste
48 |
49 | munge_underscores=true
50 |
51 | module.name_mapper='^image![a-zA-Z0-9$_-]+$' -> 'GlobalImageStub'
52 | module.name_mapper='^[./a-zA-Z0-9$_-]+\.png$' -> 'RelativeImageStub'
53 |
54 | suppress_type=$FlowIssue
55 | suppress_type=$FlowFixMe
56 | suppress_type=$FixMe
57 |
58 | suppress_comment=\\(.\\|\n\\)*\\$FlowFixMe\\($\\|[^(]\\|(\\(>=0\\.\\(1[0-9]\\|[0-9]\\).[0-9]\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\)
59 | suppress_comment=\\(.\\|\n\\)*\\$FlowIssue\\((\\(>=0\\.\\(1[0-9]\\|[0-9]\\).[0-9]\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\)? #[0-9]+
60 | suppress_comment=\\(.\\|\n\\)*\\$FlowFixedInNextDeploy
61 |
62 | [version]
63 | 0.19.0
64 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/.gitignore:
--------------------------------------------------------------------------------
1 | # OSX
2 | #
3 | .DS_Store
4 |
5 | # Xcode
6 | #
7 | build/
8 | *.pbxuser
9 | !default.pbxuser
10 | *.mode1v3
11 | !default.mode1v3
12 | *.mode2v3
13 | !default.mode2v3
14 | *.perspectivev3
15 | !default.perspectivev3
16 | xcuserdata
17 | *.xccheckout
18 | *.moved-aside
19 | DerivedData
20 | *.hmap
21 | *.ipa
22 | *.xcuserstate
23 | project.xcworkspace
24 |
25 | # Android/IJ
26 | #
27 | .idea
28 | .gradle
29 | local.properties
30 |
31 | # node.js
32 | #
33 | node_modules/
34 | npm-debug.log
35 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/.watchmanconfig:
--------------------------------------------------------------------------------
1 | {}
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/build.gradle:
--------------------------------------------------------------------------------
1 | apply plugin: "com.android.application"
2 |
3 | /**
4 | * The react.gradle file registers two tasks: bundleDebugJsAndAssets and bundleReleaseJsAndAssets.
5 | * These basically call `react-native bundle` with the correct arguments during the Android build
6 | * cycle. By default, bundleDebugJsAndAssets is skipped, as in debug/dev mode we prefer to load the
7 | * bundle directly from the development server. Below you can see all the possible configurations
8 | * and their defaults. If you decide to add a configuration block, make sure to add it before the
9 | * `apply from: "react.gradle"` line.
10 | *
11 | * project.ext.react = [
12 | * // the name of the generated asset file containing your JS bundle
13 | * bundleAssetName: "index.android.bundle",
14 | *
15 | * // the entry file for bundle generation
16 | * entryFile: "index.android.js",
17 | *
18 | * // whether to bundle JS and assets in debug mode
19 | * bundleInDebug: false,
20 | *
21 | * // whether to bundle JS and assets in release mode
22 | * bundleInRelease: true,
23 | *
24 | * // the root of your project, i.e. where "package.json" lives
25 | * root: "../../",
26 | *
27 | * // where to put the JS bundle asset in debug mode
28 | * jsBundleDirDebug: "$buildDir/intermediates/assets/debug",
29 | *
30 | * // where to put the JS bundle asset in release mode
31 | * jsBundleDirRelease: "$buildDir/intermediates/assets/release",
32 | *
33 | * // where to put drawable resources / React Native assets, e.g. the ones you use via
34 | * // require('./image.png')), in debug mode
35 | * resourcesDirDebug: "$buildDir/intermediates/res/merged/debug",
36 | *
37 | * // where to put drawable resources / React Native assets, e.g. the ones you use via
38 | * // require('./image.png')), in release mode
39 | * resourcesDirRelease: "$buildDir/intermediates/res/merged/release",
40 | *
41 | * // by default the gradle tasks are skipped if none of the JS files or assets change; this means
42 | * // that we don't look at files in android/ or ios/ to determine whether the tasks are up to
43 | * // date; if you have any other folders that you want to ignore for performance reasons (gradle
44 | * // indexes the entire tree), add them here. Alternatively, if you have JS files in android/
45 | * // for example, you might want to remove it from here.
46 | * inputExcludes: ["android/**", "ios/**"]
47 | * ]
48 | */
49 |
50 | apply from: "react.gradle"
51 |
52 | android {
53 | compileSdkVersion 23
54 | buildToolsVersion "23.0.1"
55 |
56 | defaultConfig {
57 | applicationId "com.refreshablelistviewtest"
58 | minSdkVersion 16
59 | targetSdkVersion 22
60 | versionCode 1
61 | versionName "1.0"
62 | ndk {
63 | abiFilters "armeabi-v7a", "x86"
64 | }
65 | }
66 | buildTypes {
67 | release {
68 | minifyEnabled false // Set this to true to enable Proguard
69 | proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro"
70 | }
71 | }
72 | }
73 |
74 | dependencies {
75 | compile fileTree(dir: "libs", include: ["*.jar"])
76 | compile "com.android.support:appcompat-v7:23.0.1"
77 | compile "com.facebook.react:react-native:0.17.+"
78 | }
79 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # By default, the flags in this file are appended to flags specified
3 | # in /usr/local/Cellar/android-sdk/24.3.3/tools/proguard/proguard-android.txt
4 | # You can edit the include path and order by changing the proguardFiles
5 | # directive in build.gradle.
6 | #
7 | # For more details, see
8 | # http://developer.android.com/guide/developing/tools/proguard.html
9 |
10 | # Add any project specific keep options here:
11 |
12 | # If your project uses WebView with JS, uncomment the following
13 | # and specify the fully qualified class name to the JavaScript interface
14 | # class:
15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
16 | # public *;
17 | #}
18 |
19 | # Disabling obfuscation is useful if you collect stack traces from production crashes
20 | # (unless you are using a system that supports de-obfuscate the stack traces).
21 | -dontobfuscate
22 |
23 | # React Native
24 |
25 | # Keep our interfaces so they can be used by other ProGuard rules.
26 | # See http://sourceforge.net/p/proguard/bugs/466/
27 | -keep,allowobfuscation @interface com.facebook.proguard.annotations.DoNotStrip
28 | -keep,allowobfuscation @interface com.facebook.proguard.annotations.KeepGettersAndSetters
29 |
30 | # Do not strip any method/class that is annotated with @DoNotStrip
31 | -keep @com.facebook.proguard.annotations.DoNotStrip class *
32 | -keepclassmembers class * {
33 | @com.facebook.proguard.annotations.DoNotStrip *;
34 | }
35 |
36 | -keepclassmembers @com.facebook.proguard.annotations.KeepGettersAndSetters class * {
37 | void set*(***);
38 | *** get*();
39 | }
40 |
41 | -keep class * extends com.facebook.react.bridge.JavaScriptModule { *; }
42 | -keep class * extends com.facebook.react.bridge.NativeModule { *; }
43 | -keepclassmembers class * { @com.facebook.react.uimanager.UIProp ; }
44 | -keepclassmembers class * { @com.facebook.react.uimanager.ReactProp ; }
45 | -keepclassmembers class * { @com.facebook.react.uimanager.ReactPropGroup ; }
46 |
47 | # okhttp
48 |
49 | -keepattributes Signature
50 | -keepattributes *Annotation*
51 | -keep class com.squareup.okhttp.** { *; }
52 | -keep interface com.squareup.okhttp.** { *; }
53 | -dontwarn com.squareup.okhttp.**
54 |
55 | # okio
56 |
57 | -keep class sun.misc.Unsafe { *; }
58 | -dontwarn java.nio.file.*
59 | -dontwarn org.codehaus.mojo.animal_sniffer.IgnoreJRERequirement
60 | -dontwarn okio.**
61 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/react.gradle:
--------------------------------------------------------------------------------
1 | import org.apache.tools.ant.taskdefs.condition.Os
2 |
3 | def config = project.hasProperty("react") ? project.react : [];
4 |
5 | def bundleAssetName = config.bundleAssetName ?: "index.android.bundle"
6 | def entryFile = config.entryFile ?: "index.android.js"
7 |
8 | // because elvis operator
9 | def elvisFile(thing) {
10 | return thing ? file(thing) : null;
11 | }
12 |
13 | def reactRoot = elvisFile(config.root) ?: file("../../")
14 | def jsBundleDirDebug = elvisFile(config.jsBundleDirDebug) ?:
15 | file("$buildDir/intermediates/assets/debug")
16 | def jsBundleDirRelease = elvisFile(config.jsBundleDirRelease) ?:
17 | file("$buildDir/intermediates/assets/release")
18 | def resourcesDirDebug = elvisFile(config.resourcesDirDebug) ?:
19 | file("$buildDir/intermediates/res/merged/debug")
20 | def resourcesDirRelease = elvisFile(config.resourcesDirRelease) ?:
21 | file("$buildDir/intermediates/res/merged/release")
22 | def inputExcludes = config.inputExcludes ?: ["android/**", "ios/**"]
23 |
24 | def jsBundleFileDebug = file("$jsBundleDirDebug/$bundleAssetName")
25 | def jsBundleFileRelease = file("$jsBundleDirRelease/$bundleAssetName")
26 |
27 | task bundleDebugJsAndAssets(type: Exec) {
28 | // create dirs if they are not there (e.g. the "clean" task just ran)
29 | doFirst {
30 | jsBundleDirDebug.mkdirs()
31 | resourcesDirDebug.mkdirs()
32 | }
33 |
34 | // set up inputs and outputs so gradle can cache the result
35 | inputs.files fileTree(dir: reactRoot, excludes: inputExcludes)
36 | outputs.dir jsBundleDirDebug
37 | outputs.dir resourcesDirDebug
38 |
39 | // set up the call to the react-native cli
40 | workingDir reactRoot
41 | if (Os.isFamily(Os.FAMILY_WINDOWS)) {
42 | commandLine "cmd", "/c", "react-native", "bundle", "--platform", "android", "--dev", "true", "--entry-file",
43 | entryFile, "--bundle-output", jsBundleFileDebug, "--assets-dest", resourcesDirDebug
44 | } else {
45 | commandLine "react-native", "bundle", "--platform", "android", "--dev", "true", "--entry-file",
46 | entryFile, "--bundle-output", jsBundleFileDebug, "--assets-dest", resourcesDirDebug
47 | }
48 |
49 | enabled config.bundleInDebug ?: false
50 | }
51 |
52 | task bundleReleaseJsAndAssets(type: Exec) {
53 | // create dirs if they are not there (e.g. the "clean" task just ran)
54 | doFirst {
55 | jsBundleDirRelease.mkdirs()
56 | resourcesDirRelease.mkdirs()
57 | }
58 |
59 | // set up inputs and outputs so gradle can cache the result
60 | inputs.files fileTree(dir: reactRoot, excludes: inputExcludes)
61 | outputs.dir jsBundleDirRelease
62 | outputs.dir resourcesDirRelease
63 |
64 | // set up the call to the react-native cli
65 | workingDir reactRoot
66 | if (Os.isFamily(Os.FAMILY_WINDOWS)) {
67 | commandLine "cmd","/c", "react-native", "bundle", "--platform", "android", "--dev", "false", "--entry-file",
68 | entryFile, "--bundle-output", jsBundleFileRelease, "--assets-dest", resourcesDirRelease
69 | } else {
70 | commandLine "react-native", "bundle", "--platform", "android", "--dev", "false", "--entry-file",
71 | entryFile, "--bundle-output", jsBundleFileRelease, "--assets-dest", resourcesDirRelease
72 | }
73 |
74 | enabled config.bundleInRelease ?: true
75 | }
76 |
77 | gradle.projectsEvaluated {
78 | // hook bundleDebugJsAndAssets into the android build process
79 | bundleDebugJsAndAssets.dependsOn mergeDebugResources
80 | bundleDebugJsAndAssets.dependsOn mergeDebugAssets
81 | processDebugResources.dependsOn bundleDebugJsAndAssets
82 |
83 | // hook bundleReleaseJsAndAssets into the android build process
84 | bundleReleaseJsAndAssets.dependsOn mergeReleaseResources
85 | bundleReleaseJsAndAssets.dependsOn mergeReleaseAssets
86 | processReleaseResources.dependsOn bundleReleaseJsAndAssets
87 | }
88 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
3 |
4 |
5 |
6 |
11 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/java/com/refreshablelistviewtest/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.refreshablelistviewtest;
2 |
3 | import android.app.Activity;
4 | import android.os.Bundle;
5 | import android.view.KeyEvent;
6 |
7 | import com.facebook.react.LifecycleState;
8 | import com.facebook.react.ReactInstanceManager;
9 | import com.facebook.react.ReactRootView;
10 | import com.facebook.react.modules.core.DefaultHardwareBackBtnHandler;
11 | import com.facebook.react.shell.MainReactPackage;
12 | import com.facebook.soloader.SoLoader;
13 |
14 | public class MainActivity extends Activity implements DefaultHardwareBackBtnHandler {
15 |
16 | private ReactInstanceManager mReactInstanceManager;
17 | private ReactRootView mReactRootView;
18 |
19 | @Override
20 | protected void onCreate(Bundle savedInstanceState) {
21 | super.onCreate(savedInstanceState);
22 | mReactRootView = new ReactRootView(this);
23 |
24 | mReactInstanceManager = ReactInstanceManager.builder()
25 | .setApplication(getApplication())
26 | .setBundleAssetName("index.android.bundle")
27 | .setJSMainModuleName("index.android")
28 | .addPackage(new MainReactPackage())
29 | .setUseDeveloperSupport(BuildConfig.DEBUG)
30 | .setInitialLifecycleState(LifecycleState.RESUMED)
31 | .build();
32 |
33 | mReactRootView.startReactApplication(mReactInstanceManager, "RefreshableListViewTest", null);
34 |
35 | setContentView(mReactRootView);
36 | }
37 |
38 | @Override
39 | public boolean onKeyUp(int keyCode, KeyEvent event) {
40 | if (keyCode == KeyEvent.KEYCODE_MENU && mReactInstanceManager != null) {
41 | mReactInstanceManager.showDevOptionsDialog();
42 | return true;
43 | }
44 | return super.onKeyUp(keyCode, event);
45 | }
46 |
47 | @Override
48 | public void onBackPressed() {
49 | if (mReactInstanceManager != null) {
50 | mReactInstanceManager.onBackPressed();
51 | } else {
52 | super.onBackPressed();
53 | }
54 | }
55 |
56 | @Override
57 | public void invokeDefaultOnBackPressed() {
58 | super.onBackPressed();
59 | }
60 |
61 | @Override
62 | protected void onPause() {
63 | super.onPause();
64 |
65 | if (mReactInstanceManager != null) {
66 | mReactInstanceManager.onPause();
67 | }
68 | }
69 |
70 | @Override
71 | protected void onResume() {
72 | super.onResume();
73 |
74 | if (mReactInstanceManager != null) {
75 | mReactInstanceManager.onResume(this, this);
76 | }
77 | }
78 | }
79 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/res/mipmap-hdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsdf/react-native-refreshable-listview/701fb1bff6d85d428dd9b46f38a62eccf4d55291/test/RefreshableListViewTest/android/app/src/main/res/mipmap-hdpi/ic_launcher.png
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/res/mipmap-mdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsdf/react-native-refreshable-listview/701fb1bff6d85d428dd9b46f38a62eccf4d55291/test/RefreshableListViewTest/android/app/src/main/res/mipmap-mdpi/ic_launcher.png
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsdf/react-native-refreshable-listview/701fb1bff6d85d428dd9b46f38a62eccf4d55291/test/RefreshableListViewTest/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsdf/react-native-refreshable-listview/701fb1bff6d85d428dd9b46f38a62eccf4d55291/test/RefreshableListViewTest/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | RefreshableListViewTest
3 |
4 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/app/src/main/res/values/styles.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/build.gradle:
--------------------------------------------------------------------------------
1 | // Top-level build file where you can add configuration options common to all sub-projects/modules.
2 |
3 | buildscript {
4 | repositories {
5 | jcenter()
6 | }
7 | dependencies {
8 | classpath 'com.android.tools.build:gradle:1.3.1'
9 |
10 | // NOTE: Do not place your application dependencies here; they belong
11 | // in the individual module build.gradle files
12 | }
13 | }
14 |
15 | allprojects {
16 | repositories {
17 | mavenLocal()
18 | jcenter()
19 | }
20 | }
21 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 |
3 | # IDE (e.g. Android Studio) users:
4 | # Gradle settings configured through the IDE *will override*
5 | # any settings specified in this file.
6 |
7 | # For more details on how to configure your build environment visit
8 | # http://www.gradle.org/docs/current/userguide/build_environment.html
9 |
10 | # Specifies the JVM arguments used for the daemon process.
11 | # The setting is particularly useful for tweaking memory settings.
12 | # Default value: -Xmx10248m -XX:MaxPermSize=256m
13 | # org.gradle.jvmargs=-Xmx2048m -XX:MaxPermSize=512m -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=UTF-8
14 |
15 | # When configured, Gradle will run in incubating parallel mode.
16 | # This option should only be used with decoupled projects. More details, visit
17 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
18 | # org.gradle.parallel=true
19 |
20 | android.useDeprecatedNdk=true
21 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/jsdf/react-native-refreshable-listview/701fb1bff6d85d428dd9b46f38a62eccf4d55291/test/RefreshableListViewTest/android/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | distributionBase=GRADLE_USER_HOME
2 | distributionPath=wrapper/dists
3 | zipStoreBase=GRADLE_USER_HOME
4 | zipStorePath=wrapper/dists
5 | distributionUrl=https\://services.gradle.org/distributions/gradle-2.4-all.zip
6 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | ##############################################################################
4 | ##
5 | ## Gradle start up script for UN*X
6 | ##
7 | ##############################################################################
8 |
9 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
10 | DEFAULT_JVM_OPTS=""
11 |
12 | APP_NAME="Gradle"
13 | APP_BASE_NAME=`basename "$0"`
14 |
15 | # Use the maximum available, or set MAX_FD != -1 to use that value.
16 | MAX_FD="maximum"
17 |
18 | warn ( ) {
19 | echo "$*"
20 | }
21 |
22 | die ( ) {
23 | echo
24 | echo "$*"
25 | echo
26 | exit 1
27 | }
28 |
29 | # OS specific support (must be 'true' or 'false').
30 | cygwin=false
31 | msys=false
32 | darwin=false
33 | case "`uname`" in
34 | CYGWIN* )
35 | cygwin=true
36 | ;;
37 | Darwin* )
38 | darwin=true
39 | ;;
40 | MINGW* )
41 | msys=true
42 | ;;
43 | esac
44 |
45 | # For Cygwin, ensure paths are in UNIX format before anything is touched.
46 | if $cygwin ; then
47 | [ -n "$JAVA_HOME" ] && JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
48 | fi
49 |
50 | # Attempt to set APP_HOME
51 | # Resolve links: $0 may be a link
52 | PRG="$0"
53 | # Need this for relative symlinks.
54 | while [ -h "$PRG" ] ; do
55 | ls=`ls -ld "$PRG"`
56 | link=`expr "$ls" : '.*-> \(.*\)$'`
57 | if expr "$link" : '/.*' > /dev/null; then
58 | PRG="$link"
59 | else
60 | PRG=`dirname "$PRG"`"/$link"
61 | fi
62 | done
63 | SAVED="`pwd`"
64 | cd "`dirname \"$PRG\"`/" >&-
65 | APP_HOME="`pwd -P`"
66 | cd "$SAVED" >&-
67 |
68 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
69 |
70 | # Determine the Java command to use to start the JVM.
71 | if [ -n "$JAVA_HOME" ] ; then
72 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
73 | # IBM's JDK on AIX uses strange locations for the executables
74 | JAVACMD="$JAVA_HOME/jre/sh/java"
75 | else
76 | JAVACMD="$JAVA_HOME/bin/java"
77 | fi
78 | if [ ! -x "$JAVACMD" ] ; then
79 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
80 |
81 | Please set the JAVA_HOME variable in your environment to match the
82 | location of your Java installation."
83 | fi
84 | else
85 | JAVACMD="java"
86 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
87 |
88 | Please set the JAVA_HOME variable in your environment to match the
89 | location of your Java installation."
90 | fi
91 |
92 | # Increase the maximum file descriptors if we can.
93 | if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
94 | MAX_FD_LIMIT=`ulimit -H -n`
95 | if [ $? -eq 0 ] ; then
96 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
97 | MAX_FD="$MAX_FD_LIMIT"
98 | fi
99 | ulimit -n $MAX_FD
100 | if [ $? -ne 0 ] ; then
101 | warn "Could not set maximum file descriptor limit: $MAX_FD"
102 | fi
103 | else
104 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
105 | fi
106 | fi
107 |
108 | # For Darwin, add options to specify how the application appears in the dock
109 | if $darwin; then
110 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
111 | fi
112 |
113 | # For Cygwin, switch paths to Windows format before running java
114 | if $cygwin ; then
115 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
116 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
117 |
118 | # We build the pattern for arguments to be converted via cygpath
119 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
120 | SEP=""
121 | for dir in $ROOTDIRSRAW ; do
122 | ROOTDIRS="$ROOTDIRS$SEP$dir"
123 | SEP="|"
124 | done
125 | OURCYGPATTERN="(^($ROOTDIRS))"
126 | # Add a user-defined pattern to the cygpath arguments
127 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
128 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
129 | fi
130 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
131 | i=0
132 | for arg in "$@" ; do
133 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
134 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
135 |
136 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
137 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
138 | else
139 | eval `echo args$i`="\"$arg\""
140 | fi
141 | i=$((i+1))
142 | done
143 | case $i in
144 | (0) set -- ;;
145 | (1) set -- "$args0" ;;
146 | (2) set -- "$args0" "$args1" ;;
147 | (3) set -- "$args0" "$args1" "$args2" ;;
148 | (4) set -- "$args0" "$args1" "$args2" "$args3" ;;
149 | (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
150 | (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
151 | (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
152 | (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
153 | (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
154 | esac
155 | fi
156 |
157 | # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
158 | function splitJvmOpts() {
159 | JVM_OPTS=("$@")
160 | }
161 | eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
162 | JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
163 |
164 | exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
165 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/gradlew.bat:
--------------------------------------------------------------------------------
1 | @if "%DEBUG%" == "" @echo off
2 | @rem ##########################################################################
3 | @rem
4 | @rem Gradle startup script for Windows
5 | @rem
6 | @rem ##########################################################################
7 |
8 | @rem Set local scope for the variables with windows NT shell
9 | if "%OS%"=="Windows_NT" setlocal
10 |
11 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
12 | set DEFAULT_JVM_OPTS=
13 |
14 | set DIRNAME=%~dp0
15 | if "%DIRNAME%" == "" set DIRNAME=.
16 | set APP_BASE_NAME=%~n0
17 | set APP_HOME=%DIRNAME%
18 |
19 | @rem Find java.exe
20 | if defined JAVA_HOME goto findJavaFromJavaHome
21 |
22 | set JAVA_EXE=java.exe
23 | %JAVA_EXE% -version >NUL 2>&1
24 | if "%ERRORLEVEL%" == "0" goto init
25 |
26 | echo.
27 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
28 | echo.
29 | echo Please set the JAVA_HOME variable in your environment to match the
30 | echo location of your Java installation.
31 |
32 | goto fail
33 |
34 | :findJavaFromJavaHome
35 | set JAVA_HOME=%JAVA_HOME:"=%
36 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
37 |
38 | if exist "%JAVA_EXE%" goto init
39 |
40 | echo.
41 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
42 | echo.
43 | echo Please set the JAVA_HOME variable in your environment to match the
44 | echo location of your Java installation.
45 |
46 | goto fail
47 |
48 | :init
49 | @rem Get command-line arguments, handling Windowz variants
50 |
51 | if not "%OS%" == "Windows_NT" goto win9xME_args
52 | if "%@eval[2+2]" == "4" goto 4NT_args
53 |
54 | :win9xME_args
55 | @rem Slurp the command line arguments.
56 | set CMD_LINE_ARGS=
57 | set _SKIP=2
58 |
59 | :win9xME_args_slurp
60 | if "x%~1" == "x" goto execute
61 |
62 | set CMD_LINE_ARGS=%*
63 | goto execute
64 |
65 | :4NT_args
66 | @rem Get arguments from the 4NT Shell from JP Software
67 | set CMD_LINE_ARGS=%$
68 |
69 | :execute
70 | @rem Setup the command line
71 |
72 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
73 |
74 | @rem Execute Gradle
75 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
76 |
77 | :end
78 | @rem End local scope for the variables with windows NT shell
79 | if "%ERRORLEVEL%"=="0" goto mainEnd
80 |
81 | :fail
82 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
83 | rem the _cmd.exe /c_ return code!
84 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
85 | exit /b 1
86 |
87 | :mainEnd
88 | if "%OS%"=="Windows_NT" endlocal
89 |
90 | :omega
91 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/android/settings.gradle:
--------------------------------------------------------------------------------
1 | rootProject.name = 'RefreshableListViewTest'
2 |
3 | include ':app'
4 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/index.android.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Sample React Native App
3 | * https://github.com/facebook/react-native
4 | */
5 | 'use strict';
6 |
7 | var React = require('react-native');
8 | var {
9 | AppRegistry,
10 | StyleSheet,
11 | Text,
12 | View,
13 | } = React;
14 |
15 | var RefreshableListViewTest = React.createClass({
16 | render: function() {
17 | return (
18 |
19 |
20 | Welcome to React Native!
21 |
22 |
23 | To get started, edit index.android.js
24 |
25 |
26 | Shake or press menu button for dev menu
27 |
28 |
29 | );
30 | }
31 | });
32 |
33 | var styles = StyleSheet.create({
34 | container: {
35 | flex: 1,
36 | justifyContent: 'center',
37 | alignItems: 'center',
38 | backgroundColor: '#F5FCFF',
39 | },
40 | welcome: {
41 | fontSize: 20,
42 | textAlign: 'center',
43 | margin: 10,
44 | },
45 | instructions: {
46 | textAlign: 'center',
47 | color: '#333333',
48 | marginBottom: 5,
49 | },
50 | });
51 |
52 | AppRegistry.registerComponent('RefreshableListViewTest', () => RefreshableListViewTest);
53 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/index.ios.js:
--------------------------------------------------------------------------------
1 | var React = require('react-native');
2 | var {
3 | AppRegistry,
4 | StyleSheet,
5 | Text,
6 | View,
7 | ListView,
8 | StatusBarIOS,
9 | } = React;
10 |
11 | var RefreshableListView = require('react-native-refreshable-listview')
12 | var delay = require('react-native-refreshable-listview/lib/delay')
13 |
14 | // make an array containing a sequence of numbers from 0..n
15 | var makeSequence = (n) => Array.apply(null, {length: n}).map((v, i) => i)
16 |
17 | var ds = new ListView.DataSource({
18 | rowHasChanged: (r1, r2) => r1.text !== r2.text
19 | })
20 |
21 | var RefreshableListViewTest = React.createClass({
22 | getInitialState() {
23 | var rows = makeSequence(100).map((n) => ({text: 'not refreshed '+n}))
24 | return {dataSource: ds.cloneWithRows(rows)}
25 | },
26 | componentDidMount() {
27 | StatusBarIOS.setStyle(0)
28 | },
29 | reloadItems() {
30 | return delay(1000).then(() => {
31 | var rows = makeSequence(100).map((n) => ({text: 'refreshed '+n}))
32 | this.setState({dataSource: ds.cloneWithRows(rows)})
33 | })
34 | },
35 | renderItem(item) {
36 | return (
37 |
38 |
39 | {item.text}
40 |
41 |
42 | )
43 | },
44 | render() {
45 | return (
46 |
47 |
54 |
55 | )
56 | },
57 | });
58 |
59 | AppRegistry.registerComponent('RefreshableListViewTest', () => RefreshableListViewTest);
60 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest.xcodeproj/project.pbxproj:
--------------------------------------------------------------------------------
1 | // !$*UTF8*$!
2 | {
3 | archiveVersion = 1;
4 | classes = {
5 | };
6 | objectVersion = 46;
7 | objects = {
8 |
9 | /* Begin PBXBuildFile section */
10 | 00C302E51ABCBA2D00DB3ED1 /* libRCTActionSheet.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */; };
11 | 00C302E71ABCBA2D00DB3ED1 /* libRCTGeolocation.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */; };
12 | 00C302E81ABCBA2D00DB3ED1 /* libRCTImage.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */; };
13 | 00C302E91ABCBA2D00DB3ED1 /* libRCTNetwork.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */; };
14 | 00C302EA1ABCBA2D00DB3ED1 /* libRCTVibration.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */; };
15 | 00E356F31AD99517003FC87E /* RefreshableListViewTestTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 00E356F21AD99517003FC87E /* RefreshableListViewTestTests.m */; };
16 | 133E29F31AD74F7200F7D852 /* libRCTLinking.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 78C398B91ACF4ADC00677621 /* libRCTLinking.a */; };
17 | 139105C61AF99C1200B5F7CC /* libRCTSettings.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */; };
18 | 139FDEF61B0652A700C62182 /* libRCTWebSocket.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */; };
19 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB01A68108700A75B9A /* AppDelegate.m */; };
20 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB11A68108700A75B9A /* LaunchScreen.xib */; };
21 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; };
22 | 13B07FC11A68108700A75B9A /* main.m in Sources */ = {isa = PBXBuildFile; fileRef = 13B07FB71A68108700A75B9A /* main.m */; };
23 | 146834051AC3E58100842450 /* libReact.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 146834041AC3E56700842450 /* libReact.a */; };
24 | 832341BD1AAA6AB300B99B32 /* libRCTText.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 832341B51AAA6A8300B99B32 /* libRCTText.a */; };
25 | /* End PBXBuildFile section */
26 |
27 | /* Begin PBXContainerItemProxy section */
28 | 00C302AB1ABCB8CE00DB3ED1 /* PBXContainerItemProxy */ = {
29 | isa = PBXContainerItemProxy;
30 | containerPortal = 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */;
31 | proxyType = 2;
32 | remoteGlobalIDString = 134814201AA4EA6300B7C361;
33 | remoteInfo = RCTActionSheet;
34 | };
35 | 00C302B91ABCB90400DB3ED1 /* PBXContainerItemProxy */ = {
36 | isa = PBXContainerItemProxy;
37 | containerPortal = 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */;
38 | proxyType = 2;
39 | remoteGlobalIDString = 134814201AA4EA6300B7C361;
40 | remoteInfo = RCTGeolocation;
41 | };
42 | 00C302BF1ABCB91800DB3ED1 /* PBXContainerItemProxy */ = {
43 | isa = PBXContainerItemProxy;
44 | containerPortal = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */;
45 | proxyType = 2;
46 | remoteGlobalIDString = 58B5115D1A9E6B3D00147676;
47 | remoteInfo = RCTImage;
48 | };
49 | 00C302DB1ABCB9D200DB3ED1 /* PBXContainerItemProxy */ = {
50 | isa = PBXContainerItemProxy;
51 | containerPortal = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */;
52 | proxyType = 2;
53 | remoteGlobalIDString = 58B511DB1A9E6C8500147676;
54 | remoteInfo = RCTNetwork;
55 | };
56 | 00C302E31ABCB9EE00DB3ED1 /* PBXContainerItemProxy */ = {
57 | isa = PBXContainerItemProxy;
58 | containerPortal = 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */;
59 | proxyType = 2;
60 | remoteGlobalIDString = 832C81801AAF6DEF007FA2F7;
61 | remoteInfo = RCTVibration;
62 | };
63 | 00E356F41AD99517003FC87E /* PBXContainerItemProxy */ = {
64 | isa = PBXContainerItemProxy;
65 | containerPortal = 83CBB9F71A601CBA00E9B192 /* Project object */;
66 | proxyType = 1;
67 | remoteGlobalIDString = 13B07F861A680F5B00A75B9A;
68 | remoteInfo = RefreshableListViewTest;
69 | };
70 | 139105C01AF99BAD00B5F7CC /* PBXContainerItemProxy */ = {
71 | isa = PBXContainerItemProxy;
72 | containerPortal = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */;
73 | proxyType = 2;
74 | remoteGlobalIDString = 134814201AA4EA6300B7C361;
75 | remoteInfo = RCTSettings;
76 | };
77 | 139FDEF31B06529B00C62182 /* PBXContainerItemProxy */ = {
78 | isa = PBXContainerItemProxy;
79 | containerPortal = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */;
80 | proxyType = 2;
81 | remoteGlobalIDString = 3C86DF461ADF2C930047B81A;
82 | remoteInfo = RCTWebSocket;
83 | };
84 | 146834031AC3E56700842450 /* PBXContainerItemProxy */ = {
85 | isa = PBXContainerItemProxy;
86 | containerPortal = 146833FF1AC3E56700842450 /* React.xcodeproj */;
87 | proxyType = 2;
88 | remoteGlobalIDString = 83CBBA2E1A601D0E00E9B192;
89 | remoteInfo = React;
90 | };
91 | 78C398B81ACF4ADC00677621 /* PBXContainerItemProxy */ = {
92 | isa = PBXContainerItemProxy;
93 | containerPortal = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */;
94 | proxyType = 2;
95 | remoteGlobalIDString = 134814201AA4EA6300B7C361;
96 | remoteInfo = RCTLinking;
97 | };
98 | 832341B41AAA6A8300B99B32 /* PBXContainerItemProxy */ = {
99 | isa = PBXContainerItemProxy;
100 | containerPortal = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */;
101 | proxyType = 2;
102 | remoteGlobalIDString = 58B5119B1A9E6C1200147676;
103 | remoteInfo = RCTText;
104 | };
105 | /* End PBXContainerItemProxy section */
106 |
107 | /* Begin PBXFileReference section */
108 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; name = main.jsbundle; path = main.jsbundle; sourceTree = ""; };
109 | 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTActionSheet.xcodeproj; path = ../node_modules/react-native/Libraries/ActionSheetIOS/RCTActionSheet.xcodeproj; sourceTree = ""; };
110 | 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTGeolocation.xcodeproj; path = ../node_modules/react-native/Libraries/Geolocation/RCTGeolocation.xcodeproj; sourceTree = ""; };
111 | 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTImage.xcodeproj; path = ../node_modules/react-native/Libraries/Image/RCTImage.xcodeproj; sourceTree = ""; };
112 | 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTNetwork.xcodeproj; path = ../node_modules/react-native/Libraries/Network/RCTNetwork.xcodeproj; sourceTree = ""; };
113 | 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTVibration.xcodeproj; path = ../node_modules/react-native/Libraries/Vibration/RCTVibration.xcodeproj; sourceTree = ""; };
114 | 00E356EE1AD99517003FC87E /* RefreshableListViewTestTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RefreshableListViewTestTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
115 | 00E356F11AD99517003FC87E /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; };
116 | 00E356F21AD99517003FC87E /* RefreshableListViewTestTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = RefreshableListViewTestTests.m; sourceTree = ""; };
117 | 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTSettings.xcodeproj; path = ../node_modules/react-native/Libraries/Settings/RCTSettings.xcodeproj; sourceTree = ""; };
118 | 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTWebSocket.xcodeproj; path = ../node_modules/react-native/Libraries/WebSocket/RCTWebSocket.xcodeproj; sourceTree = ""; };
119 | 13B07F961A680F5B00A75B9A /* RefreshableListViewTest.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = RefreshableListViewTest.app; sourceTree = BUILT_PRODUCTS_DIR; };
120 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = AppDelegate.h; path = RefreshableListViewTest/AppDelegate.h; sourceTree = ""; };
121 | 13B07FB01A68108700A75B9A /* AppDelegate.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = AppDelegate.m; path = RefreshableListViewTest/AppDelegate.m; sourceTree = ""; };
122 | 13B07FB21A68108700A75B9A /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = Base.lproj/LaunchScreen.xib; sourceTree = ""; };
123 | 13B07FB51A68108700A75B9A /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Images.xcassets; path = RefreshableListViewTest/Images.xcassets; sourceTree = ""; };
124 | 13B07FB61A68108700A75B9A /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = Info.plist; path = RefreshableListViewTest/Info.plist; sourceTree = ""; };
125 | 13B07FB71A68108700A75B9A /* main.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = main.m; path = RefreshableListViewTest/main.m; sourceTree = ""; };
126 | 146833FF1AC3E56700842450 /* React.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = React.xcodeproj; path = ../node_modules/react-native/React/React.xcodeproj; sourceTree = ""; };
127 | 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTLinking.xcodeproj; path = ../node_modules/react-native/Libraries/LinkingIOS/RCTLinking.xcodeproj; sourceTree = ""; };
128 | 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTText.xcodeproj; path = ../node_modules/react-native/Libraries/Text/RCTText.xcodeproj; sourceTree = ""; };
129 | /* End PBXFileReference section */
130 |
131 | /* Begin PBXFrameworksBuildPhase section */
132 | 00E356EB1AD99517003FC87E /* Frameworks */ = {
133 | isa = PBXFrameworksBuildPhase;
134 | buildActionMask = 2147483647;
135 | files = (
136 | );
137 | runOnlyForDeploymentPostprocessing = 0;
138 | };
139 | 13B07F8C1A680F5B00A75B9A /* Frameworks */ = {
140 | isa = PBXFrameworksBuildPhase;
141 | buildActionMask = 2147483647;
142 | files = (
143 | 146834051AC3E58100842450 /* libReact.a in Frameworks */,
144 | 00C302E51ABCBA2D00DB3ED1 /* libRCTActionSheet.a in Frameworks */,
145 | 00C302E71ABCBA2D00DB3ED1 /* libRCTGeolocation.a in Frameworks */,
146 | 00C302E81ABCBA2D00DB3ED1 /* libRCTImage.a in Frameworks */,
147 | 133E29F31AD74F7200F7D852 /* libRCTLinking.a in Frameworks */,
148 | 00C302E91ABCBA2D00DB3ED1 /* libRCTNetwork.a in Frameworks */,
149 | 139105C61AF99C1200B5F7CC /* libRCTSettings.a in Frameworks */,
150 | 832341BD1AAA6AB300B99B32 /* libRCTText.a in Frameworks */,
151 | 00C302EA1ABCBA2D00DB3ED1 /* libRCTVibration.a in Frameworks */,
152 | 139FDEF61B0652A700C62182 /* libRCTWebSocket.a in Frameworks */,
153 | );
154 | runOnlyForDeploymentPostprocessing = 0;
155 | };
156 | /* End PBXFrameworksBuildPhase section */
157 |
158 | /* Begin PBXGroup section */
159 | 00C302A81ABCB8CE00DB3ED1 /* Products */ = {
160 | isa = PBXGroup;
161 | children = (
162 | 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */,
163 | );
164 | name = Products;
165 | sourceTree = "";
166 | };
167 | 00C302B61ABCB90400DB3ED1 /* Products */ = {
168 | isa = PBXGroup;
169 | children = (
170 | 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */,
171 | );
172 | name = Products;
173 | sourceTree = "";
174 | };
175 | 00C302BC1ABCB91800DB3ED1 /* Products */ = {
176 | isa = PBXGroup;
177 | children = (
178 | 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */,
179 | );
180 | name = Products;
181 | sourceTree = "";
182 | };
183 | 00C302D41ABCB9D200DB3ED1 /* Products */ = {
184 | isa = PBXGroup;
185 | children = (
186 | 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */,
187 | );
188 | name = Products;
189 | sourceTree = "";
190 | };
191 | 00C302E01ABCB9EE00DB3ED1 /* Products */ = {
192 | isa = PBXGroup;
193 | children = (
194 | 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */,
195 | );
196 | name = Products;
197 | sourceTree = "";
198 | };
199 | 00E356EF1AD99517003FC87E /* RefreshableListViewTestTests */ = {
200 | isa = PBXGroup;
201 | children = (
202 | 00E356F21AD99517003FC87E /* RefreshableListViewTestTests.m */,
203 | 00E356F01AD99517003FC87E /* Supporting Files */,
204 | );
205 | path = RefreshableListViewTestTests;
206 | sourceTree = "";
207 | };
208 | 00E356F01AD99517003FC87E /* Supporting Files */ = {
209 | isa = PBXGroup;
210 | children = (
211 | 00E356F11AD99517003FC87E /* Info.plist */,
212 | );
213 | name = "Supporting Files";
214 | sourceTree = "";
215 | };
216 | 139105B71AF99BAD00B5F7CC /* Products */ = {
217 | isa = PBXGroup;
218 | children = (
219 | 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */,
220 | );
221 | name = Products;
222 | sourceTree = "";
223 | };
224 | 139FDEE71B06529A00C62182 /* Products */ = {
225 | isa = PBXGroup;
226 | children = (
227 | 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */,
228 | );
229 | name = Products;
230 | sourceTree = "";
231 | };
232 | 13B07FAE1A68108700A75B9A /* RefreshableListViewTest */ = {
233 | isa = PBXGroup;
234 | children = (
235 | 008F07F21AC5B25A0029DE68 /* main.jsbundle */,
236 | 13B07FAF1A68108700A75B9A /* AppDelegate.h */,
237 | 13B07FB01A68108700A75B9A /* AppDelegate.m */,
238 | 13B07FB51A68108700A75B9A /* Images.xcassets */,
239 | 13B07FB61A68108700A75B9A /* Info.plist */,
240 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */,
241 | 13B07FB71A68108700A75B9A /* main.m */,
242 | );
243 | name = RefreshableListViewTest;
244 | sourceTree = "";
245 | };
246 | 146834001AC3E56700842450 /* Products */ = {
247 | isa = PBXGroup;
248 | children = (
249 | 146834041AC3E56700842450 /* libReact.a */,
250 | );
251 | name = Products;
252 | sourceTree = "";
253 | };
254 | 78C398B11ACF4ADC00677621 /* Products */ = {
255 | isa = PBXGroup;
256 | children = (
257 | 78C398B91ACF4ADC00677621 /* libRCTLinking.a */,
258 | );
259 | name = Products;
260 | sourceTree = "";
261 | };
262 | 832341AE1AAA6A7D00B99B32 /* Libraries */ = {
263 | isa = PBXGroup;
264 | children = (
265 | 146833FF1AC3E56700842450 /* React.xcodeproj */,
266 | 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */,
267 | 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */,
268 | 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */,
269 | 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */,
270 | 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */,
271 | 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */,
272 | 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */,
273 | 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */,
274 | 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */,
275 | );
276 | name = Libraries;
277 | sourceTree = "";
278 | };
279 | 832341B11AAA6A8300B99B32 /* Products */ = {
280 | isa = PBXGroup;
281 | children = (
282 | 832341B51AAA6A8300B99B32 /* libRCTText.a */,
283 | );
284 | name = Products;
285 | sourceTree = "";
286 | };
287 | 83CBB9F61A601CBA00E9B192 = {
288 | isa = PBXGroup;
289 | children = (
290 | 13B07FAE1A68108700A75B9A /* RefreshableListViewTest */,
291 | 832341AE1AAA6A7D00B99B32 /* Libraries */,
292 | 00E356EF1AD99517003FC87E /* RefreshableListViewTestTests */,
293 | 83CBBA001A601CBA00E9B192 /* Products */,
294 | );
295 | indentWidth = 2;
296 | sourceTree = "";
297 | tabWidth = 2;
298 | };
299 | 83CBBA001A601CBA00E9B192 /* Products */ = {
300 | isa = PBXGroup;
301 | children = (
302 | 13B07F961A680F5B00A75B9A /* RefreshableListViewTest.app */,
303 | 00E356EE1AD99517003FC87E /* RefreshableListViewTestTests.xctest */,
304 | );
305 | name = Products;
306 | sourceTree = "";
307 | };
308 | /* End PBXGroup section */
309 |
310 | /* Begin PBXNativeTarget section */
311 | 00E356ED1AD99517003FC87E /* RefreshableListViewTestTests */ = {
312 | isa = PBXNativeTarget;
313 | buildConfigurationList = 00E357021AD99517003FC87E /* Build configuration list for PBXNativeTarget "RefreshableListViewTestTests" */;
314 | buildPhases = (
315 | 00E356EA1AD99517003FC87E /* Sources */,
316 | 00E356EB1AD99517003FC87E /* Frameworks */,
317 | 00E356EC1AD99517003FC87E /* Resources */,
318 | );
319 | buildRules = (
320 | );
321 | dependencies = (
322 | 00E356F51AD99517003FC87E /* PBXTargetDependency */,
323 | );
324 | name = RefreshableListViewTestTests;
325 | productName = RefreshableListViewTestTests;
326 | productReference = 00E356EE1AD99517003FC87E /* RefreshableListViewTestTests.xctest */;
327 | productType = "com.apple.product-type.bundle.unit-test";
328 | };
329 | 13B07F861A680F5B00A75B9A /* RefreshableListViewTest */ = {
330 | isa = PBXNativeTarget;
331 | buildConfigurationList = 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "RefreshableListViewTest" */;
332 | buildPhases = (
333 | 13B07F871A680F5B00A75B9A /* Sources */,
334 | 13B07F8C1A680F5B00A75B9A /* Frameworks */,
335 | 13B07F8E1A680F5B00A75B9A /* Resources */,
336 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */,
337 | );
338 | buildRules = (
339 | );
340 | dependencies = (
341 | );
342 | name = RefreshableListViewTest;
343 | productName = "Hello World";
344 | productReference = 13B07F961A680F5B00A75B9A /* RefreshableListViewTest.app */;
345 | productType = "com.apple.product-type.application";
346 | };
347 | /* End PBXNativeTarget section */
348 |
349 | /* Begin PBXProject section */
350 | 83CBB9F71A601CBA00E9B192 /* Project object */ = {
351 | isa = PBXProject;
352 | attributes = {
353 | LastUpgradeCheck = 0610;
354 | ORGANIZATIONNAME = Facebook;
355 | TargetAttributes = {
356 | 00E356ED1AD99517003FC87E = {
357 | CreatedOnToolsVersion = 6.2;
358 | TestTargetID = 13B07F861A680F5B00A75B9A;
359 | };
360 | };
361 | };
362 | buildConfigurationList = 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "RefreshableListViewTest" */;
363 | compatibilityVersion = "Xcode 3.2";
364 | developmentRegion = English;
365 | hasScannedForEncodings = 0;
366 | knownRegions = (
367 | en,
368 | Base,
369 | );
370 | mainGroup = 83CBB9F61A601CBA00E9B192;
371 | productRefGroup = 83CBBA001A601CBA00E9B192 /* Products */;
372 | projectDirPath = "";
373 | projectReferences = (
374 | {
375 | ProductGroup = 00C302A81ABCB8CE00DB3ED1 /* Products */;
376 | ProjectRef = 00C302A71ABCB8CE00DB3ED1 /* RCTActionSheet.xcodeproj */;
377 | },
378 | {
379 | ProductGroup = 00C302B61ABCB90400DB3ED1 /* Products */;
380 | ProjectRef = 00C302B51ABCB90400DB3ED1 /* RCTGeolocation.xcodeproj */;
381 | },
382 | {
383 | ProductGroup = 00C302BC1ABCB91800DB3ED1 /* Products */;
384 | ProjectRef = 00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */;
385 | },
386 | {
387 | ProductGroup = 78C398B11ACF4ADC00677621 /* Products */;
388 | ProjectRef = 78C398B01ACF4ADC00677621 /* RCTLinking.xcodeproj */;
389 | },
390 | {
391 | ProductGroup = 00C302D41ABCB9D200DB3ED1 /* Products */;
392 | ProjectRef = 00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */;
393 | },
394 | {
395 | ProductGroup = 139105B71AF99BAD00B5F7CC /* Products */;
396 | ProjectRef = 139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */;
397 | },
398 | {
399 | ProductGroup = 832341B11AAA6A8300B99B32 /* Products */;
400 | ProjectRef = 832341B01AAA6A8300B99B32 /* RCTText.xcodeproj */;
401 | },
402 | {
403 | ProductGroup = 00C302E01ABCB9EE00DB3ED1 /* Products */;
404 | ProjectRef = 00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */;
405 | },
406 | {
407 | ProductGroup = 139FDEE71B06529A00C62182 /* Products */;
408 | ProjectRef = 139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */;
409 | },
410 | {
411 | ProductGroup = 146834001AC3E56700842450 /* Products */;
412 | ProjectRef = 146833FF1AC3E56700842450 /* React.xcodeproj */;
413 | },
414 | );
415 | projectRoot = "";
416 | targets = (
417 | 13B07F861A680F5B00A75B9A /* RefreshableListViewTest */,
418 | 00E356ED1AD99517003FC87E /* RefreshableListViewTestTests */,
419 | );
420 | };
421 | /* End PBXProject section */
422 |
423 | /* Begin PBXReferenceProxy section */
424 | 00C302AC1ABCB8CE00DB3ED1 /* libRCTActionSheet.a */ = {
425 | isa = PBXReferenceProxy;
426 | fileType = archive.ar;
427 | path = libRCTActionSheet.a;
428 | remoteRef = 00C302AB1ABCB8CE00DB3ED1 /* PBXContainerItemProxy */;
429 | sourceTree = BUILT_PRODUCTS_DIR;
430 | };
431 | 00C302BA1ABCB90400DB3ED1 /* libRCTGeolocation.a */ = {
432 | isa = PBXReferenceProxy;
433 | fileType = archive.ar;
434 | path = libRCTGeolocation.a;
435 | remoteRef = 00C302B91ABCB90400DB3ED1 /* PBXContainerItemProxy */;
436 | sourceTree = BUILT_PRODUCTS_DIR;
437 | };
438 | 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */ = {
439 | isa = PBXReferenceProxy;
440 | fileType = archive.ar;
441 | path = libRCTImage.a;
442 | remoteRef = 00C302BF1ABCB91800DB3ED1 /* PBXContainerItemProxy */;
443 | sourceTree = BUILT_PRODUCTS_DIR;
444 | };
445 | 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */ = {
446 | isa = PBXReferenceProxy;
447 | fileType = archive.ar;
448 | path = libRCTNetwork.a;
449 | remoteRef = 00C302DB1ABCB9D200DB3ED1 /* PBXContainerItemProxy */;
450 | sourceTree = BUILT_PRODUCTS_DIR;
451 | };
452 | 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */ = {
453 | isa = PBXReferenceProxy;
454 | fileType = archive.ar;
455 | path = libRCTVibration.a;
456 | remoteRef = 00C302E31ABCB9EE00DB3ED1 /* PBXContainerItemProxy */;
457 | sourceTree = BUILT_PRODUCTS_DIR;
458 | };
459 | 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */ = {
460 | isa = PBXReferenceProxy;
461 | fileType = archive.ar;
462 | path = libRCTSettings.a;
463 | remoteRef = 139105C01AF99BAD00B5F7CC /* PBXContainerItemProxy */;
464 | sourceTree = BUILT_PRODUCTS_DIR;
465 | };
466 | 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */ = {
467 | isa = PBXReferenceProxy;
468 | fileType = archive.ar;
469 | path = libRCTWebSocket.a;
470 | remoteRef = 139FDEF31B06529B00C62182 /* PBXContainerItemProxy */;
471 | sourceTree = BUILT_PRODUCTS_DIR;
472 | };
473 | 146834041AC3E56700842450 /* libReact.a */ = {
474 | isa = PBXReferenceProxy;
475 | fileType = archive.ar;
476 | path = libReact.a;
477 | remoteRef = 146834031AC3E56700842450 /* PBXContainerItemProxy */;
478 | sourceTree = BUILT_PRODUCTS_DIR;
479 | };
480 | 78C398B91ACF4ADC00677621 /* libRCTLinking.a */ = {
481 | isa = PBXReferenceProxy;
482 | fileType = archive.ar;
483 | path = libRCTLinking.a;
484 | remoteRef = 78C398B81ACF4ADC00677621 /* PBXContainerItemProxy */;
485 | sourceTree = BUILT_PRODUCTS_DIR;
486 | };
487 | 832341B51AAA6A8300B99B32 /* libRCTText.a */ = {
488 | isa = PBXReferenceProxy;
489 | fileType = archive.ar;
490 | path = libRCTText.a;
491 | remoteRef = 832341B41AAA6A8300B99B32 /* PBXContainerItemProxy */;
492 | sourceTree = BUILT_PRODUCTS_DIR;
493 | };
494 | /* End PBXReferenceProxy section */
495 |
496 | /* Begin PBXResourcesBuildPhase section */
497 | 00E356EC1AD99517003FC87E /* Resources */ = {
498 | isa = PBXResourcesBuildPhase;
499 | buildActionMask = 2147483647;
500 | files = (
501 | );
502 | runOnlyForDeploymentPostprocessing = 0;
503 | };
504 | 13B07F8E1A680F5B00A75B9A /* Resources */ = {
505 | isa = PBXResourcesBuildPhase;
506 | buildActionMask = 2147483647;
507 | files = (
508 | 13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */,
509 | 13B07FBD1A68108700A75B9A /* LaunchScreen.xib in Resources */,
510 | );
511 | runOnlyForDeploymentPostprocessing = 0;
512 | };
513 | /* End PBXResourcesBuildPhase section */
514 |
515 | /* Begin PBXShellScriptBuildPhase section */
516 | 00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */ = {
517 | isa = PBXShellScriptBuildPhase;
518 | buildActionMask = 2147483647;
519 | files = (
520 | );
521 | inputPaths = (
522 | );
523 | name = "Bundle React Native code and images";
524 | outputPaths = (
525 | );
526 | runOnlyForDeploymentPostprocessing = 0;
527 | shellPath = /bin/sh;
528 | shellScript = "../node_modules/react-native/packager/react-native-xcode.sh";
529 | showEnvVarsInLog = 1;
530 | };
531 | /* End PBXShellScriptBuildPhase section */
532 |
533 | /* Begin PBXSourcesBuildPhase section */
534 | 00E356EA1AD99517003FC87E /* Sources */ = {
535 | isa = PBXSourcesBuildPhase;
536 | buildActionMask = 2147483647;
537 | files = (
538 | 00E356F31AD99517003FC87E /* RefreshableListViewTestTests.m in Sources */,
539 | );
540 | runOnlyForDeploymentPostprocessing = 0;
541 | };
542 | 13B07F871A680F5B00A75B9A /* Sources */ = {
543 | isa = PBXSourcesBuildPhase;
544 | buildActionMask = 2147483647;
545 | files = (
546 | 13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */,
547 | 13B07FC11A68108700A75B9A /* main.m in Sources */,
548 | );
549 | runOnlyForDeploymentPostprocessing = 0;
550 | };
551 | /* End PBXSourcesBuildPhase section */
552 |
553 | /* Begin PBXTargetDependency section */
554 | 00E356F51AD99517003FC87E /* PBXTargetDependency */ = {
555 | isa = PBXTargetDependency;
556 | target = 13B07F861A680F5B00A75B9A /* RefreshableListViewTest */;
557 | targetProxy = 00E356F41AD99517003FC87E /* PBXContainerItemProxy */;
558 | };
559 | /* End PBXTargetDependency section */
560 |
561 | /* Begin PBXVariantGroup section */
562 | 13B07FB11A68108700A75B9A /* LaunchScreen.xib */ = {
563 | isa = PBXVariantGroup;
564 | children = (
565 | 13B07FB21A68108700A75B9A /* Base */,
566 | );
567 | name = LaunchScreen.xib;
568 | path = RefreshableListViewTest;
569 | sourceTree = "";
570 | };
571 | /* End PBXVariantGroup section */
572 |
573 | /* Begin XCBuildConfiguration section */
574 | 00E356F61AD99517003FC87E /* Debug */ = {
575 | isa = XCBuildConfiguration;
576 | buildSettings = {
577 | BUNDLE_LOADER = "$(TEST_HOST)";
578 | FRAMEWORK_SEARCH_PATHS = (
579 | "$(SDKROOT)/Developer/Library/Frameworks",
580 | "$(inherited)",
581 | );
582 | GCC_PREPROCESSOR_DEFINITIONS = (
583 | "DEBUG=1",
584 | "$(inherited)",
585 | );
586 | INFOPLIST_FILE = RefreshableListViewTestTests/Info.plist;
587 | IPHONEOS_DEPLOYMENT_TARGET = 8.2;
588 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks";
589 | PRODUCT_NAME = "$(TARGET_NAME)";
590 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/RefreshableListViewTest.app/RefreshableListViewTest";
591 | };
592 | name = Debug;
593 | };
594 | 00E356F71AD99517003FC87E /* Release */ = {
595 | isa = XCBuildConfiguration;
596 | buildSettings = {
597 | BUNDLE_LOADER = "$(TEST_HOST)";
598 | COPY_PHASE_STRIP = NO;
599 | FRAMEWORK_SEARCH_PATHS = (
600 | "$(SDKROOT)/Developer/Library/Frameworks",
601 | "$(inherited)",
602 | );
603 | INFOPLIST_FILE = RefreshableListViewTestTests/Info.plist;
604 | IPHONEOS_DEPLOYMENT_TARGET = 8.2;
605 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks @loader_path/Frameworks";
606 | PRODUCT_NAME = "$(TARGET_NAME)";
607 | TEST_HOST = "$(BUILT_PRODUCTS_DIR)/RefreshableListViewTest.app/RefreshableListViewTest";
608 | };
609 | name = Release;
610 | };
611 | 13B07F941A680F5B00A75B9A /* Debug */ = {
612 | isa = XCBuildConfiguration;
613 | buildSettings = {
614 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
615 | DEAD_CODE_STRIPPING = NO;
616 | HEADER_SEARCH_PATHS = (
617 | "$(inherited)",
618 | /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include,
619 | "$(SRCROOT)/../node_modules/react-native/React/**",
620 | );
621 | INFOPLIST_FILE = "RefreshableListViewTest/Info.plist";
622 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
623 | OTHER_LDFLAGS = "-ObjC";
624 | PRODUCT_NAME = RefreshableListViewTest;
625 | };
626 | name = Debug;
627 | };
628 | 13B07F951A680F5B00A75B9A /* Release */ = {
629 | isa = XCBuildConfiguration;
630 | buildSettings = {
631 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
632 | HEADER_SEARCH_PATHS = (
633 | "$(inherited)",
634 | /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include,
635 | "$(SRCROOT)/../node_modules/react-native/React/**",
636 | );
637 | INFOPLIST_FILE = "RefreshableListViewTest/Info.plist";
638 | LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
639 | OTHER_LDFLAGS = "-ObjC";
640 | PRODUCT_NAME = RefreshableListViewTest;
641 | };
642 | name = Release;
643 | };
644 | 83CBBA201A601CBA00E9B192 /* Debug */ = {
645 | isa = XCBuildConfiguration;
646 | buildSettings = {
647 | ALWAYS_SEARCH_USER_PATHS = NO;
648 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
649 | CLANG_CXX_LIBRARY = "libc++";
650 | CLANG_ENABLE_MODULES = YES;
651 | CLANG_ENABLE_OBJC_ARC = YES;
652 | CLANG_WARN_BOOL_CONVERSION = YES;
653 | CLANG_WARN_CONSTANT_CONVERSION = YES;
654 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
655 | CLANG_WARN_EMPTY_BODY = YES;
656 | CLANG_WARN_ENUM_CONVERSION = YES;
657 | CLANG_WARN_INT_CONVERSION = YES;
658 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
659 | CLANG_WARN_UNREACHABLE_CODE = YES;
660 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
661 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
662 | COPY_PHASE_STRIP = NO;
663 | ENABLE_STRICT_OBJC_MSGSEND = YES;
664 | GCC_C_LANGUAGE_STANDARD = gnu99;
665 | GCC_DYNAMIC_NO_PIC = NO;
666 | GCC_OPTIMIZATION_LEVEL = 0;
667 | GCC_PREPROCESSOR_DEFINITIONS = (
668 | "DEBUG=1",
669 | "$(inherited)",
670 | );
671 | GCC_SYMBOLS_PRIVATE_EXTERN = NO;
672 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
673 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
674 | GCC_WARN_UNDECLARED_SELECTOR = YES;
675 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
676 | GCC_WARN_UNUSED_FUNCTION = YES;
677 | GCC_WARN_UNUSED_VARIABLE = YES;
678 | HEADER_SEARCH_PATHS = (
679 | "$(inherited)",
680 | /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include,
681 | "$(SRCROOT)/../node_modules/react-native/React/**",
682 | );
683 | IPHONEOS_DEPLOYMENT_TARGET = 7.0;
684 | MTL_ENABLE_DEBUG_INFO = YES;
685 | ONLY_ACTIVE_ARCH = YES;
686 | SDKROOT = iphoneos;
687 | };
688 | name = Debug;
689 | };
690 | 83CBBA211A601CBA00E9B192 /* Release */ = {
691 | isa = XCBuildConfiguration;
692 | buildSettings = {
693 | ALWAYS_SEARCH_USER_PATHS = NO;
694 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++0x";
695 | CLANG_CXX_LIBRARY = "libc++";
696 | CLANG_ENABLE_MODULES = YES;
697 | CLANG_ENABLE_OBJC_ARC = YES;
698 | CLANG_WARN_BOOL_CONVERSION = YES;
699 | CLANG_WARN_CONSTANT_CONVERSION = YES;
700 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
701 | CLANG_WARN_EMPTY_BODY = YES;
702 | CLANG_WARN_ENUM_CONVERSION = YES;
703 | CLANG_WARN_INT_CONVERSION = YES;
704 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
705 | CLANG_WARN_UNREACHABLE_CODE = YES;
706 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
707 | "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
708 | COPY_PHASE_STRIP = YES;
709 | ENABLE_NS_ASSERTIONS = NO;
710 | ENABLE_STRICT_OBJC_MSGSEND = YES;
711 | GCC_C_LANGUAGE_STANDARD = gnu99;
712 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
713 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
714 | GCC_WARN_UNDECLARED_SELECTOR = YES;
715 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
716 | GCC_WARN_UNUSED_FUNCTION = YES;
717 | GCC_WARN_UNUSED_VARIABLE = YES;
718 | HEADER_SEARCH_PATHS = (
719 | "$(inherited)",
720 | /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/include,
721 | "$(SRCROOT)/../node_modules/react-native/React/**",
722 | );
723 | IPHONEOS_DEPLOYMENT_TARGET = 7.0;
724 | MTL_ENABLE_DEBUG_INFO = NO;
725 | SDKROOT = iphoneos;
726 | VALIDATE_PRODUCT = YES;
727 | };
728 | name = Release;
729 | };
730 | /* End XCBuildConfiguration section */
731 |
732 | /* Begin XCConfigurationList section */
733 | 00E357021AD99517003FC87E /* Build configuration list for PBXNativeTarget "RefreshableListViewTestTests" */ = {
734 | isa = XCConfigurationList;
735 | buildConfigurations = (
736 | 00E356F61AD99517003FC87E /* Debug */,
737 | 00E356F71AD99517003FC87E /* Release */,
738 | );
739 | defaultConfigurationIsVisible = 0;
740 | defaultConfigurationName = Release;
741 | };
742 | 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "RefreshableListViewTest" */ = {
743 | isa = XCConfigurationList;
744 | buildConfigurations = (
745 | 13B07F941A680F5B00A75B9A /* Debug */,
746 | 13B07F951A680F5B00A75B9A /* Release */,
747 | );
748 | defaultConfigurationIsVisible = 0;
749 | defaultConfigurationName = Release;
750 | };
751 | 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "RefreshableListViewTest" */ = {
752 | isa = XCConfigurationList;
753 | buildConfigurations = (
754 | 83CBBA201A601CBA00E9B192 /* Debug */,
755 | 83CBBA211A601CBA00E9B192 /* Release */,
756 | );
757 | defaultConfigurationIsVisible = 0;
758 | defaultConfigurationName = Release;
759 | };
760 | /* End XCConfigurationList section */
761 | };
762 | rootObject = 83CBB9F71A601CBA00E9B192 /* Project object */;
763 | }
764 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest.xcodeproj/xcshareddata/xcschemes/RefreshableListViewTest.xcscheme:
--------------------------------------------------------------------------------
1 |
2 |
5 |
8 |
9 |
15 |
21 |
22 |
23 |
29 |
35 |
36 |
37 |
38 |
39 |
44 |
45 |
47 |
53 |
54 |
55 |
56 |
57 |
63 |
64 |
65 |
66 |
75 |
77 |
83 |
84 |
85 |
86 |
87 |
88 |
94 |
96 |
102 |
103 |
104 |
105 |
107 |
108 |
111 |
112 |
113 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest/AppDelegate.h:
--------------------------------------------------------------------------------
1 | /**
2 | * Copyright (c) 2015-present, Facebook, Inc.
3 | * All rights reserved.
4 | *
5 | * This source code is licensed under the BSD-style license found in the
6 | * LICENSE file in the root directory of this source tree. An additional grant
7 | * of patent rights can be found in the PATENTS file in the same directory.
8 | */
9 |
10 | #import
11 |
12 | @interface AppDelegate : UIResponder
13 |
14 | @property (nonatomic, strong) UIWindow *window;
15 |
16 | @end
17 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest/AppDelegate.m:
--------------------------------------------------------------------------------
1 | /**
2 | * Copyright (c) 2015-present, Facebook, Inc.
3 | * All rights reserved.
4 | *
5 | * This source code is licensed under the BSD-style license found in the
6 | * LICENSE file in the root directory of this source tree. An additional grant
7 | * of patent rights can be found in the PATENTS file in the same directory.
8 | */
9 |
10 | #import "AppDelegate.h"
11 |
12 | #import "RCTRootView.h"
13 |
14 | @implementation AppDelegate
15 |
16 | - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
17 | {
18 | NSURL *jsCodeLocation;
19 |
20 | /**
21 | * Loading JavaScript code - uncomment the one you want.
22 | *
23 | * OPTION 1
24 | * Load from development server. Start the server from the repository root:
25 | *
26 | * $ npm start
27 | *
28 | * To run on device, change `localhost` to the IP address of your computer
29 | * (you can get this by typing `ifconfig` into the terminal and selecting the
30 | * `inet` value under `en0:`) and make sure your computer and iOS device are
31 | * on the same Wi-Fi network.
32 | */
33 |
34 | jsCodeLocation = [NSURL URLWithString:@"http://localhost:8081/index.ios.bundle?platform=ios&dev=true"];
35 |
36 | /**
37 | * OPTION 2
38 | * Load from pre-bundled file on disk. The static bundle is automatically
39 | * generated by "Bundle React Native code and images" build step.
40 | */
41 |
42 | // jsCodeLocation = [[NSBundle mainBundle] URLForResource:@"main" withExtension:@"jsbundle"];
43 |
44 | RCTRootView *rootView = [[RCTRootView alloc] initWithBundleURL:jsCodeLocation
45 | moduleName:@"RefreshableListViewTest"
46 | initialProperties:nil
47 | launchOptions:launchOptions];
48 |
49 | self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
50 | UIViewController *rootViewController = [UIViewController new];
51 | rootViewController.view = rootView;
52 | self.window.rootViewController = rootViewController;
53 | [self.window makeKeyAndVisible];
54 | return YES;
55 | }
56 |
57 | @end
58 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest/Base.lproj/LaunchScreen.xib:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
21 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest/Images.xcassets/AppIcon.appiconset/Contents.json:
--------------------------------------------------------------------------------
1 | {
2 | "images" : [
3 | {
4 | "idiom" : "iphone",
5 | "size" : "29x29",
6 | "scale" : "2x"
7 | },
8 | {
9 | "idiom" : "iphone",
10 | "size" : "29x29",
11 | "scale" : "3x"
12 | },
13 | {
14 | "idiom" : "iphone",
15 | "size" : "40x40",
16 | "scale" : "2x"
17 | },
18 | {
19 | "idiom" : "iphone",
20 | "size" : "40x40",
21 | "scale" : "3x"
22 | },
23 | {
24 | "idiom" : "iphone",
25 | "size" : "60x60",
26 | "scale" : "2x"
27 | },
28 | {
29 | "idiom" : "iphone",
30 | "size" : "60x60",
31 | "scale" : "3x"
32 | }
33 | ],
34 | "info" : {
35 | "version" : 1,
36 | "author" : "xcode"
37 | }
38 | }
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest/Info.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | CFBundleDevelopmentRegion
6 | en
7 | CFBundleExecutable
8 | $(EXECUTABLE_NAME)
9 | CFBundleIdentifier
10 | org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)
11 | CFBundleInfoDictionaryVersion
12 | 6.0
13 | CFBundleName
14 | $(PRODUCT_NAME)
15 | CFBundlePackageType
16 | APPL
17 | CFBundleShortVersionString
18 | 1.0
19 | CFBundleSignature
20 | ????
21 | CFBundleVersion
22 | 1
23 | LSRequiresIPhoneOS
24 |
25 | UILaunchStoryboardName
26 | LaunchScreen
27 | UIRequiredDeviceCapabilities
28 |
29 | armv7
30 |
31 | UISupportedInterfaceOrientations
32 |
33 | UIInterfaceOrientationPortrait
34 | UIInterfaceOrientationLandscapeLeft
35 | UIInterfaceOrientationLandscapeRight
36 |
37 | UIViewControllerBasedStatusBarAppearance
38 |
39 | NSLocationWhenInUseUsageDescription
40 |
41 | NSAppTransportSecurity
42 |
43 |
44 | NSAllowsArbitraryLoads
45 |
46 |
47 |
48 |
49 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTest/main.m:
--------------------------------------------------------------------------------
1 | /**
2 | * Copyright (c) 2015-present, Facebook, Inc.
3 | * All rights reserved.
4 | *
5 | * This source code is licensed under the BSD-style license found in the
6 | * LICENSE file in the root directory of this source tree. An additional grant
7 | * of patent rights can be found in the PATENTS file in the same directory.
8 | */
9 |
10 | #import
11 |
12 | #import "AppDelegate.h"
13 |
14 | int main(int argc, char * argv[]) {
15 | @autoreleasepool {
16 | return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
17 | }
18 | }
19 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTestTests/Info.plist:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | CFBundleDevelopmentRegion
6 | en
7 | CFBundleExecutable
8 | $(EXECUTABLE_NAME)
9 | CFBundleIdentifier
10 | org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)
11 | CFBundleInfoDictionaryVersion
12 | 6.0
13 | CFBundleName
14 | $(PRODUCT_NAME)
15 | CFBundlePackageType
16 | BNDL
17 | CFBundleShortVersionString
18 | 1.0
19 | CFBundleSignature
20 | ????
21 | CFBundleVersion
22 | 1
23 |
24 |
25 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/ios/RefreshableListViewTestTests/RefreshableListViewTestTests.m:
--------------------------------------------------------------------------------
1 | /**
2 | * Copyright (c) 2015-present, Facebook, Inc.
3 | * All rights reserved.
4 | *
5 | * This source code is licensed under the BSD-style license found in the
6 | * LICENSE file in the root directory of this source tree. An additional grant
7 | * of patent rights can be found in the PATENTS file in the same directory.
8 | */
9 |
10 | #import
11 | #import
12 |
13 | #import "RCTLog.h"
14 | #import "RCTRootView.h"
15 |
16 | #define TIMEOUT_SECONDS 240
17 | #define TEXT_TO_LOOK_FOR @"Welcome to React Native!"
18 |
19 | @interface RefreshableListViewTestTests : XCTestCase
20 |
21 | @end
22 |
23 | @implementation RefreshableListViewTestTests
24 |
25 | - (BOOL)findSubviewInView:(UIView *)view matching:(BOOL(^)(UIView *view))test
26 | {
27 | if (test(view)) {
28 | return YES;
29 | }
30 | for (UIView *subview in [view subviews]) {
31 | if ([self findSubviewInView:subview matching:test]) {
32 | return YES;
33 | }
34 | }
35 | return NO;
36 | }
37 |
38 | - (void)testRendersWelcomeScreen
39 | {
40 | UIViewController *vc = [[[[UIApplication sharedApplication] delegate] window] rootViewController];
41 | NSDate *date = [NSDate dateWithTimeIntervalSinceNow:TIMEOUT_SECONDS];
42 | BOOL foundElement = NO;
43 |
44 | __block NSString *redboxError = nil;
45 | RCTSetLogFunction(^(RCTLogLevel level, RCTLogSource source, NSString *fileName, NSNumber *lineNumber, NSString *message) {
46 | if (level >= RCTLogLevelError) {
47 | redboxError = message;
48 | }
49 | });
50 |
51 | while ([date timeIntervalSinceNow] > 0 && !foundElement && !redboxError) {
52 | [[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
53 | [[NSRunLoop mainRunLoop] runMode:NSRunLoopCommonModes beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
54 |
55 | foundElement = [self findSubviewInView:vc.view matching:^BOOL(UIView *view) {
56 | if ([view.accessibilityLabel isEqualToString:TEXT_TO_LOOK_FOR]) {
57 | return YES;
58 | }
59 | return NO;
60 | }];
61 | }
62 |
63 | RCTSetLogFunction(RCTDefaultLogFunction);
64 |
65 | XCTAssertNil(redboxError, @"RedBox error: %@", redboxError);
66 | XCTAssertTrue(foundElement, @"Couldn't find element with text '%@' in %d seconds", TEXT_TO_LOOK_FOR, TIMEOUT_SECONDS);
67 | }
68 |
69 |
70 | @end
71 |
--------------------------------------------------------------------------------
/test/RefreshableListViewTest/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "RefreshableListViewTest",
3 | "version": "0.0.1",
4 | "private": true,
5 | "scripts": {
6 | "start": "react-native start"
7 | },
8 | "dependencies": {
9 | "react-native-refreshable-listview": "file:../../",
10 | "react-native": "^0.17.0"
11 | }
12 | }
13 |
--------------------------------------------------------------------------------
/test/support/preprocessor.js:
--------------------------------------------------------------------------------
1 | var fs = require('fs')
2 | var babel = require('babel-core')
3 | var sourceMapPath = require('./sourceMapPath')
4 |
5 | module.exports = {
6 | process: function (src, filename) {
7 | // Allow the stage to be configured by an environment
8 | // variable, but use Babel's default stage (2) if
9 | // no environment variable is specified.
10 | var stage = process.env.BABEL_JEST_STAGE || 2
11 |
12 | // Ignore all files within node_modules
13 | // babel files can be .js, .es, .jsx or .es6
14 | if (filename.indexOf("node_modules") === -1 && babel.canCompile(filename)) {
15 | var result = babel.transform(src, { filename: filename, stage: stage, sourceMap: true })
16 |
17 | fs.writeFileSync(sourceMapPath(filename), JSON.stringify(result.map))
18 |
19 | return result.code
20 | } else {
21 | return src
22 | }
23 | }
24 | }
25 |
--------------------------------------------------------------------------------
/test/support/setupTestFramework.js:
--------------------------------------------------------------------------------
1 | let fs = require('fs')
2 | let sourceMapSupport = require('source-map-support')
3 | let absolutePath = require('absolute-path')
4 | let inspectReactElement = require('inspect-react-element')
5 | let sourceMapPath = require('./sourceMapPath')
6 |
7 | sourceMapSupport.install({
8 | retrieveSourceMap: (srcpath) => {
9 | if (srcpath.indexOf("node_modules") > -1) return
10 | if (!absolutePath(srcpath)) return
11 |
12 | try {
13 | let map = fs.readFileSync(sourceMapPath(srcpath), 'utf8')
14 | if (map) return {map}
15 | } catch (err) {
16 | console.error(err.code + ' when loading sourcemap for ' + srcpath)
17 | }
18 | }
19 | })
20 |
21 | jasmine.getEnv().beforeEach(function() {
22 | let React = require.requireActual('react/dist/react-with-addons')
23 | this.shallowRender = require('./shallowRender')
24 | window.printElement = (el) => console.log('\n' + inspectReactElement(el))
25 |
26 | jest.setMock('react-native', require.requireActual('../../lib/__mocks__/react-native'))
27 |
28 | this.addMatchers(require('jasmine-object-matchers-jest')['1.3'])
29 |
30 | this.addMatchers({
31 | toBeObject() {
32 | return this.actual && typeof this.actual == 'object'
33 | },
34 | toBeFunction() {
35 | return typeof this.actual == 'function'
36 | },
37 | toBeReactElement() {
38 | return React.isValidElement(this.actual)
39 | },
40 | toBeReactElementOfType(type) {
41 | return (
42 | React.isValidElement(this.actual) &&
43 | this.actual.type === type
44 | )
45 | },
46 | })
47 | })
48 |
--------------------------------------------------------------------------------
/test/support/shallowRender.js:
--------------------------------------------------------------------------------
1 | var TestUtils = require('react-addons-test-utils')
2 |
3 | function shallowRender(element) {
4 | var shallowRenderer = TestUtils.createRenderer()
5 |
6 | // add a method to expose instantiated component
7 | shallowRenderer.getComponent = () => shallowRenderer._instance._instance
8 |
9 | shallowRenderer.render(element)
10 | return shallowRenderer
11 | }
12 |
13 | module.exports = shallowRender
14 |
--------------------------------------------------------------------------------
/test/support/sourceMapPath.js:
--------------------------------------------------------------------------------
1 | var fs = require('fs')
2 | var os = require('os')
3 | var path = require('path')
4 | var crypto = require('crypto')
5 |
6 | var TMPDIR = path.join(os.tmpdir(), 'react-native-refreshable-listview-jest')
7 | try {
8 | fs.mkdirSync(TMPDIR)
9 | } catch (err) {
10 | if (err.code != 'EEXIST') console.error(err)
11 | }
12 |
13 | function sourceMapPath(srcpath) {
14 | return path.join(TMPDIR, crypto.createHash('md5').update(srcpath).digest('hex') + '.map')
15 | }
16 |
17 | sourceMapPath.TMPDIR = TMPDIR
18 |
19 | module.exports = sourceMapPath
20 |
--------------------------------------------------------------------------------