├── .eslintrc.js ├── .gitignore ├── CHANGELOG.md ├── LICENSE.md ├── README.md ├── components ├── SledComponent.js ├── SledImage.js ├── github-logo.js ├── header.js ├── images.js ├── settings.js ├── state.js └── useWindowSize.js ├── next-env.d.ts ├── next.config.js ├── package.json ├── pages ├── css │ ├── header.css │ ├── index.css │ ├── menue.css │ ├── reset.css │ ├── settings.css │ └── toggle.css ├── custom.js ├── index.js ├── test-basic.js └── test.js ├── public ├── favicon.ico ├── plex-mono-bold.woff └── plex-mono.woff ├── react-sled-logo.jpg ├── rollup.config.js ├── sled ├── control │ ├── hooks │ │ ├── useClassName.ts │ │ ├── useClick.ts │ │ ├── useDirectionDisabled.ts │ │ └── useLabel.ts │ ├── index.css │ └── index.tsx ├── hooks │ ├── useAutoPlay.ts │ ├── useConfig.ts │ ├── useContainerStyles.ts │ ├── useCursor.ts │ ├── useDimensions.ts │ ├── useDimensionsDOM.ts │ ├── useDirection.ts │ ├── useDragGesture.ts │ ├── useDragging.ts │ ├── useFocus.ts │ ├── useKeyboard.ts │ ├── useMouseOver.ts │ ├── usePause.ts │ ├── useProportion.ts │ ├── useRewind.ts │ ├── useSelect.ts │ ├── useShowElements.ts │ ├── useSlideBy.ts │ ├── useSlideSteps.ts │ ├── useSliderSize.ts │ ├── useStopOnInteraction.ts │ ├── useViewCount.ts │ └── useX.ts ├── index.css ├── index.ts ├── progress │ ├── controls.tsx │ ├── index.css │ ├── index.tsx │ ├── separators.tsx │ └── track.tsx ├── sled.tsx ├── springs.tsx ├── state │ ├── index.tsx │ ├── reducer.ts │ └── types-defaults.ts ├── utils │ ├── clamp.ts │ └── debounce.ts └── views.tsx ├── tsconfig.json ├── tsconfig.rollup.json └── yarn.lock /.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | extends: "react-app", 3 | rules: { 4 | quotes: ["error", "single"], 5 | semi: ["error", "never"], 6 | indent: ["error", 2, { 7 | SwitchCase: 1 8 | }], 9 | "no-multiple-empty-lines": ["error"], 10 | "jsx-a11y/anchor-is-valid": 0 11 | } 12 | } 13 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | _local-assets 2 | /.vscode 3 | sled.code-workspace 4 | /dist 5 | 6 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 7 | 8 | # dependencies 9 | /node_modules 10 | /.pnp 11 | .pnp.js 12 | 13 | # testing 14 | /coverage 15 | 16 | # next.js 17 | /.next/ 18 | /out/ 19 | 20 | # production 21 | /build 22 | 23 | # misc 24 | .DS_Store 25 | .env* 26 | 27 | # debug 28 | npm-debug.log* 29 | yarn-debug.log* 30 | yarn-error.log* 31 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | #### 2.0.4 (2020-02-02) 2 | 3 | ##### Bug Fixes 4 | 5 | * update react-use-gesture (3050ce3f) 6 | 7 | #### 2.0.3 (2020-01-20) 8 | 9 | ##### Documentation Changes 10 | 11 | * update example-link (93e3a934) 12 | 13 | #### 2.0.2 (2020-01-20) 14 | 15 | ##### Documentation Changes 16 | 17 | * correct changelog (b707cf88) 18 | * corrections (35d6ee18) 19 | 20 | #### 2.0.1 (2020-01-20) 21 | 22 | ##### Documentation Changes 23 | 24 | * correct readme (66e0291a) 25 | 26 | ##### Bug Fixes 27 | 28 | * refactor sizing (3c959a70) 29 | 30 | ## 2.0.0 (2020-01-20) 31 | 32 | #### 1.2.6 (2019-12-22) 33 | 34 | ##### Other Changes 35 | 36 | * remove images from repo (5749455d) 37 | 38 | #### 1.2.5 (2019-10-26) 39 | 40 | ##### Bug Fixes 41 | 42 | * fix dragging. fixes #5 (e4121bdf) 43 | 44 | #### 1.2.4 (2019-10-10) 45 | 46 | ##### Bug Fixes 47 | 48 | * update gitignore (783b062a) 49 | 50 | #### 1.2.3 (2019-10-10) 51 | 52 | ##### Bug Fixes 53 | 54 | * fix package.json (c8fb5338) 55 | 56 | #### 1.2.2 (2019-10-10) 57 | 58 | ##### Bug Fixes 59 | 60 | * update package.json and readme (5b54fcf5) 61 | 62 | #### 1.2.1 (2019-10-10) 63 | 64 | ### 1.2.0 (2019-10-10) 65 | 66 | ##### New Features 67 | 68 | * replace react-with-gesture with react-use-gesture. improve performance. (5a0dd72d) 69 | * add lazy-loading to example (4740b3d3) 70 | 71 | ##### Bug Fixes 72 | 73 | * Fix onSledEnd to be called just once. (cfa9de6b) 74 | * example - minor style fixes (88912fd6) 75 | * update readme (e1212f05) 76 | * change linter from standard to eslint (c4284ead) 77 | 78 | ### 1.2.0 (2019-10-10) 79 | 80 | ##### New Features 81 | 82 | * replace react-with-gesture with react-use-gesture. improve performance. (5a0dd72d) 83 | * add lazy-loading to example (4740b3d3) 84 | 85 | ##### Bug Fixes 86 | 87 | * Fix onSledEnd to be called just once. (cfa9de6b) 88 | * example - minor style fixes (88912fd6) 89 | * update readme (e1212f05) 90 | * change linter from standard to eslint (c4284ead) 91 | 92 | #### 1.1.2 (2019-10-09) 93 | 94 | ##### Bug Fixes 95 | 96 | * fix example ssr (d34bc9ff) 97 | 98 | #### 1.1.1 (2019-10-08) 99 | 100 | ### 1.1.0 (2019-10-08) 101 | 102 | ##### Bug Fixes 103 | 104 | * fix content-z-index in example. (96193878) 105 | 106 | ##### Other Changes 107 | 108 | * fix callback bug (3b909b98) 109 | 110 | #### 1.0.8 (2019-09-07) 111 | 112 | ##### Bug Fixes 113 | 114 | * update dependecies and fix typo in readme (639efd77) 115 | 116 | #### 1.0.7 (2019-06-06) 117 | 118 | ##### Bug Fixes 119 | 120 | * Fix gatsby-config. (ef75f8f9) 121 | 122 | #### 1.0.6 (2019-06-06) 123 | 124 | ##### Documentation Changes 125 | 126 | * Fix props-table. (96942b78) 127 | 128 | #### 1.0.5 (2019-06-05) 129 | 130 | #### 1.0.4 (2019-06-05) 131 | 132 | ##### Other Changes 133 | 134 | * fix first animation. (49ea3a1d) 135 | 136 | #### 1.0.3 (2019-06-05) 137 | 138 | ##### Bug Fixes 139 | 140 | * Small change. (0549fbf7) 141 | 142 | #### 1.0.2 (2019-06-05) 143 | 144 | ##### Bug Fixes 145 | 146 | * Small change. (60ce31b2) 147 | 148 | #### 1.0.1 (2019-06-05) 149 | 150 | ##### Documentation Changes 151 | 152 | * Small change. (336465d8) 153 | 154 | ## 1.0.0 (2019-06-05) 155 | 156 | ##### New Features 157 | 158 | * First major release. (03fd1bb8) 159 | 160 | 161 | ### 0.15.0 (2019-06-03) 162 | 163 | ##### New Features 164 | 165 | * Add pause-prop. Expose state-management. (40371acc) 166 | 167 | #### 0.14.1 (2019-06-02) 168 | 169 | ### 0.14.0 (2019-06-02) 170 | 171 | ### 0.14.0 (2019-06-02) 172 | 173 | ### 0.12.0 (2019-06-02) 174 | 175 | ### 0.12.0 (2019-06-02) 176 | 177 | ### 0.13.0 (2019-06-02) 178 | 179 | ### 0.12.0 (2019-06-02) 180 | 181 | ##### New Features 182 | 183 | * Add props onSledEnd and stopOnInteraction. (97d50729) 184 | 185 | ##### Bug Fixes 186 | 187 | * Disable focus-outline. (7cb5bd06) 188 | 189 | #### 0.11.1 (2019-05-11) 190 | 191 | ##### Bug Fixes 192 | 193 | * Progress: Fix bug, when there are only 2 views. (15d8537b) 194 | 195 | ### 0.11.0 (2019-05-03) 196 | 197 | ##### New Features 198 | 199 | * Add classname ot view. (0ff5d0bf) 200 | 201 | #### 0.10.1 (2019-05-03) 202 | 203 | ##### Documentation Changes 204 | 205 | * fix docs. (74d0f7d8) 206 | 207 | ### 0.10.0 (2019-05-02) 208 | 209 | ##### Documentation Changes 210 | 211 | * improve docs and example. (da6cfdb3) 212 | 213 | ##### Bug Fixes 214 | 215 | * change cursor if dragging is deactivated. (4188989e) 216 | 217 | #### 0.9.1 (2019-04-27) 218 | 219 | ##### Bug Fixes 220 | 221 | * Fix preset-names. (43834d9b) 222 | 223 | ### 0.9.0 (2019-04-27) 224 | 225 | ##### New Features 226 | 227 | * complete controls. (6e8182c5) 228 | 229 | #### 0.8.4 (2019-04-27) 230 | 231 | ##### Bug Fixes 232 | 233 | * Go back to old domain. (584761bf) 234 | 235 | #### 0.8.3 (2019-04-27) 236 | 237 | ##### Bug Fixes 238 | 239 | * Change example-domain. (056e1a16) 240 | 241 | #### 0.8.2 (2019-04-27) 242 | 243 | ##### Bug Fixes 244 | 245 | * Make autoplay restartable. (488bc048) 246 | 247 | #### 0.8.1 (2019-04-27) 248 | 249 | ##### Bug Fixes 250 | 251 | * Correction in example. (1452f1f6) 252 | 253 | ### 0.8.0 (2019-04-27) 254 | 255 | ##### New Features 256 | 257 | * Make all props dynamic. (0801e73d) 258 | 259 | #### 0.7.2 (2019-04-27) 260 | 261 | ##### Documentation Changes 262 | 263 | * Change logo. (05fc3609) 264 | 265 | #### 0.7.1 (2019-04-27) 266 | 267 | ##### Documentation Changes 268 | 269 | * Change logo. (7e6321b3) 270 | 271 | ### 0.7.0 (2019-04-27) 272 | 273 | ##### New Features 274 | 275 | * Add rewind-prop. (0a2208f7) 276 | 277 | ##### Bug Fixes 278 | 279 | * Reverse z-indices. (2e90dd31) 280 | 281 | #### 0.6.1 (2019-04-27) 282 | 283 | ##### Documentation Changes 284 | 285 | * Fix docs. (2801f1e4) 286 | 287 | ### 0.6.0 (2019-04-27) 288 | 289 | ##### Refactors 290 | 291 | * Clean up codebase. (21e59655) 292 | 293 | ### 0.5.0 (2019-04-26) 294 | 295 | ##### New Features 296 | 297 | * Add goto-feature. (dc930a9c) 298 | 299 | ##### Refactors 300 | 301 | * Change prop 'keys' to 'keysboard'. (d83d69bb) 302 | 303 | ### 0.4.0 (2019-04-26) 304 | 305 | ##### Refactors 306 | 307 | * Change and improve sizing. (e3086bf1) 308 | 309 | ### 0.3.0 (2019-04-26) 310 | 311 | ##### New Features 312 | 313 | * Keep sled's height intact. (14686209) 314 | 315 | #### 0.2.10 (2019-04-26) 316 | 317 | #### 0.2.9 (2019-04-26) 318 | 319 | #### 0.2.8 (2019-04-26) 320 | 321 | ##### Bug Fixes 322 | 323 | * Fix readme. (9dae398a) 324 | 325 | #### 0.2.7 (2019-04-26) 326 | 327 | ##### Bug Fixes 328 | 329 | * Fix readme. (77f57b97) 330 | 331 | #### 0.2.6 (2019-04-26) 332 | 333 | ##### Bug Fixes 334 | 335 | * Fix readme. (8d0d7236) 336 | 337 | #### 0.2.5 (2019-04-26) 338 | 339 | ##### Bug Fixes 340 | 341 | * Add logo. (4af8b68b) 342 | 343 | #### 0.2.4 (2019-04-26) 344 | 345 | ##### Bug Fixes 346 | 347 | * Progress: Small bug. (0ed91265) 348 | 349 | #### 0.2.3 (2019-04-26) 350 | 351 | ##### Bug Fixes 352 | 353 | * Fix typo in readme. (e10ae2c2) 354 | 355 | #### 0.2.3 (2019-04-26) 356 | 357 | ##### Bug Fixes 358 | 359 | * Fix typo in readme. (e10ae2c2) 360 | 361 | #### 0.2.3 (2019-04-26) 362 | 363 | ##### Bug Fixes 364 | 365 | * Fix typo in readme. (e10ae2c2) 366 | 367 | #### 0.2.2 (2019-04-26) 368 | 369 | ##### Bug Fixes 370 | 371 | * Small fix. (5ed748a8) 372 | 373 | ### 0.2.0 (2019-04-26) 374 | 375 | ##### New Features 376 | 377 | * Add new name. (6b86f188) 378 | 379 | ### 0.1.0 (2019-04-26) 380 | 381 | ##### New Features 382 | 383 | * Getting close to version 1. (c2bf2baf) 384 | 385 | ##### Other Changes 386 | 387 | * Add some features. (7a382afe) 388 | 389 | #### 0.0.1 (2019-04-24) 390 | 391 | #### 1.4.8 (2019-04-15) 392 | 393 | ##### Bug Fixes 394 | 395 | * Fix imports in Control. (f330363d) 396 | 397 | #### 1.4.7 (2019-04-15) 398 | 399 | ##### Bug Fixes 400 | 401 | * Rewire Components for docs. (a6e82d75) 402 | 403 | #### 1.4.6 (2019-04-07) 404 | 405 | ##### Bug Fixes 406 | 407 | * Change align defaults to empty string. (5d9b9519) 408 | 409 | #### 1.4.5 (2019-04-07) 410 | 411 | ##### Bug Fixes 412 | 413 | * Remove overflow:hidden; Change default breakpoints. (b460725b) 414 | 415 | #### 1.4.4 (2019-04-07) 416 | 417 | #### 1.4.3 (2019-04-07) 418 | 419 | #### 1.4.2 (2019-04-07) 420 | 421 | ##### Other Changes 422 | 423 | * Add absolute links for documentation images. (c79a9ed9) 424 | 425 | #### 1.4.1 (2019-04-07) 426 | 427 | ##### Other Changes 428 | 429 | * Add illustration for breakpoints. (7dffd8cb) 430 | 431 | ### 1.4.0 (2019-04-07) 432 | 433 | ##### New Features 434 | 435 | * encapsulate breakpoint mediaqueries with max-width (9d43afc0) 436 | 437 | #### 1.3.9 (2019-04-06) 438 | 439 | ##### Documentation Changes 440 | 441 | * Fix some details. (bd958e22) 442 | 443 | #### 1.3.8 (2019-04-06) 444 | 445 | ##### Other Changes 446 | 447 | * Fixes. (460995eb) 448 | 449 | #### 1.3.7 (2019-04-06) 450 | 451 | ##### Other Changes 452 | 453 | * Add logo. (79596737) 454 | 455 | #### 1.3.6 (2019-04-05) 456 | 457 | ##### Documentation Changes 458 | 459 | * Small fixes. (9bb4c2b4) 460 | 461 | #### 1.3.5 (2019-04-05) 462 | 463 | ##### Documentation Changes 464 | 465 | * Some details. (72c0a401) 466 | 467 | #### 1.3.4 (2019-04-05) 468 | 469 | ##### Documentation Changes 470 | 471 | * Some details. (abb397f7) 472 | 473 | #### 1.3.3 (2019-04-05) 474 | 475 | ##### Documentation Changes 476 | 477 | * Change opening description. (3a66984d) 478 | 479 | #### 1.3.2 (2019-04-05) 480 | 481 | ##### Documentation Changes 482 | 483 | * Fix examples. (6379ef86) 484 | 485 | #### 1.3.1 (2019-04-05) 486 | 487 | ##### Documentation Changes 488 | 489 | * Fix examples. (e7ef84b4) 490 | 491 | ### 1.3.0 (2019-04-05) 492 | 493 | ##### New Features 494 | 495 | * Add custom control-color. (ba912939) 496 | 497 | ### 1.2.0 (2019-03-30) 498 | 499 | ##### Refactors 500 | 501 | * Change prop reset to hasChildBoxes (5072a8c4) 502 | * Improve resetting structure (7b2988e8) 503 | 504 | #### 1.1.1 (2019-03-28) 505 | 506 | ##### Bug Fixes 507 | 508 | * Add box-sizing. Remove width: 100%; (5c27360b) 509 | 510 | ### 1.1.0 (2019-03-27) 511 | 512 | ##### New Features 513 | 514 | * Add manual reset-prop to box. (09923992) 515 | 516 | ## 1.0.0 (2019-03-27) 517 | 518 | ##### Breaking Changes 519 | 520 | * First working version (6834f7f2) 521 | 522 | -------------------------------------------------------------------------------- /LICENSE.md: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2014 Jason Quense 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: 6 | 7 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. 8 | 9 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 10 | 11 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ![react-sled Logo](https://raw.github.com/andreasfaust/react-sled/master/react-sled-logo.jpg) 2 | 3 | **react-sled** is a carousel made with **react-spring**. 4 | 5 | [![NPM](https://img.shields.io/npm/v/react-sled.svg)](https://www.npmjs.com/package/react-sled) 6 | 7 | - Super-smooth spring animations (thanks to **react-spring**) 8 | - Lightweight and performant architecture 9 | - Touch- and Mousedrag (thanks to **react-with-gesture**) 10 | - Easy to style 11 | - Ready for server-side-rendering 12 | - All props are dynamically changeable 13 | - (Should be) Compatible with older Browsers from Internet Explorer 11 (Needs testing!) 14 | 15 | 🛷 [Have a look at the example!](https://react-sled.andreasfaust.com/) 16 | 17 | ## New Major release 2.0 18 | 19 | **Breaking Changes:** 20 | - Removed styled-components 21 | - Removed custom `ow`-unit 22 | - Use `react-spring` Version 9 and `react-with-gesture` Version 7 23 | 24 | **New Features:** 25 | - Full Type-Script support 26 | - Vertical Sliding 27 | - Set fixed proportion 28 | - Show multiple elements at once (`showElements`) 29 | - Move by multiple elements (`slideBy`) 30 | 31 | 32 | ## Install 33 | 34 | Install all dependencies via Yarn or NPM. 35 | 36 | ```bash 37 | yarn add react-sled react-spring@next react-use-gesture react react-dom 38 | ``` 39 | 40 | ## Usage 41 | 42 | ```jsx 43 | import React from "react"; 44 | import { Sled, Views, Progress, Control } from "react-sled"; 45 | import "react-sled/dist/main.css"; 46 | 47 | const images = ["my-image-1.jpg", "my-image-2.jpg"]; 48 | 49 | const App = () => { 50 | return ( 51 | 52 | 53 | {images.map((image, index) => ( 54 | {`My 55 | ))} 56 | 57 | 58 |
59 | 60 | 61 |
62 |
63 | {images.map((image, index) => ( 64 | 65 | ))} 66 |
67 |
68 | ); 69 | }; 70 | 71 | export default App; 72 | ``` 73 | 74 | ## Sled 75 | 76 | Sled is the wrapper-component. It takes no props. 77 | 78 | 79 | ## Views 80 | 81 | Render all your views into this component. 82 | It takes these optional props: 83 | 84 | | **Name** | **Type** | **Default** | **Description** | 85 | | :------------------- | :--------------- | :----------------------------------------------------- | :--------------------------------------------------------------------------------------------------------------------------------------------------------------- | 86 | | **width** | String | `'100%'` | Sets the viewpager’s width. Allowed units: all CSS-units | 87 | | **height** | String | `null` | Sets the viewpager’s height. | 88 | | **proportion** | String | `2:1` | Provide either a width or height and set the other dimension proportional to it. If you provide a height and a width `proportion` is disabled. | 89 | | **showElements** | Number | `1` | Determines how many Slides/Views fit in the Sled’s viewport. | 90 | | **slideBy** | Number | `1` | Determines how many Slides/Views the Sled’s slides with one movement . | 91 | | **select** | Number | `undefined` | Select certain view. | 92 | | **style** | Object | `null` | Add inline styles to the view-wrapper.  | 93 | | **keyboard** | Boolean | `true` | Set Keyboard controls.  | 94 | | **dragging** | Boolean | `true` | Set Mouse- and Touch-Dragging.  | 95 | | **dragDistance** | Number or String | `40` | Distance the user has to drag the slider to trigger action. A number is calculated in Pixel. A string is converted to a number unless it has the unit `%`, which means "percent of Sled’s width".  | 96 | | **autoPlay** | Number | `undefined` | Activates automatic Sliding-Interval in Milliseconds.  | 97 | | **config** | Number | `{ mass: 1, tension: 210, friction: 20, clamp: true }` | react-spring animation-settings.  | 98 | | **pause** | Boolean | `false` | `autoPlay` (if activated) gets paused.  | 99 | | **pauseOnMouseOver** | Boolean | `true` | `autoPlay` (if activated) gets paused, as long as the user hovers over the sled.  | 100 | | **stopOnInteraction** | Boolean | `false` | `autoPlay` (if activated) gets stopped, after the user interacted with the sled.  | 101 | | **rewind** | Boolean | `false` | Rewind sled, when you want to go beyond the first or last view.  | 102 | | **onSledEnd** | function | `null` | Callback, that gets triggered after last view. | 103 | | **onAnimationStart** | function | `null` | Callback, that gets triggered when a sliding-animation starts. | 104 | | **onAnimationEnd** | function | `null` | Callback, that gets triggered when a sliding-animation ends. | 105 | 106 | 107 | ## Controls 108 | 109 | There is only one control-component for **Arrows** and **Selecting-Dots**. 110 | The prop `select` decides what the Control-element is: A string called `next` or `prev` will activate Arrow-functionality, a number Select-functionality. 111 | 112 | You can easily style it via CSS. The default-styles are scoped to the class-name `sled-progress-default`. They are contained in the file `dist/index.css`. 113 | If you give it a custom `className`-prop, the default-class will be overridden and the Progress will be completely unstyled. Then you can copy the default-styles from **[here](https://raw.github.com/andreasfaust/react-sled/master/sled/control/index.css)** as a starting-point. 114 | 115 | 116 | **Control Props Overview:** 117 | 118 | | **Name** | **Type** | **Default** | **Description** | 119 | | :--------- | :--------------- | :------------------------ | :------------------------------------------------------------------------------------------------------------------------------------------------------------ | 120 | | **select** | String or Number | `'next'` | Defines, if the `Control` has arrow- or dot-functionality. A number is the index of the target-view. A string can be `'prev'` or `'next'` | 121 | | **className** | String | Default depends on `select` | | 122 | | **style** | String | `''` | If you provide a `style` and no `preset`, the default `preset` gets completely replaced. If you provide a `style` and a `preset`, the `preset` gets extended. | 123 | 124 | **Arrow:** 125 | Default-Design: 126 | ```jsx 127 | 130 | ``` 131 | 132 | Your Custom-Design: 133 | ```jsx 134 | 141 | My custom arrow! 142 | 143 | ``` 144 | 145 | **Selection-Dot:** 146 | ```jsx 147 | 150 | ``` 151 | 152 | ## Progress 153 | 154 | react-sled has an Instagram-like progress-bar. 155 | You can easily style it via CSS. 156 | The default-styles are scoped to the class-name `sled-progress-default`. 157 | If you give it a custom `className`-prop, the Progress will be completely unstyled. You can copy the default-styles from **[here](https://raw.github.com/andreasfaust/react-sled/master/sled/progress/index.css)** as a starting-point. 158 | 159 | 160 | **Progress Props Overview:** 161 | 162 | | **Name** | **Type** | **Default** | **Description** | 163 | | :--------- | :--------------- | :------------------------ | :------------------------------------------------------------------------------------------------------------------------------------------------------------ | 164 | | **className** | String | Default depends on `select` | | 165 | | **style** | String | `''` | If you provide a `style` and no `preset`, the default `preset` gets completely replaced. If you provide a `style` and a `preset`, the `preset` gets extended. | 166 | 167 | ```jsx 168 | 172 | ``` 173 | 174 | ## useSledStore 175 | 176 | A hook, that exposes the plugin’s state-management. 177 | `useSledStore` is only useable inside the `Sled`-Component. 178 | It returns an `Array` with 2 elements: 179 | 180 | 1. **state** of type `object` 181 | 2. **dispatch** of type `function` 182 | 183 | 184 | 185 | ## To-Do 186 | 187 | - [ ] Control animation by frame on drag 188 | - [ ] Infinity-Mode 189 | - [ ] Nice documentation with live examples (using Docz) 190 | - [ ] automated testing 191 | 192 | ## Contributing 193 | 194 | Every contribution is very much appreciated. 195 | 196 | **If you like react-sled, don't hesitate to star it on [GitHub](https://github.com/AndreasFaust/react-sled).** 197 | 198 | ## License 199 | 200 | MIT © [AndreasFaust](https://github.com/AndreasFaust) 201 | 202 | ## Thanks 203 | 204 | This library is derived from the great work and especially this [code-sandbox-example](https://codesandbox.io/embed/n9vo1my91p) provided by [Paul Henschel](https://github.com/drcmda) and the react-spring-team. 205 | -------------------------------------------------------------------------------- /components/SledComponent.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { Sled, Views, Progress, Control } from '../sled' 3 | import { useStateValue } from './state' 4 | import images from './images' 5 | 6 | const SledComponent = ({ children }) => { 7 | const [state] = useStateValue() 8 | return ( 9 |
10 | 11 | { 13 | console.log('THIS IS THE END') 14 | }} 15 | width={state.width} 16 | height={state.height} 17 | proportion={state.proportion} 18 | direction={state.direction} 19 | select={state.select} 20 | autoPlayInterval={state.autoPlayInterval} 21 | rewind={state.rewind} 22 | pauseOnMouseOver={state.pauseOnMouseOver} 23 | stopOnInteraction={state.stopOnInteraction} 24 | keyboard={state.keyboard} 25 | dragging={state.dragging} 26 | dragDistance={state.dragDistance} 27 | showElements={state.showElements} 28 | slideBy={state.slideBy} 29 | config={{ 30 | mass: state.mass, 31 | tension: state.tension, 32 | friction: state.friction, 33 | clamp: state.clamp 34 | }} 35 | onAnimationStart={() => { 36 | console.log('START Animation!') 37 | }} 38 | onAnimationEnd={() => { 39 | console.log('END Animation!') 40 | }} 41 | style={{ 42 | background: 'red' 43 | }} 44 | > 45 | {children} 46 | 47 | 48 |
49 | 50 | 51 |
52 |
53 | {images.map((image, index) => ( 54 | 55 | ))} 56 |
57 |
58 |
59 | ) 60 | } 61 | export default SledComponent 62 | -------------------------------------------------------------------------------- /components/SledImage.js: -------------------------------------------------------------------------------- 1 | import React, { useState, useEffect } from 'react' 2 | import { useSledStore } from '../sled' 3 | 4 | const defaultImage = '' 5 | 6 | const SledImage = ({ image, index }) => { 7 | const [{ currentIndex }] = useSledStore() 8 | const [src, setSrc] = useState(defaultImage) 9 | const [hasLoaded, setHasLoaded] = useState(false) 10 | 11 | useEffect(() => { 12 | if (image === src) return 13 | switch (index) { 14 | case currentIndex: 15 | case currentIndex - 1: 16 | case currentIndex + 1: 17 | setSrc(image) 18 | break 19 | default: 20 | } 21 | }, [currentIndex, image, index, src]) 22 | 23 | function onLoad() { 24 | if (src !== defaultImage) { 25 | setHasLoaded(true) 26 | } 27 | } 28 | 29 | return ( 30 |
45 |

{index}

46 | {'This 63 |
64 | ) 65 | } 66 | 67 | export default SledImage 68 | -------------------------------------------------------------------------------- /components/github-logo.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | export default () => ( 4 | 9 | 14 | 15 | 16 | ) 17 | -------------------------------------------------------------------------------- /components/header.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Github from './github-logo' 3 | 4 | export default () => ( 5 |
6 |

7 | 8 | 🛷 9 | 10 | {' '} 11 | react-sled 12 |

13 | 14 | 15 | 16 |
17 | ) -------------------------------------------------------------------------------- /components/images.js: -------------------------------------------------------------------------------- 1 | const images = [ 2 | 'https://source.unsplash.com/ANCoz0JMhiQ/1600x900', 3 | 'https://source.unsplash.com/uR6dIgDnt38/1600x900', 4 | 'https://source.unsplash.com/E2_k8SsuS7s/1600x900', 5 | 'https://source.unsplash.com/mGy1Jjr2e6M/1600x900', 6 | 'https://source.unsplash.com/TMHL7wald8I/1600x900', 7 | 'https://source.unsplash.com/-QKpblZde5I/1600x900', 8 | 'https://source.unsplash.com/o8cMgOUB-Z0/1600x900', 9 | 'https://source.unsplash.com/lzOzsGmAg3s/1600x900', 10 | 'https://source.unsplash.com/_7IUgAL60nc/1600x900', 11 | 'https://source.unsplash.com/7jwHx5q7WeA/1600x900', 12 | ] 13 | 14 | export default images -------------------------------------------------------------------------------- /components/settings.js: -------------------------------------------------------------------------------- 1 | import React, { useEffect, useState, useRef } from 'react' 2 | import Toggle from 'react-toggle' 3 | import debounce from 'lodash/debounce' 4 | import Slider from 'rc-slider' 5 | import 'rc-slider/assets/index.css' 6 | import { slide as Menu } from 'react-burger-menu' 7 | import { useStateValue } from './state' 8 | import useWindowSize from './useWindowSize' 9 | import Select from 'react-select' 10 | 11 | const customStyles = { 12 | 13 | container: () => ({ 14 | width: 150, 15 | position: 'relative', 16 | }), 17 | placeholder: () => ({ 18 | fontSize: '0.9rem' 19 | }), 20 | option: (provided) => ({ 21 | ...provided, 22 | fontSize: '0.9rem', 23 | }), 24 | 25 | } 26 | 27 | const useDebounce = (defaultValue) => { 28 | const [value, setValue] = useState(defaultValue) 29 | const dSetValue = useRef((event) => { 30 | event.persist() 31 | debounce(() => setValue(event.target.value), 150)() 32 | }) 33 | return [value, dSetValue] 34 | } 35 | 36 | const Wrapper = ({ children }) => { 37 | const { width } = useWindowSize() 38 | return width > 1024 39 | ?
{children}
40 | : {children} 41 | } 42 | 43 | const Settings = () => { 44 | const [state, dispatch] = useStateValue() 45 | 46 | const [proportion, setProportion] = React.useState(state.proportion) 47 | const [direction, setDirection] = React.useState(state.direction) 48 | 49 | const [width, setWidth] = useDebounce(state.width) 50 | const [height, setHeight] = useDebounce(state.height) 51 | const [autoPlayInterval, setAutoPlayInterval] = useDebounce(state.autoPlayInterval) 52 | const [select, setSelect] = useDebounce(state.select) 53 | 54 | 55 | useEffect(() => { 56 | dispatch({ type: 'width', value: width }) 57 | dispatch({ type: 'height', value: height }) 58 | dispatch({ type: 'direction', value: direction }) 59 | dispatch({ type: 'autoPlayInterval', value: +autoPlayInterval }) 60 | dispatch({ type: 'proportion', value: proportion }) 61 | }, [width, height, direction, autoPlayInterval, proportion]) 62 | 63 | useEffect(() => { 64 | if (parseInt(select, 10)) { 65 | dispatch({ type: 'select', value: parseInt(select, 10) }) 66 | } 67 | if (select === 'prev' || select === 'next') { 68 | dispatch({ type: 'select', value: select }) 69 | } 70 | }, [select]) 71 | 72 | return ( 73 | 74 |

78 | Settings: 79 |

80 |
81 | 91 | 102 | 133 | 134 | 144 | 154 | 164 | 174 | 184 |
185 | 186 |
187 | 195 | 196 | 204 | 212 | 213 | 221 | 229 | 230 |
231 | 232 |
233 | 236 | 244 | 252 | 260 | 268 |
269 |
270 | ) 271 | } 272 | 273 | export default Settings 274 | -------------------------------------------------------------------------------- /components/state.js: -------------------------------------------------------------------------------- 1 | import React, { createContext, useContext, useReducer } from 'react' 2 | export const StateContext = createContext() 3 | 4 | const initialState = { 5 | select: 3, 6 | rewind: true, 7 | direction: 'horizontal', 8 | showElements: 1, 9 | slideBy: 1, 10 | pauseOnMouseOver: true, 11 | stopOnInteraction: true, 12 | keyboard: true, 13 | dragging: true, 14 | dragDistance: 100, 15 | autoPlayInterval: 3000, 16 | width: '100%', 17 | proportion: '2:1', 18 | clamp: true, 19 | mass: 1, 20 | tension: 170, 21 | friction: 26 22 | } 23 | 24 | const reducer = (state, action) => { 25 | switch (action.type) { 26 | case 'select': return { ...state, select: action.value } 27 | case 'proportion': return { ...state, proportion: action.value } 28 | case 'direction': return { ...state, direction: action.value } 29 | case 'rewind': return { ...state, rewind: action.value } 30 | case 'pauseOnMouseOver': return { ...state, pauseOnMouseOver: action.value } 31 | case 'stopOnInteraction': return { ...state, stopOnInteraction: action.value } 32 | case 'keyboard': return { ...state, keyboard: action.value } 33 | case 'dragging': return { ...state, dragging: action.value } 34 | case 'dragDistance': return { ...state, dragDistance: action.value } 35 | case 'autoPlayInterval': return { ...state, autoPlayInterval: action.value } 36 | case 'width': return { ...state, width: action.value } 37 | case 'height': return { ...state, height: action.value } 38 | case 'clamp': return { ...state, clamp: action.value } 39 | case 'mass': return { ...state, mass: action.value } 40 | case 'tension': return { ...state, tension: action.value } 41 | case 'friction': return { ...state, friction: action.value } 42 | case 'showElements': return { ...state, showElements: action.value } 43 | case 'slideBy': return { ...state, slideBy: action.value } 44 | default: return state 45 | } 46 | } 47 | 48 | export const StateProvider = ({ children }) => ( 49 | 50 | {children} 51 | 52 | ) 53 | 54 | export const useStateValue = () => useContext(StateContext) 55 | -------------------------------------------------------------------------------- /components/useWindowSize.js: -------------------------------------------------------------------------------- 1 | import { useEffect, useState } from 'react' 2 | 3 | export default function useWindowSize() { 4 | function getSize() { 5 | return { 6 | width: window.innerWidth, 7 | height: window.innerHeight 8 | } 9 | } 10 | 11 | const [windowSize, setWindowSize] = useState({ width: undefined, height: undefined }) 12 | 13 | useEffect(() => { 14 | setWindowSize(getSize()) 15 | 16 | function handleResize() { 17 | setWindowSize(getSize()) 18 | } 19 | 20 | window.addEventListener('resize', handleResize) 21 | return () => window.removeEventListener('resize', handleResize) 22 | }, []) 23 | 24 | return windowSize 25 | } 26 | -------------------------------------------------------------------------------- /next-env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | /// 3 | -------------------------------------------------------------------------------- /next.config.js: -------------------------------------------------------------------------------- 1 | const withCSS = require('@zeit/next-css') 2 | 3 | module.exports = withCSS({}) 4 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "react-sled", 3 | "version": "2.0.4", 4 | "description": "react-sled is a carousel made with react-spring.", 5 | "author": "AndreasFaust", 6 | "license": "MIT", 7 | "repository": "AndreasFaust/react-sled", 8 | "homepage": "https://react-sled.andreasfaust.com/", 9 | "files": [ 10 | "/dist" 11 | ], 12 | "main": "./dist/index.js", 13 | "module": "./dist/index.es.js", 14 | "jsnext:main": "./dist/index.es.js", 15 | "types": "./dist/index.d.ts", 16 | "engines": { 17 | "node": ">=8", 18 | "npm": ">=5" 19 | }, 20 | "dependencies": {}, 21 | "devDependencies": { 22 | "@rollup/plugin-commonjs": "^11.0.1", 23 | "@rollup/plugin-node-resolve": "^6.1.0", 24 | "@rollup/plugin-url": "^4.0.0", 25 | "@types/node": "^13.1.4", 26 | "@types/react": "^16.9.17", 27 | "@typescript-eslint/eslint-plugin": "2.14.0", 28 | "@typescript-eslint/parser": "2.14.0", 29 | "@zeit/next-css": "^1.0.1", 30 | "babel-eslint": "^10.0.3", 31 | "csstype": "^2.6.8", 32 | "eslint": "6.8.0", 33 | "eslint-config-react-app": "^5.1.0", 34 | "eslint-plugin-flowtype": "4.5.3", 35 | "eslint-plugin-import": "2.19.1", 36 | "eslint-plugin-jsx-a11y": "6.x", 37 | "eslint-plugin-react": "7.17.0", 38 | "eslint-plugin-react-hooks": "2.3.0", 39 | "generate-changelog": "^1.8.0", 40 | "gh-pages": "^2.1.1", 41 | "lodash": "^4.17.15", 42 | "next": "9.1.7", 43 | "prop-types": "^15.7.2", 44 | "rc-slider": "^8.7.1", 45 | "react": "16.12.0", 46 | "react-burger-menu": "^2.6.13", 47 | "react-dom": "^16.12.0", 48 | "react-select": "^3.0.8", 49 | "react-spring": "^9.0.0-beta.34", 50 | "react-toggle": "^4.1.1", 51 | "react-use-gesture": "^7.0.1", 52 | "rollup": "^1.28.0", 53 | "rollup-plugin-css-only": "^2.0.0", 54 | "rollup-plugin-peer-deps-external": "^2.2.0", 55 | "rollup-plugin-typescript2": "^0.25.3", 56 | "typescript": "^3.7.4" 57 | }, 58 | "peerDependencies": { 59 | "react": "^16.8.5", 60 | "react-dom": "^16.8.5", 61 | "react-spring": "^9.0.0-beta.34", 62 | "react-use-gesture": "^7.0.0" 63 | }, 64 | "keywords": [ 65 | "React", 66 | "Viewpager", 67 | "Carousel", 68 | "Gallery", 69 | "Slideshow", 70 | "Slider", 71 | "Spring", 72 | "Animation", 73 | "react-spring", 74 | "react-use-gesture" 75 | ], 76 | "scripts": { 77 | "dev": "next dev", 78 | "build": "rollup -c", 79 | "release:major": "yarn build && git add . && changelog -M && git add CHANGELOG.md && git commit -m 'updated CHANGELOG.md' && npm version major && git push origin && git push origin --tags && npm publish", 80 | "release:minor": "yarn build && git add . && changelog -m && git add CHANGELOG.md && git commit -m 'updated CHANGELOG.md' && npm version minor && git push origin && git push origin --tags && npm publish", 81 | "release:patch": "yarn build && git add . && changelog -p && git add CHANGELOG.md && git commit -m 'updated CHANGELOG.md' && npm version patch && git push origin && git push origin --tags && npm publish", 82 | "deploy-files": "rsync -vrtu --delete -e ssh ./out/ box@finlay.uberspace.de:/var/www/virtual/box/html/react-sled.andreasfaust.com", 83 | "deploy": "next build && next export && yarn deploy-files" 84 | } 85 | } 86 | -------------------------------------------------------------------------------- /pages/css/header.css: -------------------------------------------------------------------------------- 1 | .header { 2 | position: fixed; 3 | left: 0; 4 | right: 0; 5 | top: 0; 6 | z-index: 1000; 7 | display: flex; 8 | display: flex; 9 | height: 4rem; 10 | align-items: center; 11 | justify-content: center; 12 | background: #fff; 13 | box-shadow: 0 0 1rem hsla(0, 0%, 0%, 0.1); 14 | } 15 | 16 | h1 { 17 | /* font-family: plex-serif; */ 18 | font-size: 1.25rem; 19 | letter-spacing: -0.01rem; 20 | padding: 0 1rem; 21 | display: flex; 22 | align-items: center; 23 | position: relative; 24 | /* text-align: center; */ 25 | } 26 | 27 | h1 span { 28 | font-size: 2em; 29 | margin: 0 0.5rem 0 0; 30 | } 31 | 32 | .github { 33 | position: absolute; 34 | display: block; 35 | width: 2.25rem; 36 | height: 2.25rem; 37 | right: 0.5rem; 38 | } 39 | 40 | .github svg { 41 | width: 100%; 42 | } 43 | 44 | .github:hover path, 45 | .github:active path { 46 | fill: #cc1d1c; 47 | } 48 | 49 | 50 | @media (min-width: 1025px) { 51 | h1 { 52 | font-size: 2rem; 53 | } 54 | h1 span { 55 | margin: 0 1rem 0 0; 56 | } 57 | .github { 58 | position: absolute; 59 | display: block; 60 | width: 2.75rem; 61 | height: 2.75rem; 62 | right: 0.5rem; 63 | } 64 | } 65 | -------------------------------------------------------------------------------- /pages/css/index.css: -------------------------------------------------------------------------------- 1 | @font-face { 2 | font-family: 'ia'; 3 | src: url('/plex-mono.woff') format('woff'); 4 | } 5 | @font-face { 6 | font-family: 'ia-bold'; 7 | src: url('/plex-mono-bold.woff') format('woff'); 8 | } 9 | 10 | .bold { 11 | font-family: 'ia-bold'; 12 | } 13 | 14 | body { 15 | font-family: 'ia'; 16 | margin: 0; 17 | padding: 0; 18 | overflow-x: hidden; 19 | } 20 | 21 | .wrapper { 22 | } 23 | 24 | .content { 25 | margin: 8rem 0; 26 | width: 100%; 27 | } 28 | @media (min-width: 1025px) { 29 | .content { 30 | width: calc(100% - 300px); 31 | margin-left: 300px; 32 | } 33 | } 34 | 35 | .sled-wrapper { 36 | flex-grow: 1; 37 | padding: 0 1rem; 38 | } 39 | @media (min-width: 1025px) { 40 | .sled-wrapper { 41 | padding: 0 2rem; 42 | } 43 | } 44 | 45 | .controls { 46 | display: flex; 47 | justify-content: center; 48 | width: 100%; 49 | } 50 | .controls > button { 51 | margin: 10px; 52 | } 53 | 54 | .testContent { 55 | position: absolute; 56 | left: 0; 57 | right: 0; 58 | top: 0; 59 | bottom: 0; 60 | display: flex; 61 | align-items: center; 62 | justify-content: center; 63 | font-size: 4rem; 64 | color: #fff; 65 | } -------------------------------------------------------------------------------- /pages/css/menue.css: -------------------------------------------------------------------------------- 1 | /* Position and sizing of burger button */ 2 | .bm-burger-button { 3 | position: fixed; 4 | width: 2rem; 5 | height: 1rem; 6 | left: 1rem; 7 | top: 1.5rem; 8 | } 9 | 10 | /* Color/shape of burger icon bars */ 11 | .bm-burger-bars { 12 | background: #373a47; 13 | } 14 | 15 | /* Color/shape of burger icon bars on hover*/ 16 | .bm-burger-bars-hover { 17 | background: #a90000; 18 | } 19 | 20 | /* Position and sizing of clickable cross button */ 21 | .bm-cross-button { 22 | height: 24px; 23 | width: 24px; 24 | } 25 | 26 | /* Color/shape of close button cross */ 27 | .bm-cross { 28 | background: #bdc3c7; 29 | } 30 | 31 | .bm-overlay { 32 | top: 0; 33 | } -------------------------------------------------------------------------------- /pages/css/reset.css: -------------------------------------------------------------------------------- 1 | html { 2 | font-family: sans-serif; 3 | -ms-text-size-adjust: 100%; 4 | -webkit-text-size-adjust: 100%; 5 | } 6 | body { 7 | margin: 0; 8 | } 9 | article, 10 | aside, 11 | details, 12 | figcaption, 13 | figure, 14 | footer, 15 | header, 16 | main, 17 | menu, 18 | nav, 19 | section, 20 | summary { 21 | display: block; 22 | } 23 | audio, 24 | canvas, 25 | progress, 26 | video { 27 | display: inline-block; 28 | } 29 | audio:not([controls]) { 30 | display: none; 31 | height: 0; 32 | } 33 | progress { 34 | vertical-align: baseline; 35 | } 36 | [hidden], 37 | template { 38 | display: none; 39 | } 40 | a { 41 | background-color: transparent; 42 | -webkit-text-decoration-skip: objects; 43 | color: $greyText; 44 | text-decoration: none; 45 | } 46 | a:active, 47 | a:hover { 48 | outline-width: 0; 49 | } 50 | abbr[title] { 51 | border-bottom: none; 52 | text-decoration: underline; 53 | text-decoration: underline dotted; 54 | } 55 | b, 56 | strong { 57 | font-weight: normal; 58 | } 59 | dfn { 60 | font-style: normal; 61 | } 62 | mark { 63 | background-color: #ff0; 64 | color: #000; 65 | } 66 | small { 67 | } 68 | sub, 69 | sup { 70 | font-size: 75%; 71 | line-height: 0; 72 | position: relative; 73 | vertical-align: baseline; 74 | } 75 | sub { 76 | bottom: -0.25em; 77 | } 78 | sup { 79 | top: -0.5em; 80 | } 81 | img { 82 | border-style: none; 83 | } 84 | svg:not(:root) { 85 | overflow: hidden; 86 | } 87 | code, 88 | kbd, 89 | pre, 90 | samp { 91 | font-family: monospace, monospace; 92 | font-size: 1em; 93 | } 94 | figure { 95 | /* // margin: 1em 40px; */ 96 | } 97 | hr { 98 | box-sizing: content-box; 99 | height: 0; 100 | overflow: visible; 101 | } 102 | button, 103 | input, 104 | optgroup, 105 | select, 106 | textarea { 107 | font: inherit; 108 | margin: 0; 109 | } 110 | optgroup { 111 | font-weight: 700; 112 | } 113 | button, 114 | input { 115 | overflow: visible; 116 | } 117 | button, 118 | select { 119 | text-transform: none; 120 | } 121 | [type="reset"], 122 | [type="submit"], 123 | button, 124 | html [type="button"] { 125 | -webkit-appearance: button; 126 | } 127 | [type="button"]::-moz-focus-inner, 128 | [type="reset"]::-moz-focus-inner, 129 | [type="submit"]::-moz-focus-inner, 130 | button::-moz-focus-inner { 131 | border-style: none; 132 | padding: 0; 133 | } 134 | [type="button"]:-moz-focusring, 135 | [type="reset"]:-moz-focusring, 136 | [type="submit"]:-moz-focusring, 137 | button:-moz-focusring { 138 | outline: 1px dotted ButtonText; 139 | } 140 | fieldset { 141 | border: 1px solid silver; 142 | margin: 0 2px; 143 | padding: 0.35em 0.625em 0.75em; 144 | } 145 | legend { 146 | box-sizing: border-box; 147 | color: inherit; 148 | display: table; 149 | max-width: 100%; 150 | padding: 0; 151 | white-space: normal; 152 | } 153 | textarea { 154 | overflow: auto; 155 | } 156 | [type="checkbox"], 157 | [type="radio"] { 158 | box-sizing: border-box; 159 | padding: 0; 160 | } 161 | [type="number"]::-webkit-inner-spin-button, 162 | [type="number"]::-webkit-outer-spin-button { 163 | height: auto; 164 | } 165 | [type="search"] { 166 | -webkit-appearance: textfield; 167 | outline-offset: -2px; 168 | } 169 | [type="search"]::-webkit-search-cancel-button, 170 | [type="search"]::-webkit-search-decoration { 171 | -webkit-appearance: none; 172 | } 173 | ::-webkit-input-placeholder { 174 | color: inherit; 175 | opacity: 0.54; 176 | } 177 | ::-webkit-file-upload-button { 178 | -webkit-appearance: button; 179 | font: inherit; 180 | } 181 | html { 182 | box-sizing: border-box; 183 | } 184 | * { 185 | box-sizing: inherit; 186 | } 187 | *:before { 188 | box-sizing: inherit; 189 | } 190 | *:after { 191 | box-sizing: inherit; 192 | } 193 | body { 194 | color: $greyText; 195 | font-weight: normal; 196 | word-wrap: break-word; 197 | font-kerning: normal; 198 | -moz-font-feature-settings: "kern", "liga", "clig", "calt"; 199 | -ms-font-feature-settings: "kern", "liga", "clig", "calt"; 200 | -webkit-font-feature-settings: "kern", "liga", "clig", "calt"; 201 | font-feature-settings: "kern", "liga", "clig", "calt"; 202 | } 203 | img { 204 | max-width: 100%; 205 | margin-left: 0; 206 | margin-right: 0; 207 | margin-top: 0; 208 | margin-bottom: 0; 209 | padding-bottom: 0; 210 | padding-left: 0; 211 | padding-right: 0; 212 | padding-top: 0; 213 | } 214 | h1 { 215 | margin-left: 0; 216 | margin-right: 0; 217 | margin-top: 0; 218 | margin-bottom: 0; 219 | padding-bottom: 0; 220 | padding-left: 0; 221 | padding-right: 0; 222 | padding-top: 0; 223 | color: inherit; 224 | font-weight: normal; 225 | text-rendering: optimizeLegibility; 226 | } 227 | h2 { 228 | margin-left: 0; 229 | margin-right: 0; 230 | margin-top: 0; 231 | margin-bottom: 0; 232 | padding-bottom: 0; 233 | padding-left: 0; 234 | padding-right: 0; 235 | padding-top: 0; 236 | color: inherit; 237 | font-weight: normal; 238 | text-rendering: optimizeLegibility; 239 | } 240 | h3 { 241 | margin-left: 0; 242 | margin-right: 0; 243 | margin-top: 0; 244 | margin-bottom: 0; 245 | padding-bottom: 0; 246 | padding-left: 0; 247 | padding-right: 0; 248 | padding-top: 0; 249 | color: inherit; 250 | font-weight: normal; 251 | text-rendering: optimizeLegibility; 252 | } 253 | h4 { 254 | margin-left: 0; 255 | margin-right: 0; 256 | margin-top: 0; 257 | margin-bottom: 0; 258 | padding-bottom: 0; 259 | padding-left: 0; 260 | padding-right: 0; 261 | padding-top: 0; 262 | color: inherit; 263 | font-weight: normal; 264 | text-rendering: optimizeLegibility; 265 | } 266 | h5 { 267 | margin-left: 0; 268 | margin-right: 0; 269 | margin-top: 0; 270 | margin-bottom: 0; 271 | padding-bottom: 0; 272 | padding-left: 0; 273 | padding-right: 0; 274 | padding-top: 0; 275 | color: inherit; 276 | font-weight: normal; 277 | text-rendering: optimizeLegibility; 278 | } 279 | h6 { 280 | margin-left: 0; 281 | margin-right: 0; 282 | margin-top: 0; 283 | margin-bottom: 0; 284 | padding-bottom: 0; 285 | padding-left: 0; 286 | padding-right: 0; 287 | padding-top: 0; 288 | color: inherit; 289 | font-weight: normal; 290 | text-rendering: optimizeLegibility; 291 | } 292 | hgroup { 293 | margin-left: 0; 294 | margin-right: 0; 295 | margin-top: 0; 296 | padding-bottom: 0; 297 | padding-left: 0; 298 | padding-right: 0; 299 | padding-top: 0; 300 | margin-bottom: 1.45rem; 301 | } 302 | ul { 303 | margin-left: 0; 304 | margin-right: 0; 305 | margin-top: 0; 306 | margin-bottom: 0; 307 | padding-bottom: 0; 308 | padding-left: 0; 309 | padding-right: 0; 310 | padding-top: 0; 311 | list-style: none; 312 | list-style-image: none; 313 | } 314 | ol { 315 | margin-left: 0; 316 | margin-right: 0; 317 | margin-top: 0; 318 | margin-bottom: 0; 319 | padding-bottom: 0; 320 | padding-left: 0; 321 | padding-right: 0; 322 | padding-top: 0; 323 | list-style-position: outside; 324 | list-style-image: none; 325 | } 326 | dl { 327 | margin-left: 0; 328 | margin-right: 0; 329 | margin-top: 0; 330 | padding-bottom: 0; 331 | padding-left: 0; 332 | padding-right: 0; 333 | padding-top: 0; 334 | margin-bottom: 0; 335 | } 336 | dd { 337 | margin-left: 0; 338 | margin-right: 0; 339 | margin-top: 0; 340 | padding-bottom: 0; 341 | padding-left: 0; 342 | padding-right: 0; 343 | padding-top: 0; 344 | margin-bottom: 0; 345 | } 346 | p { 347 | margin-left: 0; 348 | margin-right: 0; 349 | margin-top: 0; 350 | padding-bottom: 0; 351 | padding-left: 0; 352 | padding-right: 0; 353 | padding-top: 0; 354 | margin-bottom: 0; 355 | } 356 | figure { 357 | margin-left: 0; 358 | margin-right: 0; 359 | margin-top: 0; 360 | padding-bottom: 0; 361 | padding-left: 0; 362 | padding-right: 0; 363 | padding-top: 0; 364 | margin-bottom: 0; 365 | } 366 | pre { 367 | margin-left: 0; 368 | margin-right: 0; 369 | margin-top: 0; 370 | padding-bottom: 0; 371 | padding-left: 0; 372 | padding-right: 0; 373 | padding-top: 0; 374 | margin-bottom: 0; 375 | font-size: 0.85rem; 376 | line-height: 1.42; 377 | background: hsla(0, 0%, 0%, 0.04); 378 | border-radius: 3px; 379 | overflow: auto; 380 | word-wrap: normal; 381 | padding: 0; 382 | } 383 | table { 384 | margin-left: 0; 385 | margin-right: 0; 386 | margin-top: 0; 387 | padding-bottom: 0; 388 | padding-left: 0; 389 | padding-right: 0; 390 | padding-top: 0; 391 | margin-bottom: 0; 392 | font-size: 1rem; 393 | line-height: 1; 394 | border-collapse: collapse; 395 | width: 100%; 396 | } 397 | fieldset { 398 | margin-left: 0; 399 | margin-right: 0; 400 | margin-top: 0; 401 | padding-bottom: 0; 402 | padding-left: 0; 403 | padding-right: 0; 404 | padding-top: 0; 405 | margin-bottom: 0; 406 | } 407 | blockquote { 408 | margin-left: 0; 409 | margin-right: 0; 410 | margin-top: 0; 411 | padding-bottom: 0; 412 | padding-left: 0; 413 | padding-right: 0; 414 | padding-top: 0; 415 | margin-bottom: 0; 416 | } 417 | form { 418 | margin-left: 0; 419 | margin-right: 0; 420 | margin-top: 0; 421 | padding-bottom: 0; 422 | padding-left: 0; 423 | padding-right: 0; 424 | padding-top: 0; 425 | margin-bottom: 0; 426 | } 427 | noscript { 428 | margin-left: 0; 429 | margin-right: 0; 430 | margin-top: 0; 431 | padding-bottom: 0; 432 | padding-left: 0; 433 | padding-right: 0; 434 | padding-top: 0; 435 | margin-bottom: 0; 436 | } 437 | iframe { 438 | margin-left: 0; 439 | margin-right: 0; 440 | margin-top: 0; 441 | padding-bottom: 0; 442 | padding-left: 0; 443 | padding-right: 0; 444 | padding-top: 0; 445 | margin-bottom: 0; 446 | } 447 | hr { 448 | margin-left: 0; 449 | margin-right: 0; 450 | margin-top: 0; 451 | padding-bottom: 0; 452 | padding-left: 0; 453 | padding-right: 0; 454 | padding-top: 0; 455 | margin-bottom: 0; 456 | background: hsla(0, 0%, 0%, 0.2); 457 | border: none; 458 | height: 1px; 459 | } 460 | address { 461 | margin-left: 0; 462 | margin-right: 0; 463 | margin-top: 0; 464 | padding-bottom: 0; 465 | padding-left: 0; 466 | padding-right: 0; 467 | padding-top: 0; 468 | margin-bottom: 0; 469 | } 470 | b { 471 | font-weight: normal; 472 | } 473 | strong { 474 | font-weight: normal; 475 | } 476 | dt { 477 | font-weight: normal; 478 | } 479 | th { 480 | font-weight: normal; 481 | } 482 | li { 483 | margin-bottom: 0; 484 | } 485 | ol li { 486 | padding-left: 0; 487 | } 488 | ul li { 489 | padding-left: 0; 490 | } 491 | li > ol { 492 | } 493 | li > ul { 494 | } 495 | blockquote *:last-child { 496 | } 497 | li *:last-child { 498 | } 499 | p *:last-child { 500 | } 501 | li > p { 502 | } 503 | code { 504 | font-size: 0.85rem; 505 | line-height: 1.45rem; 506 | } 507 | kbd { 508 | font-size: 0.85rem; 509 | line-height: 1.45rem; 510 | } 511 | samp { 512 | font-size: 0.85rem; 513 | line-height: 1.45rem; 514 | } 515 | abbr { 516 | border-bottom: 1px dotted hsla(0, 0%, 0%, 0.5); 517 | cursor: help; 518 | } 519 | acronym { 520 | border-bottom: 1px dotted hsla(0, 0%, 0%, 0.5); 521 | cursor: help; 522 | } 523 | abbr[title] { 524 | border-bottom: 1px dotted hsla(0, 0%, 0%, 0.5); 525 | cursor: help; 526 | text-decoration: none; 527 | } 528 | thead { 529 | text-align: left; 530 | } 531 | td, 532 | th { 533 | text-align: left; 534 | border-bottom: 1px solid hsla(0, 0%, 0%, 0.12); 535 | font-feature-settings: "tnum"; 536 | -moz-font-feature-settings: "tnum"; 537 | -ms-font-feature-settings: "tnum"; 538 | -webkit-font-feature-settings: "tnum"; 539 | padding-left: 0.96667rem; 540 | padding-right: 0.96667rem; 541 | padding-top: 0.725rem; 542 | padding-bottom: calc(0.725rem - 1px); 543 | } 544 | th:first-child, 545 | td:first-child { 546 | padding-left: 0; 547 | } 548 | th:last-child, 549 | td:last-child { 550 | padding-right: 0; 551 | } 552 | tt, 553 | code { 554 | background-color: hsla(0, 0%, 0%, 0.04); 555 | border-radius: 3px; 556 | font-family: "SFMono-Regular", Consolas, "Roboto Mono", "Droid Sans Mono", 557 | "Liberation Mono", Menlo, Courier, monospace; 558 | padding: 0; 559 | padding-top: 0.2em; 560 | padding-bottom: 0.2em; 561 | } 562 | pre code { 563 | background: none; 564 | line-height: 1.42; 565 | } 566 | code:before, 567 | code:after, 568 | tt:before, 569 | tt:after { 570 | letter-spacing: -0.2em; 571 | content: " "; 572 | } 573 | pre code:before, 574 | pre code:after, 575 | pre tt:before, 576 | pre tt:after { 577 | content: ""; 578 | } 579 | @media only screen and (max-width: 480px) { 580 | html { 581 | font-size: 100%; 582 | } 583 | } 584 | -------------------------------------------------------------------------------- /pages/css/settings.css: -------------------------------------------------------------------------------- 1 | .settings { 2 | position: fixed; 3 | width: 300px; 4 | z-index: 100; 5 | background: #fff; 6 | box-shadow: 0 0 1rem hsla(0, 0%, 0%, 0.1); 7 | top: 0; 8 | bottom: 0; 9 | padding: 1rem 0; 10 | overflow-y: scroll; 11 | -webkit-overflow-scrolling: touch; 12 | } 13 | 14 | @media (min-width: 1025px) { 15 | .settings { 16 | top: 4rem; 17 | } 18 | } 19 | 20 | .bm-item-list { 21 | padding: 1rem 0 8rem; 22 | } 23 | 24 | .settings__column--3 { 25 | margin-top: 1rem; 26 | } 27 | 28 | 29 | 30 | .background { 31 | } 32 | 33 | 34 | h2 { 35 | 36 | } 37 | 38 | .settings__h2 { 39 | font-size: 0.9rem; 40 | padding: 0.5rem 1rem; 41 | height: 3.5rem; 42 | user-select: none; 43 | display: flex; 44 | align-items: center; 45 | } 46 | .settings__h2:focus { 47 | outline: none; 48 | box-shadow: none; 49 | } 50 | 51 | .settings__h3 { 52 | font-weight: normal; 53 | font-size: 0.9rem; 54 | margin: 0 10px 0 0; 55 | min-width: 150px; 56 | flex-grow: 1; 57 | } 58 | 59 | .settings__label { 60 | display: flex; 61 | align-items: center; 62 | padding: 0.5rem 1rem 0.5rem 0; 63 | margin-left: 2rem; 64 | height: 2.5rem; 65 | border-top: 1px solid #ebebeb; 66 | } 67 | 68 | .settings__label--config .settings__h3 { 69 | min-width: 120px; 70 | } 71 | 72 | .settings__label--disabled > * { 73 | pointer-events: none; 74 | opacity: 0.5; 75 | } 76 | .settings__label--select .settings__h3 { 77 | min-width: auto; 78 | } 79 | 80 | .react-toggle { 81 | margin-left: 1rem; 82 | } 83 | 84 | .settings__input { 85 | font-family: 'ia'; 86 | width: 4rem; 87 | font-size: 0.9rem; 88 | display: flex; 89 | align-items: center; 90 | justify-content: center; 91 | padding: 0.5rem; 92 | margin-left: 0.5rem; 93 | border-radius: 0.25rem; 94 | border: none; 95 | background: #efefef; 96 | box-shadow: none; 97 | } 98 | 99 | .rc-slider-handle { 100 | border-color: red !important; 101 | } 102 | .rc-slider-rail { 103 | background-color: #ccc !important; 104 | } 105 | .rc-slider-track { 106 | background-color: red !important; 107 | } 108 | -------------------------------------------------------------------------------- /pages/css/toggle.css: -------------------------------------------------------------------------------- 1 | .react-toggle { 2 | touch-action: pan-x; 3 | 4 | display: inline-block; 5 | position: relative; 6 | cursor: pointer; 7 | background-color: transparent; 8 | border: 0; 9 | padding: 0; 10 | 11 | -webkit-touch-callout: none; 12 | -webkit-user-select: none; 13 | -khtml-user-select: none; 14 | -moz-user-select: none; 15 | -ms-user-select: none; 16 | user-select: none; 17 | 18 | -webkit-tap-highlight-color: rgba(0, 0, 0, 0); 19 | -webkit-tap-highlight-color: transparent; 20 | } 21 | 22 | .react-toggle-screenreader-only { 23 | border: 0; 24 | clip: rect(0 0 0 0); 25 | height: 1px; 26 | margin: -1px; 27 | overflow: hidden; 28 | padding: 0; 29 | position: absolute; 30 | width: 1px; 31 | } 32 | 33 | .react-toggle--disabled { 34 | cursor: not-allowed; 35 | opacity: 0.5; 36 | -webkit-transition: opacity 0.25s; 37 | transition: opacity 0.25s; 38 | } 39 | 40 | .react-toggle-track { 41 | width: 50px; 42 | height: 24px; 43 | padding: 0; 44 | border-radius: 30px; 45 | background-color: #4d4d4d; 46 | -webkit-transition: all 0.2s ease; 47 | -moz-transition: all 0.2s ease; 48 | transition: all 0.2s ease; 49 | } 50 | 51 | .react-toggle:hover:not(.react-toggle--disabled) .react-toggle-track { 52 | background-color: #000000; 53 | } 54 | 55 | .react-toggle--checked .react-toggle-track { 56 | background-color: #19ab27; 57 | } 58 | 59 | .react-toggle--checked:hover:not(.react-toggle--disabled) .react-toggle-track { 60 | background-color: #128d15; 61 | } 62 | 63 | .react-toggle-track-check { 64 | position: absolute; 65 | width: 14px; 66 | height: 10px; 67 | top: 0px; 68 | bottom: 0px; 69 | margin-top: auto; 70 | margin-bottom: auto; 71 | line-height: 0; 72 | left: 8px; 73 | opacity: 0; 74 | -webkit-transition: opacity 0.25s ease; 75 | -moz-transition: opacity 0.25s ease; 76 | transition: opacity 0.25s ease; 77 | } 78 | 79 | .react-toggle--checked .react-toggle-track-check { 80 | opacity: 1; 81 | -webkit-transition: opacity 0.25s ease; 82 | -moz-transition: opacity 0.25s ease; 83 | transition: opacity 0.25s ease; 84 | } 85 | 86 | .react-toggle-track-x { 87 | position: absolute; 88 | width: 10px; 89 | height: 10px; 90 | top: 0px; 91 | bottom: 0px; 92 | margin-top: auto; 93 | margin-bottom: auto; 94 | line-height: 0; 95 | right: 10px; 96 | opacity: 1; 97 | -webkit-transition: opacity 0.25s ease; 98 | -moz-transition: opacity 0.25s ease; 99 | transition: opacity 0.25s ease; 100 | } 101 | 102 | .react-toggle--checked .react-toggle-track-x { 103 | opacity: 0; 104 | } 105 | 106 | .react-toggle-thumb { 107 | transition: all 0.5s cubic-bezier(0.23, 1, 0.32, 1) 0ms; 108 | position: absolute; 109 | top: 1px; 110 | left: 1px; 111 | width: 22px; 112 | height: 22px; 113 | border: 1px solid #4d4d4d; 114 | border-radius: 50%; 115 | background-color: #fafafa; 116 | 117 | -webkit-box-sizing: border-box; 118 | -moz-box-sizing: border-box; 119 | box-sizing: border-box; 120 | 121 | -webkit-transition: all 0.25s ease; 122 | -moz-transition: all 0.25s ease; 123 | transition: all 0.25s ease; 124 | } 125 | 126 | .react-toggle--checked .react-toggle-thumb { 127 | left: 27px; 128 | border-color: #19ab27; 129 | } 130 | 131 | .react-toggle--focus .react-toggle-thumb { 132 | -webkit-box-shadow: 0px 0px 3px 2px #0099e0; 133 | -moz-box-shadow: 0px 0px 3px 2px #0099e0; 134 | box-shadow: 0px 0px 2px 3px #0099e0; 135 | } 136 | 137 | .react-toggle:active:not(.react-toggle--disabled) .react-toggle-thumb { 138 | -webkit-box-shadow: 0px 0px 5px 5px #0099e0; 139 | -moz-box-shadow: 0px 0px 5px 5px #0099e0; 140 | box-shadow: 0px 0px 5px 5px #0099e0; 141 | } 142 | -------------------------------------------------------------------------------- /pages/custom.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | const NotFoundPage = () => ( 4 |
5 |

I AM CUSTOM 404

6 |

You just hit a route that doesn't exist... the sadness.

7 |
8 | ) 9 | 10 | export default NotFoundPage 11 | -------------------------------------------------------------------------------- /pages/index.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './css/reset.css' 3 | import './css/toggle.css' 4 | import './css/index.css' 5 | import './css/header.css' 6 | import './css/settings.css' 7 | import './css/menue.css' 8 | import { StateProvider } from '../components/state' 9 | import SledComponent from '../components/SledComponent' 10 | import Header from '../components/header' 11 | import Settings from '../components/settings' 12 | import SledImage from '../components/SledImage' 13 | 14 | import images from '../components/images' 15 | 16 | const App = () => { 17 | return ( 18 | 19 |
20 | 21 |
22 | 23 | {images.map((image, index) => ( 24 | 29 | ))} 30 | 31 |
32 | 33 | ) 34 | } 35 | 36 | export default App 37 | -------------------------------------------------------------------------------- /pages/test-basic.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import images from '../components/images' 3 | import { Sled, Views, Progress, Control } from '../sled' 4 | 5 | const App = () => { 6 | return ( 7 |
8 | 9 | 13 | {images.map(image => ( 14 | My Image 19 | ))} 20 | 21 | 22 |
23 | 24 | 25 |
26 |
27 | {images.map((image, index) => ( 28 | 29 | ))} 30 |
31 |
32 | 38 |
39 | ) 40 | } 41 | 42 | export default App 43 | -------------------------------------------------------------------------------- /pages/test.js: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './css/reset.css' 3 | import './css/toggle.css' 4 | import './css/index.css' 5 | import './css/header.css' 6 | import './css/settings.css' 7 | import './css/menue.css' 8 | import { StateProvider } from '../components/state' 9 | import SledComponent from '../components/SledComponent' 10 | import Header from '../components/header' 11 | import Settings from '../components/settings' 12 | 13 | import images from '../components/images' 14 | 15 | const App = () => { 16 | return ( 17 | 18 |
19 | 20 |
21 | 22 | {images.map((image, index) => ( 23 |
30 | {index + 1} 31 |
32 | ))} 33 |
34 |
35 | 36 | ) 37 | } 38 | 39 | export default App 40 | -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AndreasFaust/react-sled/c9168f72fd1f062d4b8401f0597c1831cdb3027a/public/favicon.ico -------------------------------------------------------------------------------- /public/plex-mono-bold.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AndreasFaust/react-sled/c9168f72fd1f062d4b8401f0597c1831cdb3027a/public/plex-mono-bold.woff -------------------------------------------------------------------------------- /public/plex-mono.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AndreasFaust/react-sled/c9168f72fd1f062d4b8401f0597c1831cdb3027a/public/plex-mono.woff -------------------------------------------------------------------------------- /react-sled-logo.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/AndreasFaust/react-sled/c9168f72fd1f062d4b8401f0597c1831cdb3027a/react-sled-logo.jpg -------------------------------------------------------------------------------- /rollup.config.js: -------------------------------------------------------------------------------- 1 | import typescript from 'rollup-plugin-typescript2' 2 | import css from 'rollup-plugin-css-only' 3 | import commonjs from '@rollup/plugin-commonjs' 4 | import resolve from '@rollup/plugin-node-resolve' 5 | import external from 'rollup-plugin-peer-deps-external' 6 | 7 | import pkg from './package.json' 8 | 9 | export default { 10 | input: 'sled/index.ts', 11 | output: [ 12 | { 13 | file: pkg.main, 14 | format: 'cjs', 15 | exports: 'named', 16 | sourcemap: true 17 | }, 18 | { 19 | file: pkg.module, 20 | format: 'es', 21 | exports: 'named', 22 | sourcemap: true 23 | } 24 | ], 25 | plugins: [ 26 | external(), 27 | resolve({ 28 | browser: true 29 | }), 30 | typescript({ 31 | tsconfig: 'tsconfig.rollup.json', 32 | rollupCommonJSResolveHack: true, 33 | exclude: '**/__tests__/**', 34 | clean: true 35 | }), 36 | commonjs({ 37 | include: ['node_modules/**'], 38 | exclude: ['**/*.stories.js'], 39 | namedExports: { 40 | 'node_modules/react/react.js': [ 41 | 'Children', 42 | 'Component', 43 | 'PropTypes', 44 | 'createElement' 45 | ], 46 | 'node_modules/react-dom/index.js': ['render'] 47 | } 48 | }), 49 | css({ output: './dist/main.css' }) 50 | ] 51 | } 52 | -------------------------------------------------------------------------------- /sled/control/hooks/useClassName.ts: -------------------------------------------------------------------------------- 1 | import { TSelect } from "../../state/types-defaults" 2 | import { useStateContext } from '../../state' 3 | import { useDirectionDisabled } from './useDirectionDisabled' 4 | 5 | function getClassName(type: 'index' | 'direction', select: TSelect, className: string): string { 6 | const [{ currentIndex }] = useStateContext() 7 | 8 | const directionDisabled = useDirectionDisabled(select) 9 | 10 | const baseClass: string = 'sled-control' 11 | const typeClass: string = `${baseClass}-${type}` 12 | const typeClassSpecific: string = `${typeClass}-${select}` 13 | const distinctClass: string = `${baseClass}-${className || type + '-default'}` 14 | const disabledClasses: string[] = [ 15 | `${baseClass}-disabled`, 16 | `${typeClass}-disabled`, 17 | `${distinctClass}-disabled` 18 | ] 19 | const activeClasses: string[] = [ 20 | `${baseClass}-active`, 21 | `${typeClass}-active`, 22 | `${distinctClass}-active` 23 | ] 24 | 25 | const classes: string[] = [ 26 | baseClass, 27 | typeClass, 28 | typeClassSpecific, 29 | distinctClass, 30 | ] 31 | 32 | if (select === currentIndex) { // isActive 33 | return [...classes, ...activeClasses].join(' ') 34 | } 35 | if (directionDisabled) { 36 | return [...classes, ...disabledClasses].join(' ') 37 | } 38 | return classes.join(' ') 39 | } 40 | 41 | export default (select: TSelect, className: string) => { 42 | switch (typeof select) { 43 | case 'number': 44 | return getClassName('index', select, className) 45 | case 'string': 46 | default: 47 | return getClassName('direction', select, className) 48 | } 49 | } -------------------------------------------------------------------------------- /sled/control/hooks/useClick.ts: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { useStateContext } from '../../state' 3 | import { TSelect } from "../../state/types-defaults" 4 | 5 | interface IReturn { 6 | (): void; 7 | } 8 | 9 | const onClickFunction = (select: TSelect): IReturn => { 10 | const [{ stopOnInteraction }, dispatch] = useStateContext() 11 | 12 | function onClick() { 13 | if (stopOnInteraction) { 14 | dispatch({ type: 'SET_AUTOPLAY', autoPlayInterval: undefined }) 15 | } 16 | if (typeof select === 'number') { 17 | dispatch({ type: 'SELECT', index: select }) 18 | } else { 19 | dispatch({ type: select === 'next' ? 'NEXT' : 'PREV' }) 20 | } 21 | } 22 | return onClick 23 | } 24 | 25 | export default onClickFunction -------------------------------------------------------------------------------- /sled/control/hooks/useDirectionDisabled.ts: -------------------------------------------------------------------------------- 1 | import { useState, useEffect } from 'react' 2 | import { useStateContext } from '../../state' 3 | import { TSelect } from "../../state/types-defaults" 4 | 5 | export const useDirectionDisabled = (select: TSelect) => { 6 | const [{ rewind, currentIndex, viewCount }] = useStateContext() 7 | const [disabled, setDisabled] = useState(false) 8 | useEffect(() => { 9 | if (typeof select === 'string') { 10 | if (rewind) { 11 | setDisabled(false) 12 | return 13 | } 14 | if (select === 'next' && currentIndex === viewCount - 1) { 15 | setDisabled(true) 16 | } else if (select === 'prev' && currentIndex === 0) { 17 | setDisabled(true) 18 | } else { 19 | setDisabled(false) 20 | } 21 | } 22 | }, [currentIndex, rewind]) 23 | return disabled 24 | } 25 | 26 | -------------------------------------------------------------------------------- /sled/control/hooks/useLabel.ts: -------------------------------------------------------------------------------- 1 | import { TSelect } from "../../state/types-defaults" 2 | 3 | export default (select: TSelect) => { 4 | switch (typeof select) { 5 | case 'number': 6 | return `Slide to view with index ${select}.` 7 | case 'string': 8 | default: 9 | return `Slide to ${select === 'next' ? 'next' : 'previous'} view.` 10 | } 11 | } 12 | -------------------------------------------------------------------------------- /sled/control/index.css: -------------------------------------------------------------------------------- 1 | /* direction */ 2 | 3 | .sled-control-direction-default { 4 | cursor: pointer; 5 | width: 40px; 6 | height: 40px; 7 | display: flex; 8 | justify-content: center; 9 | align-items: center; 10 | background: none; 11 | box-shadow: none; 12 | border: none; 13 | transition: opacity 0.5s, transform 0.5s; 14 | opacity: 0.5; 15 | } 16 | 17 | .sled-control-direction-default.sled-control-direction-disabled { 18 | pointer-events: none; 19 | } 20 | 21 | .sled-control-direction-default.sled-control-direction-prev { 22 | transform-origin: right center; 23 | } 24 | .sled-control-direction-default.sled-control-direction-prev:before { 25 | border-left: 3px solid black; 26 | transform: rotate(-45deg); 27 | } 28 | 29 | .sled-control-direction-default.sled-control-direction-next { 30 | transform-origin: left center; 31 | } 32 | .sled-control-direction-default.sled-control-direction-next:before { 33 | border-right: 3px solid black; 34 | transform: rotate(45deg); 35 | } 36 | 37 | .sled-control-direction-default:focus { 38 | box-shadow: none; 39 | outline: none; 40 | } 41 | 42 | .sled-control-direction-default:before { 43 | content: ''; 44 | display: block; 45 | flex-shrink: 0; 46 | width: 15px; 47 | height: 15px; 48 | border-top: 3px solid black; 49 | } 50 | 51 | .sled-control-direction-default:focus { 52 | opacity: 1; 53 | } 54 | 55 | .sled-control-direction-default:hover, 56 | .sled-control-direction-default:active { 57 | opacity: 1; 58 | transform: scale(1.2); 59 | } 60 | 61 | .sled-control-direction-default-disabled { 62 | opacity: 0.25; 63 | } 64 | 65 | /* index */ 66 | 67 | .sled-control-index-default { 68 | cursor: pointer; 69 | width: 40px; 70 | height: 40px; 71 | display: flex; 72 | justify-content: center; 73 | align-items: center; 74 | background: none; 75 | box-shadow: none; 76 | border: none; 77 | transition: opacity 0.5s, transform 0.5s; 78 | opacity: 0.4; 79 | } 80 | 81 | .sled-control-index-default.sled-control-index-disabled { 82 | pointer-events: none; 83 | } 84 | 85 | .sled-control-index-default:focus { 86 | box-shadow: none; 87 | outline: none; 88 | } 89 | 90 | .sled-control-index-default:before { 91 | content: ''; 92 | display: block; 93 | flex-shrink: 0; 94 | width: 10px; 95 | height: 10px; 96 | background: grey; 97 | border-radius: 50%; 98 | } 99 | 100 | .sled-control-index-default:hover, 101 | .sled-control-index-default:focus { 102 | transform: scale(1.2); 103 | } 104 | 105 | .sled-control-index-default:hover, 106 | .sled-control-index-default:focus, 107 | .sled-control-index-default:active { 108 | opacity: 0.6; 109 | } 110 | 111 | .sled-control-index-default:disabled { 112 | pointer-events: none; 113 | } 114 | 115 | .sled-control-index-default-active { 116 | opacity: 1; 117 | pointer-events: none; 118 | } -------------------------------------------------------------------------------- /sled/control/index.tsx: -------------------------------------------------------------------------------- 1 | import React, { useRef, ReactNode } from 'react' 2 | import CSS from 'csstype'; 3 | import { TSelect } from "../state/types-defaults" 4 | import { useStateContext } from '../state' 5 | import { useDirectionDisabled } from './hooks/useDirectionDisabled' 6 | 7 | import './index.css' 8 | 9 | import useClassName from './hooks/useClassName' 10 | import useLabel from './hooks/useLabel' 11 | import useClick from './hooks/useClick' 12 | 13 | import useFocus from '../hooks/useFocus' 14 | 15 | interface IProps { 16 | select: TSelect 17 | style?: CSS.Properties 18 | className?: string 19 | children?: ReactNode 20 | } 21 | 22 | const SledControl: React.FC = ({ 23 | children, select, style, className 24 | }) => { 25 | const controlRef = useRef() 26 | useFocus(controlRef) 27 | const directionDisabled = useDirectionDisabled(select) 28 | const classNames = useClassName(select, className) 29 | const label = useLabel(select) 30 | const onClick = useClick(select) 31 | const [{ currentIndex }] = useStateContext() 32 | 33 | return ( 34 |
44 | {children} 45 |
46 | ) 47 | } 48 | 49 | export default SledControl 50 | -------------------------------------------------------------------------------- /sled/hooks/useAutoPlay.ts: -------------------------------------------------------------------------------- 1 | import { useRef, useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useInterval = (callback: any, interval: number): void => { 5 | const savedCallback = useRef(callback) 6 | 7 | useEffect(() => { 8 | savedCallback.current = callback 9 | }) 10 | 11 | useEffect(() => { 12 | function tick() { 13 | savedCallback.current() 14 | } 15 | if (typeof interval === 'number') { 16 | let id = setInterval(tick, interval) 17 | return () => clearInterval(id) 18 | } 19 | }, [interval]) 20 | } 21 | 22 | export default (autoPlayIntervalNew: number): void => { 23 | const [{ pause, autoPlayInterval }, dispatch] = useStateContext() 24 | 25 | useEffect(() => { 26 | if (typeof autoPlayIntervalNew === 'number') { 27 | dispatch({ type: 'SET_AUTOPLAY', autoPlayInterval: autoPlayIntervalNew }) 28 | } else { 29 | console.warn(`Sled-Error: "autoplay" must be of type "number", not "${typeof autoPlayIntervalNew}".`) 30 | } 31 | }, [autoPlayIntervalNew]) 32 | 33 | useInterval(() => { 34 | dispatch({ type: 'NEXT' }) 35 | }, !pause && autoPlayInterval) 36 | } 37 | -------------------------------------------------------------------------------- /sled/hooks/useConfig.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | import { SpringConfig } from 'react-spring' 4 | 5 | export default (config: SpringConfig) => { 6 | const [, dispatch] = useStateContext() 7 | useEffect(() => { 8 | dispatch({ type: 'SET_CONFIG', config }) 9 | }, [config, dispatch]) 10 | } 11 | -------------------------------------------------------------------------------- /sled/hooks/useContainerStyles.ts: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { useStateContext } from '../state' 3 | import CSS from 'csstype' 4 | 5 | export function useSliderStyles() { 6 | const [{ sliderSize, direction, dragging }] = useStateContext() 7 | const [styles, setStyles] = React.useState({}) 8 | 9 | React.useEffect(() => { 10 | const defaultStyles = { 11 | position: 'absolute', 12 | top: 0, 13 | left: 0, 14 | willChange: 'transform', 15 | overflow: 'hidden', 16 | cursor: dragging ? 'grab' : 'auto' 17 | } 18 | if (direction === 'vertical') { 19 | setStyles({ 20 | ...defaultStyles, 21 | width: '100%', 22 | height: sliderSize || 0, 23 | }) 24 | } else { 25 | setStyles({ 26 | ...defaultStyles, 27 | width: sliderSize || 0, 28 | height: '100%', 29 | display: 'flex', 30 | }) 31 | } 32 | }, [direction, dragging, sliderSize]) 33 | 34 | return styles 35 | } 36 | 37 | 38 | export function useViewStyles(): CSS.Properties { 39 | const [{ dimensions: { width, height }, viewCount, direction, showElements }] = useStateContext() 40 | const [styles, setStyles] = React.useState({}) 41 | 42 | React.useEffect(() => { 43 | const defaultStyles = { 44 | position: 'relative', 45 | } 46 | if (direction === 'vertical') { 47 | setStyles({ 48 | ...defaultStyles, 49 | width: '100%', 50 | height: height / showElements 51 | }) 52 | } else { 53 | setStyles({ 54 | ...defaultStyles, 55 | width: width / showElements, 56 | height: '100%' 57 | }) 58 | } 59 | }, [width, height, direction, viewCount, showElements]) 60 | 61 | return styles 62 | } 63 | -------------------------------------------------------------------------------- /sled/hooks/useCursor.ts: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | type TCursor = 'grab' | 'auto' 5 | 6 | export default (): TCursor => { 7 | const [{ dragging }] = useStateContext() 8 | 9 | const [cursor, setCursor] = React.useState('auto') 10 | React.useEffect(() => { 11 | setCursor(dragging ? 'grab' : 'auto') 12 | }, [dragging]) 13 | 14 | return cursor 15 | } -------------------------------------------------------------------------------- /sled/hooks/useDimensions.ts: -------------------------------------------------------------------------------- 1 | 2 | import React from 'react' 3 | import { debounce } from '../utils/debounce' 4 | import { useStateContext } from '../state' 5 | import { TRef, TDimension } from '../state/types-defaults' 6 | 7 | interface IProportion { 8 | width: TDimension 9 | height: TDimension 10 | offsetWidth: number 11 | offsetHeight: number 12 | proportion: string 13 | dispatch 14 | } 15 | 16 | interface IDimensions { 17 | width: TDimension 18 | height: TDimension 19 | } 20 | 21 | function getProportion({ 22 | width, 23 | height, 24 | offsetWidth, 25 | offsetHeight, 26 | proportion, 27 | dispatch 28 | }: IProportion) { 29 | if (!proportion) return 30 | 31 | const [proportionWidth, proportionHeight] = proportion.split(':') 32 | let dimensions: IDimensions = { width: 0, height: 0 } 33 | 34 | if (width) { 35 | const heightValue = (offsetWidth * +proportionHeight) / +proportionWidth 36 | dimensions = { 37 | width: offsetWidth, 38 | height: heightValue 39 | } 40 | } else { 41 | const widthValue = (offsetHeight * +proportionWidth) / +proportionHeight 42 | dimensions = { 43 | width: widthValue, 44 | height: offsetHeight 45 | } 46 | } 47 | dispatch({ 48 | type: 'SET_DIMENSIONS', 49 | dimensions: dimensions 50 | }) 51 | } 52 | 53 | export default (ref: TRef) => { 54 | const [{ dimensionsDOM, proportion }, dispatch] = useStateContext() 55 | 56 | React.useEffect(() => { 57 | function onResize() { 58 | const { offsetWidth, offsetHeight } = ref.current 59 | getProportion({ 60 | ...dimensionsDOM, 61 | offsetWidth, 62 | offsetHeight, 63 | proportion, 64 | dispatch 65 | }) 66 | } 67 | onResize() 68 | const dOnResize = debounce(onResize, 200) 69 | window.addEventListener('resize', dOnResize) 70 | return () => window.removeEventListener('resize', dOnResize) 71 | 72 | }, [dimensionsDOM, proportion]) 73 | 74 | } 75 | -------------------------------------------------------------------------------- /sled/hooks/useDimensionsDOM.ts: -------------------------------------------------------------------------------- 1 | 2 | import React from 'react' 3 | import { useStateContext, TDimension } from '../state' 4 | 5 | export default (width: TDimension, height: TDimension) => { 6 | const [, dispatch] = useStateContext() 7 | 8 | React.useEffect(() => { 9 | if (!width && !height) { 10 | dispatch({ 11 | type: 'SET_DIMENSIONS_DOM', 12 | dimensionsDOM: { width: '100%', height: null } 13 | }) 14 | dispatch({ type: 'SET_PROPORTION', proportion: '2:1' }) 15 | return 16 | } 17 | 18 | dispatch({ 19 | type: 'SET_DIMENSIONS_DOM', 20 | dimensionsDOM: { width, height } 21 | }) 22 | 23 | }, [width, height]) 24 | 25 | } 26 | -------------------------------------------------------------------------------- /sled/hooks/useDirection.ts: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { useStateContext } from '../state' 3 | import { TDirection } from '../state/types-defaults' 4 | 5 | export default (direction: TDirection) => { 6 | const [, dispatch] = useStateContext() 7 | 8 | React.useEffect(() => { 9 | if (direction) { 10 | dispatch({ type: 'SET_DIRECTION', direction }) 11 | } 12 | }, [direction]) 13 | } 14 | -------------------------------------------------------------------------------- /sled/hooks/useDragGesture.ts: -------------------------------------------------------------------------------- 1 | import { useDrag } from 'react-use-gesture' 2 | import { useStateContext } from '../state' 3 | import { SpringsUpdateFn } from 'react-spring' 4 | 5 | interface ISet { 6 | x: number 7 | } 8 | 9 | export default (set: SpringsUpdateFn) => { 10 | const [{ 11 | dragging, 12 | dragDistance, 13 | dimensions: { width, height }, 14 | currentIndex, 15 | direction, 16 | stopOnInteraction, 17 | showElements, 18 | slideBy 19 | }, dispatch] = useStateContext() 20 | 21 | const bind = useDrag(({ 22 | down, 23 | movement: [xDelta, yDelta], 24 | direction: [xDir, yDir], 25 | distance, 26 | cancel, 27 | canceled, 28 | }) => { 29 | if (canceled) return 30 | 31 | if (stopOnInteraction) { 32 | dispatch({ type: 'SET_AUTOPLAY', autoPlayInterval: undefined }) 33 | } 34 | if (down && distance > dragDistance) { 35 | const dirValue = direction === 'horizontal' ? xDir : yDir 36 | dispatch({ 37 | type: dirValue > 0 38 | ? 'PREV' 39 | : 'NEXT', 40 | pause: true 41 | }) 42 | cancel() 43 | } 44 | set(() => { 45 | const x = direction === 'horizontal' 46 | ? (-currentIndex * (width / showElements * slideBy)) + (down ? xDelta : 0) 47 | : (-currentIndex * (height / showElements * slideBy)) + (down ? yDelta : 0) 48 | return { 49 | x, 50 | immediate: false, 51 | cursor: down ? 'grabbing' : 'grab', 52 | onStart: undefined, 53 | onRest: undefined 54 | } 55 | }) 56 | dispatch({ type: 'SET_PAUSE', pause: true }) 57 | }) 58 | 59 | return dragging && bind 60 | } 61 | -------------------------------------------------------------------------------- /sled/hooks/useDragging.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | function getDistance(distance: string | number, width: number): number { 5 | switch (typeof distance) { 6 | case 'number': return distance 7 | case 'string': 8 | if (distance.indexOf('%') >= 0) { 9 | return (width / 100) * +distance.replace('%', '') 10 | } 11 | const distanceNumber = parseInt(distance, 10) 12 | if (distanceNumber) { 13 | return distanceNumber 14 | } 15 | console.warn('Sled-Error: dragDistance must either be a String with unit % or a number.') 16 | return 40 17 | default: 18 | return 40 19 | } 20 | } 21 | 22 | export default (dragging: boolean, dragDistance: string | number) => { 23 | const [{ dimensions: { width } }, dispatch] = useStateContext() 24 | 25 | useEffect(() => { 26 | const distance = getDistance(dragDistance, width) 27 | dispatch({ type: 'SET_DRAG_DISTANCE', dragDistance: distance }) 28 | }, [dispatch, dragDistance, width]) 29 | 30 | useEffect(() => { 31 | dispatch({ type: 'SET_DRAGGING', dragging }) 32 | }, [dispatch, dragging]) 33 | } 34 | -------------------------------------------------------------------------------- /sled/hooks/useFocus.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useFocus = (ref: React.MutableRefObject) => { 5 | const [, dispatch] = useStateContext() 6 | 7 | useEffect(() => { 8 | function onFocus() { 9 | dispatch({ type: 'SET_FOCUS', focus: true }) 10 | } 11 | function onBlur() { 12 | dispatch({ type: 'SET_FOCUS', focus: false }) 13 | } 14 | ref.current.addEventListener('focus', onFocus) 15 | ref.current.addEventListener('blur', onBlur) 16 | return () => { 17 | ref.current.removeEventListener('focus', onFocus) 18 | ref.current.removeEventListener('blur', onBlur) 19 | } 20 | }, []) 21 | } 22 | 23 | export default useFocus 24 | -------------------------------------------------------------------------------- /sled/hooks/useKeyboard.ts: -------------------------------------------------------------------------------- 1 | import { useEffect, useRef } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useKeyboard = (keyboard: boolean) => { 5 | const [{ hasFocus, stopOnInteraction }, dispatch] = useStateContext() 6 | const hasFocusRef = useRef(false) 7 | 8 | useEffect(() => { 9 | hasFocusRef.current = hasFocus 10 | }, [hasFocus]) 11 | 12 | useEffect(() => { 13 | function onKeyup(event: KeyboardEvent) { 14 | if (!hasFocusRef.current) return 15 | 16 | if (stopOnInteraction) { 17 | dispatch({ type: 'SET_AUTOPLAY', autoPlayInterval: undefined }) 18 | } 19 | 20 | switch (event.keyCode) { 21 | case 37: 22 | dispatch({ type: 'PREV' }) 23 | break 24 | case 39: 25 | dispatch({ type: 'NEXT' }) 26 | break 27 | } 28 | } 29 | if (keyboard) { 30 | document.addEventListener('keyup', onKeyup) 31 | return () => document.removeEventListener('keyup', onKeyup) 32 | } 33 | }, [keyboard, stopOnInteraction]) 34 | } 35 | 36 | export default useKeyboard 37 | -------------------------------------------------------------------------------- /sled/hooks/useMouseOver.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | export default (pauseOnMouseOver: boolean, ref: React.MutableRefObject) => { 5 | const [{ autoPlayInterval }, dispatch] = useStateContext() 6 | useEffect(() => { 7 | dispatch({ type: 'SET_PAUSE', pause: false }) 8 | 9 | function onMouseEnter() { 10 | dispatch({ type: 'SET_MOUSEOVER', pauseOnMouseOver: true }) 11 | dispatch({ type: 'SET_PAUSE', pause: true }) 12 | } 13 | function onMouseLeave() { 14 | dispatch({ type: 'SET_MOUSEOVER', pauseOnMouseOver: false }) 15 | dispatch({ type: 'SET_PAUSE', pause: false }) 16 | } 17 | if (pauseOnMouseOver && autoPlayInterval) { 18 | ref.current.addEventListener('mouseenter', onMouseEnter) 19 | ref.current.addEventListener('mouseover', onMouseEnter) 20 | ref.current.addEventListener('mouseout', onMouseLeave) 21 | return () => { 22 | dispatch({ type: 'SET_MOUSEOVER', pauseOnMouseOver: false }) 23 | ref.current.removeEventListener('mouseenter', onMouseEnter) 24 | ref.current.removeEventListener('mouseover', onMouseEnter) 25 | ref.current.removeEventListener('mouseout', onMouseLeave) 26 | } 27 | } 28 | }, [pauseOnMouseOver, autoPlayInterval, dispatch, ref]) 29 | } 30 | -------------------------------------------------------------------------------- /sled/hooks/usePause.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | export default (pause: boolean) => { 5 | const [, dispatch] = useStateContext() 6 | useEffect(() => { 7 | dispatch({ type: 'SET_PAUSE', pause }) 8 | }, [pause]) 9 | } 10 | -------------------------------------------------------------------------------- /sled/hooks/useProportion.ts: -------------------------------------------------------------------------------- 1 | 2 | import React from 'react' 3 | import { useStateContext, TDimension } from '../state' 4 | 5 | export default (width: TDimension, height: TDimension, proportion?: string): void => { 6 | const [, dispatch] = useStateContext() 7 | React.useEffect(() => { 8 | if (!height && !proportion) { 9 | dispatch({ type: 'SET_PROPORTION', proportion: '2:1' }) 10 | } else { 11 | dispatch({ type: 'SET_PROPORTION', proportion }) 12 | } 13 | }, [width, height, proportion]) 14 | } 15 | -------------------------------------------------------------------------------- /sled/hooks/useRewind.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | export default (rewind: boolean) => { 5 | const [, dispatch] = useStateContext() 6 | useEffect(() => { 7 | dispatch({ type: 'SET_REWIND', rewind }) 8 | }, [rewind]) 9 | } 10 | -------------------------------------------------------------------------------- /sled/hooks/useSelect.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useSelect = (select?: number | 'next' | 'prev') => { 5 | const [, dispatch] = useStateContext() 6 | useEffect(() => { 7 | if (typeof select === 'number') { 8 | dispatch({ type: 'SELECT', index: select }) 9 | } else { 10 | dispatch({ type: select === 'next' ? 'NEXT' : 'PREV' }) 11 | } 12 | }, [select]) 13 | } 14 | 15 | export default useSelect 16 | -------------------------------------------------------------------------------- /sled/hooks/useShowElements.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useShowElements = (showElements: number) => { 5 | const [, dispatch] = useStateContext() 6 | useEffect(() => { 7 | if (showElements) { 8 | dispatch({ type: 'SET_SHOWELEMENTS', showElements }) 9 | } 10 | }, [showElements]) 11 | } 12 | 13 | export default useShowElements 14 | -------------------------------------------------------------------------------- /sled/hooks/useSlideBy.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useShowElements = (slideBy: number) => { 5 | const [, dispatch] = useStateContext() 6 | useEffect(() => { 7 | if (slideBy) { 8 | dispatch({ type: 'SET_SLIDEBY', slideBy }) 9 | } 10 | }, [slideBy]) 11 | } 12 | 13 | export default useShowElements 14 | -------------------------------------------------------------------------------- /sled/hooks/useSlideSteps.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useSlideSteps = () => { 5 | const [{ viewCount, slideBy }, dispatch] = useStateContext() 6 | useEffect(() => { 7 | const slideSteps = Math.ceil(viewCount / slideBy) 8 | dispatch({ type: 'SET_SLIDESTEPS', slideSteps }) 9 | }, [viewCount, slideBy]) 10 | } 11 | 12 | export default useSlideSteps 13 | -------------------------------------------------------------------------------- /sled/hooks/useSliderSize.ts: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | export default function useSliderStyles() { 5 | const [{ dimensions: { width, height }, viewCount, direction, showElements }, dispatch] = useStateContext() 6 | 7 | React.useEffect(() => { 8 | if (direction === 'vertical') { 9 | dispatch({ type: 'SET_SLIDERSIZE', sliderSize: (height / showElements) * viewCount }) 10 | } else { 11 | dispatch({ type: 'SET_SLIDERSIZE', sliderSize: (width / showElements) * viewCount }) 12 | } 13 | }, [direction, width, height, showElements, viewCount]) 14 | 15 | } -------------------------------------------------------------------------------- /sled/hooks/useStopOnInteraction.ts: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | const useStopOnInteraction = (stopOnInteraction: boolean) => { 5 | const [, dispatch] = useStateContext() 6 | 7 | useEffect(() => { 8 | dispatch({ type: 'SET_STOPONINTERACTION', stopOnInteraction }) 9 | }, [stopOnInteraction]) 10 | } 11 | 12 | export default useStopOnInteraction 13 | -------------------------------------------------------------------------------- /sled/hooks/useViewCount.ts: -------------------------------------------------------------------------------- 1 | import React, { useEffect } from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | export default function (children?: React.ReactNode) { 5 | const [, dispatch] = useStateContext() 6 | 7 | useEffect(() => { 8 | dispatch({ type: 'SET_VIEWCOUNT', count: React.Children.toArray(children).length }) 9 | }, [children]) 10 | } 11 | -------------------------------------------------------------------------------- /sled/hooks/useX.ts: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | export default function () { 5 | const [{ direction, dimensions: { width, height }, showElements, slideBy, currentIndex }] = useStateContext() 6 | const [x, setX] = React.useState(0) 7 | React.useEffect(() => { 8 | if (direction === 'horizontal') { 9 | setX(-currentIndex * (width / showElements * slideBy)) 10 | } else { 11 | setX(-currentIndex * (height / showElements * slideBy)) 12 | } 13 | }, [direction, width, height, showElements, slideBy, currentIndex]) 14 | 15 | return x 16 | } 17 | -------------------------------------------------------------------------------- /sled/index.css: -------------------------------------------------------------------------------- 1 | .sled { 2 | position: relative; 3 | overflow: hidden; 4 | user-select: none; 5 | } 6 | .sled:focus, 7 | .sled-view:focus { 8 | outline: none; 9 | } 10 | 11 | /* .sled-proportion::after { 12 | content: ''; 13 | display: block; 14 | width: 100%; 15 | height: 0; 16 | } */ 17 | -------------------------------------------------------------------------------- /sled/index.ts: -------------------------------------------------------------------------------- 1 | export { default as Sled } from './sled' 2 | export { default as Views } from './views' 3 | export { default as Control } from './control' 4 | export { default as Progress } from './progress' 5 | export { useStateContext as useSledStore } from './state' 6 | -------------------------------------------------------------------------------- /sled/progress/controls.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import CSS from 'csstype' 3 | 4 | import { useStateContext } from '../state' 5 | import Control from '../control' 6 | 7 | interface IProps { 8 | className: string 9 | } 10 | 11 | const SledProgressControls: React.FC = ({ className }) => { 12 | const [{ slideSteps }] = useStateContext() 13 | 14 | return ( 15 |
16 | { 17 | [...Array(slideSteps ? slideSteps || 1 : 1)].map((view, index) => ( 18 | 31 | ))} 32 |
33 | ) 34 | } 35 | 36 | export default SledProgressControls 37 | -------------------------------------------------------------------------------- /sled/progress/index.css: -------------------------------------------------------------------------------- 1 | .sled-progress-default { 2 | position: relative; 3 | width: 100%; 4 | display: flex; 5 | align-items: center; 6 | overflow: hidden; 7 | height: 20px; 8 | } 9 | 10 | .sled-progress-default-rail { 11 | position: absolute; 12 | left: 0; 13 | width: 100%; 14 | pointer-events: none; 15 | background: black; 16 | height: 4px; 17 | } 18 | 19 | .sled-progress-default-track { 20 | background: red; 21 | height: 4px; 22 | } 23 | 24 | .sled-progress-default-separators { 25 | position: absolute; 26 | z-index: 200; 27 | top: 0; 28 | bottom: 0; 29 | left: 0; 30 | right: 0; 31 | display: flex; 32 | justify-content: space-between; 33 | pointer-events: none; 34 | } 35 | 36 | .sled-progress-default-separator { 37 | width: 4px; 38 | background: white; 39 | } 40 | 41 | /* Mimic "justify-content: space-evenly;" */ 42 | .sled-progress-default-separators:before, 43 | .sled-progress-default-separators:after { 44 | content: ''; 45 | display: block; 46 | height: 100%; 47 | } 48 | 49 | .sled-progress-default-controls { 50 | position: absolute; 51 | z-index: 100; 52 | top: 0; 53 | bottom: 0; 54 | left: 0; 55 | right: 0; 56 | display: flex; 57 | } 58 | 59 | .sled-progress-default-control { 60 | position: absolute; 61 | z-index: 100; 62 | top: 0; 63 | bottom: 0; 64 | left: 0; 65 | right: 0; 66 | display: flex; 67 | } -------------------------------------------------------------------------------- /sled/progress/index.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import SledProgressSeparators from './separators' 3 | import SledProgressTrack from './track' 4 | import SledProgressControls from './controls' 5 | import CSS from 'csstype' 6 | 7 | import './index.css' 8 | 9 | interface IProps { 10 | className?: string 11 | style?: CSS.Properties 12 | } 13 | 14 | const SledProgress: React.FC = ({ className, style }) => { 15 | const [defaultClass, setDefaultClass] = React.useState(className || ' default') 16 | React.useEffect(() => { 17 | setDefaultClass(className || 'default') 18 | }, [className]) 19 | 20 | return ( 21 |
26 |
27 | 28 | 29 | 30 |
31 | ) 32 | } 33 | 34 | export default SledProgress 35 | -------------------------------------------------------------------------------- /sled/progress/separators.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { useStateContext } from '../state' 3 | 4 | interface IProps { 5 | className: string 6 | } 7 | 8 | const SledProgressSeparators: React.FC = ({ className }) => { 9 | const [{ slideSteps }] = useStateContext() 10 | return slideSteps && ( 11 |
12 | {[...Array(slideSteps - 1)].map((_, index) => ( 13 |
21 | ))} 22 |
23 | ) 24 | } 25 | 26 | export default SledProgressSeparators 27 | -------------------------------------------------------------------------------- /sled/progress/track.tsx: -------------------------------------------------------------------------------- 1 | import React, { useEffect } from 'react' 2 | import { useSpring, animated } from 'react-spring' 3 | 4 | import { useStateContext } from '../state' 5 | 6 | function getX(slideSteps: number, currentIndex: number, goPrevNext: number = 0): number { 7 | return 100 - ((100 / slideSteps) * (currentIndex + goPrevNext)) 8 | } 9 | 10 | interface IProps { 11 | className: string 12 | } 13 | 14 | const SledProgressTrack: React.FC = ({ className }) => { 15 | const [{ 16 | currentIndex, 17 | prevIndex, 18 | slideSteps, 19 | autoPlayInterval, 20 | pause, 21 | pauseOnMouseOver, 22 | config 23 | }] = useStateContext() 24 | 25 | const [props, set] = useSpring(() => ({ 26 | from: { x: 100 } 27 | })) 28 | 29 | useEffect(() => { 30 | if (pauseOnMouseOver) { 31 | set({ 32 | config, 33 | x: getX(slideSteps, currentIndex), 34 | reset: false 35 | }) 36 | } 37 | }, [pauseOnMouseOver]) 38 | 39 | useEffect(() => { 40 | if (!slideSteps) return 41 | const xCalc = getX(slideSteps, currentIndex, !autoPlayInterval && 1) 42 | if (currentIndex === 0) { 43 | set({ 44 | config, 45 | from: { x: 100 }, 46 | x: xCalc, 47 | reset: slideSteps > 2 48 | ? prevIndex !== 1 49 | : true 50 | }) 51 | } else { 52 | set({ 53 | config, 54 | x: xCalc, 55 | reset: false 56 | }) 57 | } 58 | }, [slideSteps, currentIndex, autoPlayInterval]) 59 | 60 | useEffect(() => { 61 | if (!autoPlayInterval) return 62 | set({ 63 | config: autoPlayInterval && !pause 64 | ? { duration: autoPlayInterval } 65 | : config, 66 | from: { x: getX(slideSteps, currentIndex) }, 67 | x: getX(slideSteps, currentIndex, !pause && 1), 68 | reset: true 69 | }) 70 | }, [pause, autoPlayInterval, slideSteps, currentIndex]) 71 | 72 | return ( 73 | `translateX(-${x}%)`) 81 | }} 82 | /> 83 | ) 84 | } 85 | 86 | export default SledProgressTrack 87 | -------------------------------------------------------------------------------- /sled/sled.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | import { StateProvider } from './state' 4 | 5 | interface IProps { 6 | children?: React.ReactNode 7 | } 8 | 9 | const Sled: React.FC = (props) => { 10 | const { children } = props 11 | return ( 12 | 13 | {children} 14 | 15 | ) 16 | } 17 | 18 | export default Sled 19 | -------------------------------------------------------------------------------- /sled/springs.tsx: -------------------------------------------------------------------------------- 1 | import React, { useEffect } from 'react' 2 | import { useSpring, animated } from 'react-spring' 3 | 4 | import { useStateContext } from './state' 5 | import useDragGesture from './hooks/useDragGesture' 6 | import useFocus from './hooks/useFocus' 7 | import useCursor from './hooks/useCursor' 8 | import useX from './hooks/useX' 9 | import { useSliderStyles, useViewStyles } from './hooks/useContainerStyles' 10 | 11 | interface IProps { 12 | onAnimationStart(): void 13 | onAnimationEnd(): void 14 | onSledEnd(): void 15 | children?: React.ReactNode 16 | } 17 | 18 | const SledSprings: React.FC = ({ 19 | onAnimationStart, 20 | onAnimationEnd, 21 | onSledEnd, 22 | children 23 | }) => { 24 | const [{ 25 | currentIndex, 26 | viewCount, 27 | direction, 28 | dimensions: { width, height }, 29 | config, 30 | }, dispatch] = useStateContext() 31 | 32 | const cursor = useCursor() 33 | const x = useX() 34 | 35 | const [props, set] = useSpring(() => ({ 36 | x: 0, 37 | cursor, 38 | config, 39 | immediate: true 40 | })) 41 | const springRef = React.useRef() 42 | 43 | useFocus(springRef) 44 | 45 | const [dimensionsUpdated, setDimensionsUpdated] = React.useState(false) 46 | 47 | useEffect(() => { 48 | setDimensionsUpdated(true) 49 | }, [width, height]) 50 | 51 | useEffect(() => { 52 | if (!dimensionsUpdated) return 53 | setDimensionsUpdated(false) 54 | set({ 55 | x, 56 | immediate: true, 57 | onStart: null, 58 | onRest: null, 59 | }) 60 | }, [x, dimensionsUpdated]) 61 | 62 | useEffect(() => { 63 | if (dimensionsUpdated) return 64 | set({ 65 | config, 66 | x, 67 | cursor, 68 | immediate: false, 69 | onStart() { 70 | dispatch({ type: 'SET_PAUSE', pause: true }) 71 | if (typeof onAnimationStart === 'function') onAnimationStart() 72 | }, 73 | onRest() { 74 | dispatch({ type: 'SET_PAUSE', pause: false }) 75 | if (typeof onAnimationEnd === 'function') onAnimationEnd() 76 | if (currentIndex === viewCount - 1) { 77 | if (typeof onSledEnd === 'function') onSledEnd() 78 | } 79 | }, 80 | }) 81 | }, [x, currentIndex, dimensionsUpdated, viewCount, cursor]) 82 | 83 | const bind = useDragGesture(set) 84 | 85 | const sliderStyles = useSliderStyles() 86 | const viewStyles = useViewStyles() 87 | 88 | return ( 89 | `translate3d(${x}px,0,0)`) 97 | : props.x.to((x) => `translate3d(0,${x}px,0)`), 98 | cursor: props.cursor, 99 | }} 100 | > 101 | { 102 | React.Children.map(children, child => ( 103 |
107 | {child} 108 |
109 | )) 110 | } 111 |
112 | ) 113 | } 114 | 115 | 116 | export default SledSprings 117 | -------------------------------------------------------------------------------- /sled/state/index.tsx: -------------------------------------------------------------------------------- 1 | // https://medium.com/simply/state-management-with-react-hooks-and-context-api-at-10-lines-of-code-baf6be8302c 2 | 3 | import React, { createContext, useContext, useReducer } from 'react' 4 | import { initialState, IState } from './types-defaults' 5 | import { reducer, ActionType } from './reducer' 6 | 7 | export const StateContext = createContext(initialState) 8 | 9 | interface IProps { 10 | children?: React.ReactNode 11 | } 12 | 13 | const StateProvider: React.FC = ({ children }) => { 14 | const [state, dispatch] = useReducer(reducer, initialState) 15 | return ( 16 | 17 | {children} 18 | 19 | ) 20 | } 21 | 22 | export { StateProvider } 23 | 24 | export const useStateContext = (): [IState, React.Dispatch] => { 25 | const [state, dispatch] = useContext(StateContext) 26 | return [state, dispatch] 27 | } 28 | 29 | export * from './types-defaults' -------------------------------------------------------------------------------- /sled/state/reducer.ts: -------------------------------------------------------------------------------- 1 | import clamp from '../utils/clamp' 2 | import { IState, TDirection, TDimension } from './types-defaults' 3 | import { SpringConfig } from 'react-spring' 4 | 5 | function getNext(currentIndex: number, slideSteps: number, rewind: boolean) { 6 | if (currentIndex === slideSteps - 1 && !rewind) { 7 | return clamp(currentIndex, 0, slideSteps - 1) 8 | } 9 | return (currentIndex + 1) % slideSteps 10 | } 11 | function getPrev(currentIndex: number, slideSteps: number, rewind: boolean) { 12 | if (currentIndex === 0 && !rewind) { 13 | return clamp(currentIndex, 0, slideSteps - 1) 14 | } 15 | return (currentIndex - 1 + slideSteps) % slideSteps 16 | } 17 | 18 | export type ActionType = 19 | | { type: 'NEXT' } 20 | | { type: 'PREV' } 21 | | { type: 'SELECT', index: number } 22 | | { type: 'SET_PAUSE', pause: boolean } 23 | | { type: 'SET_DIRECTION', direction: TDirection } 24 | | { type: 'SET_MOUSEOVER', pauseOnMouseOver: boolean } 25 | | { type: 'SET_VIEWCOUNT', count: number } 26 | | { type: 'SET_FOCUS', focus: boolean } 27 | | { type: 'SET_DIMENSIONS', dimensions: { width: number, height: number } } 28 | | { type: 'SET_DIMENSIONS_DOM', dimensionsDOM: { width: TDimension, height: TDimension } } 29 | | { type: 'SET_PROPORTION', proportion?: string } 30 | | { type: 'SET_DRAGGING', dragging: boolean } 31 | | { type: 'SET_DRAG_DISTANCE', dragDistance: string | number } 32 | | { type: 'SET_AUTOPLAY', autoPlayInterval?: number } 33 | | { type: 'SET_CONFIG', config: SpringConfig } 34 | | { type: 'SET_REWIND', rewind: boolean } 35 | | { type: 'SET_STOPONINTERACTION', stopOnInteraction: boolean } 36 | | { type: 'SET_SHOWELEMENTS', showElements: number } 37 | | { type: 'SET_SLIDEBY', slideBy: number } 38 | | { type: 'SET_SLIDESTEPS', slideSteps: number } 39 | | { type: 'SET_SLIDERSIZE', sliderSize: number } 40 | 41 | 42 | export function reducer(state: IState, action: ActionType): IState { 43 | switch (action.type) { 44 | case 'NEXT': return { 45 | ...state, 46 | currentIndex: getNext(state.currentIndex, state.slideSteps, state.rewind), 47 | prevIndex: state.currentIndex 48 | } 49 | case 'PREV': return { 50 | ...state, 51 | currentIndex: getPrev(state.currentIndex, state.slideSteps, state.rewind), 52 | prevIndex: state.currentIndex 53 | } 54 | case 'SELECT': return { 55 | ...state, 56 | currentIndex: clamp(action.index, 0, state.slideSteps - 1), 57 | prevIndex: state.currentIndex 58 | } 59 | case 'SET_DIRECTION': return { 60 | ...state, 61 | direction: action.direction 62 | } 63 | case 'SET_PAUSE': return { 64 | ...state, 65 | pause: action.pause 66 | } 67 | case 'SET_MOUSEOVER': return { 68 | ...state, 69 | pauseOnMouseOver: action.pauseOnMouseOver 70 | } 71 | case 'SET_VIEWCOUNT': return { 72 | ...state, 73 | viewCount: action.count 74 | } 75 | case 'SET_FOCUS': return { 76 | ...state, 77 | hasFocus: action.focus 78 | } 79 | case 'SET_DIMENSIONS': return { 80 | ...state, 81 | dimensions: action.dimensions 82 | } 83 | case 'SET_DIMENSIONS_DOM': return { 84 | ...state, 85 | dimensionsDOM: action.dimensionsDOM 86 | } 87 | case 'SET_PROPORTION': return { 88 | ...state, 89 | proportion: action.proportion 90 | } 91 | case 'SET_DRAGGING': return { 92 | ...state, 93 | dragging: action.dragging 94 | } 95 | case 'SET_DRAG_DISTANCE': return { 96 | ...state, 97 | dragDistance: action.dragDistance 98 | } 99 | case 'SET_AUTOPLAY': return { 100 | ...state, 101 | autoPlayInterval: action.autoPlayInterval 102 | } 103 | case 'SET_CONFIG': return { 104 | ...state, 105 | config: action.config 106 | } 107 | case 'SET_REWIND': return { 108 | ...state, 109 | rewind: action.rewind 110 | } 111 | case 'SET_STOPONINTERACTION': return { 112 | ...state, 113 | stopOnInteraction: action.stopOnInteraction 114 | } 115 | case 'SET_SLIDEBY': return { 116 | ...state, 117 | slideBy: action.slideBy 118 | } 119 | case 'SET_SLIDESTEPS': return { 120 | ...state, 121 | slideSteps: action.slideSteps 122 | } 123 | case 'SET_SHOWELEMENTS': return { 124 | ...state, 125 | showElements: action.showElements 126 | } 127 | case 'SET_SLIDERSIZE': return { 128 | ...state, 129 | sliderSize: action.sliderSize 130 | } 131 | default: return state 132 | } 133 | } 134 | 135 | -------------------------------------------------------------------------------- /sled/state/types-defaults.ts: -------------------------------------------------------------------------------- 1 | import { SpringConfig } from 'react-spring' 2 | import CSS from 'csstype' 3 | 4 | export type TRef = React.MutableRefObject 5 | 6 | export type TSelect = number | 'next' | 'prev' 7 | export type TDirection = 'horizontal' | 'vertical' | null 8 | export type TProportion = string | null 9 | export type TDimension = string | number | null 10 | 11 | export interface IInternalState { 12 | currentIndex: number 13 | prevIndex?: number 14 | viewCount: number 15 | slideSteps: number 16 | sliderSize: number 17 | hasFocus: boolean 18 | dimensions: { 19 | width: number 20 | height: number 21 | } 22 | dimensionsDOM: { 23 | width: TDimension 24 | height: TDimension 25 | } 26 | } 27 | 28 | interface IStateFromProps { 29 | width?: number | string 30 | height?: number | string 31 | direction?: TDirection 32 | select?: TSelect 33 | proportion?: TProportion 34 | showElements?: number, 35 | slideBy?: number, 36 | keyboard: boolean 37 | dragging: boolean 38 | dragDistance: number | string 39 | autoPlayInterval?: number 40 | pauseOnMouseOver: boolean 41 | rewind: boolean 42 | config: SpringConfig, 43 | stopOnInteraction: boolean 44 | pause: boolean 45 | onSledEnd?(): void 46 | onAnimationStart?(): void 47 | onAnimationEnd?(): void 48 | } 49 | 50 | export interface IState extends IInternalState, IStateFromProps { } 51 | 52 | export interface IViewsProps extends IStateFromProps { 53 | children?: React.ReactNode 54 | style?: CSS.Properties 55 | } 56 | 57 | const stateFromProps = { 58 | width: null, 59 | proportion: null, 60 | select: null, 61 | keyboard: true, 62 | dragging: true, 63 | dragDistance: '25ow', 64 | autoPlayInterval: null, 65 | config: { mass: 1, tension: 210, friction: 20, clamp: true }, 66 | pauseOnMouseOver: true, 67 | stopOnInteraction: false, 68 | rewind: false, 69 | pause: false, 70 | onSledEnd: null, 71 | onAnimationStart: null, 72 | onAnimationEnd: null 73 | } 74 | 75 | export const ViewsProps: IViewsProps = { 76 | ...stateFromProps 77 | } 78 | 79 | export const initialState: IState = { 80 | currentIndex: 0, 81 | prevIndex: undefined, 82 | pause: false, 83 | viewCount: 0, 84 | hasFocus: false, 85 | direction: 'horizontal', 86 | showElements: 1, 87 | slideBy: 1, 88 | slideSteps: 1, 89 | sliderSize: 0, 90 | dimensions: { width: 0, height: 0 }, 91 | dimensionsDOM: { width: 0, height: 0 }, 92 | ...stateFromProps 93 | } 94 | -------------------------------------------------------------------------------- /sled/utils/clamp.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * https://github.com/lodash/lodash/blob/master/clamp.js 3 | * Clamps `number` within the inclusive `lower` and `upper` bounds. 4 | */ 5 | 6 | function clamp(number: number, lower: number, upper: number) { 7 | number = +number 8 | lower = +lower 9 | upper = +upper 10 | lower = lower === lower ? lower : 0 11 | upper = upper === upper ? upper : 0 12 | if (number === number) { 13 | number = number <= upper ? number : upper 14 | number = number >= lower ? number : lower 15 | } 16 | return number 17 | } 18 | 19 | export default clamp 20 | -------------------------------------------------------------------------------- /sled/utils/debounce.ts: -------------------------------------------------------------------------------- 1 | // https://github.com/chodorowicz/ts-debounce/blob/master/src/index.ts 2 | 3 | export type Procedure = (...args: any[]) => void; 4 | 5 | export type Options = { 6 | isImmediate: boolean, 7 | } 8 | 9 | export function debounce( 10 | func: F, 11 | waitMilliseconds = 50, 12 | options: Options = { 13 | isImmediate: false 14 | }, 15 | ): F { 16 | let timeoutId: ReturnType | undefined 17 | 18 | return function (this: any, ...args: any[]) { 19 | const context = this 20 | 21 | const doLater = function () { 22 | timeoutId = undefined 23 | if (!options.isImmediate) { 24 | func.apply(context, args) 25 | } 26 | } 27 | 28 | const shouldCallNow = options.isImmediate && timeoutId === undefined 29 | 30 | if (timeoutId !== undefined) { 31 | clearTimeout(timeoutId) 32 | } 33 | 34 | timeoutId = setTimeout(doLater, waitMilliseconds) 35 | 36 | if (shouldCallNow) { 37 | func.apply(context, args) 38 | } 39 | } as any 40 | } 41 | -------------------------------------------------------------------------------- /sled/views.tsx: -------------------------------------------------------------------------------- 1 | import React, { useRef } from 'react' 2 | 3 | import Springs from './springs' 4 | import { IViewsProps, ViewsProps } from './state/types-defaults' 5 | 6 | import useDirection from './hooks/useDirection' 7 | import useDimensionsDOM from './hooks/useDimensionsDOM' 8 | import useDimensions from './hooks/useDimensions' 9 | import useProportion from './hooks/useProportion' 10 | import useKeyboard from './hooks/useKeyboard' 11 | import useDragging from './hooks/useDragging' 12 | import useMouseOver from './hooks/useMouseOver' 13 | import useSelect from './hooks/useSelect' 14 | import useFocus from './hooks/useFocus' 15 | import useViewCount from './hooks/useViewCount' 16 | import useAutoPlay from './hooks/useAutoPlay' 17 | import useConfig from './hooks/useConfig' 18 | import useRewind from './hooks/useRewind' 19 | import usePause from './hooks/usePause' 20 | import useStopOnInteraction from './hooks/useStopOnInteraction' 21 | import useShowElements from './hooks/useShowElements' 22 | import useSlideBy from './hooks/useSlideBy' 23 | import useSlideSteps from './hooks/useSlideSteps' 24 | import useSliderSize from './hooks/useSliderSize' 25 | import { useStateContext } from './state' 26 | import './index.css' 27 | 28 | const SledViews: React.FC = ({ 29 | children, 30 | style, 31 | width, 32 | height, 33 | proportion, 34 | direction, 35 | select, 36 | slideBy, 37 | showElements, 38 | keyboard, 39 | dragging, 40 | dragDistance, 41 | autoPlayInterval, 42 | config, 43 | pauseOnMouseOver, 44 | stopOnInteraction, 45 | rewind, 46 | pause, 47 | onSledEnd, 48 | onAnimationStart, 49 | onAnimationEnd 50 | }) => { 51 | 52 | const viewsRef = useRef() 53 | const [{ dimensions, dimensionsDOM }] = useStateContext() 54 | 55 | 56 | useProportion(width, height, proportion) 57 | useDimensionsDOM(width, height) 58 | useDimensions(viewsRef) 59 | 60 | useDirection(direction) 61 | 62 | useFocus(viewsRef) 63 | useViewCount(children) 64 | useRewind(rewind) 65 | usePause(pause) 66 | 67 | useShowElements(showElements) 68 | useSlideBy(slideBy) 69 | useSliderSize() 70 | useSlideSteps() 71 | 72 | useStopOnInteraction(stopOnInteraction) 73 | useMouseOver(pauseOnMouseOver, viewsRef) 74 | useSelect(select) 75 | useKeyboard(keyboard) 76 | useDragging(dragging, dragDistance) 77 | useAutoPlay(autoPlayInterval) 78 | 79 | useConfig(config) 80 | 81 | return ( 82 |
91 |
95 | 100 | {children} 101 | 102 |
103 |
104 | ) 105 | } 106 | 107 | SledViews.defaultProps = ViewsProps 108 | 109 | export default SledViews 110 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es5", 4 | "lib": [ 5 | "dom", 6 | "dom.iterable", 7 | "esnext" 8 | ], 9 | "allowJs": true, 10 | "skipLibCheck": true, 11 | "strict": false, 12 | "forceConsistentCasingInFileNames": true, 13 | "noEmit": true, 14 | "esModuleInterop": true, 15 | "module": "esnext", 16 | "moduleResolution": "node", 17 | "resolveJsonModule": true, 18 | "isolatedModules": true, 19 | "jsx": "preserve" 20 | }, 21 | "exclude": [ 22 | "node_modules" 23 | ], 24 | "include": [ 25 | "next-env.d.ts", 26 | "**/*.ts", 27 | "**/*.tsx" 28 | ] 29 | } -------------------------------------------------------------------------------- /tsconfig.rollup.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "outDir": "dist", 4 | "module": "esnext", 5 | "target": "es5", 6 | "lib": [ 7 | "es6", 8 | "dom", 9 | "es2016", 10 | "es2017" 11 | ], 12 | "sourceMap": true, 13 | "allowJs": false, 14 | "jsx": "react", 15 | "declaration": true, 16 | "moduleResolution": "node", 17 | "allowSyntheticDefaultImports": true, 18 | "esModuleInterop": true 19 | }, 20 | "include": [ 21 | "sled/**/*" 22 | ], 23 | "exclude": [ 24 | "node_modules", 25 | "dist" 26 | ] 27 | } --------------------------------------------------------------------------------