├── demo ├── p.gif ├── s.gif ├── arch.png ├── pathbuilder.png ├── pathslider.png ├── colorpicker │ ├── blank.gif │ ├── slider.png │ ├── select2.png │ ├── custom_hex.png │ ├── custom_hsb_b.png │ ├── custom_hsb_h.png │ ├── custom_hsb_s.png │ ├── custom_indic.gif │ ├── custom_rgb_b.png │ ├── custom_rgb_g.png │ ├── custom_rgb_r.png │ ├── custom_submit.png │ ├── colorpicker_overlay.png │ ├── colorpicker_select.gif │ ├── custom_background.png │ ├── colorpicker.css │ └── colorpicker.js ├── wiki │ ├── coordinate-system.png │ └── pathslider-points.png ├── demo.css └── demo.js ├── images ├── chrome1.png └── chrome2.png ├── .gitattributes ├── LICENSE ├── package.json ├── Gruntfile.js ├── .gitignore ├── index.html ├── README.md ├── dist ├── pathslider.min.css ├── jquery.pathslider.min.js ├── pathslider.css └── jquery.pathslider.js ├── css └── pathslider.css └── js ├── jquery.pathslider.builder.js └── jquery.pathslider.js /demo/p.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/p.gif -------------------------------------------------------------------------------- /demo/s.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/s.gif -------------------------------------------------------------------------------- /demo/arch.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/arch.png -------------------------------------------------------------------------------- /demo/pathbuilder.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/pathbuilder.png -------------------------------------------------------------------------------- /demo/pathslider.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/pathslider.png -------------------------------------------------------------------------------- /images/chrome1.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/images/chrome1.png -------------------------------------------------------------------------------- /images/chrome2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/images/chrome2.png -------------------------------------------------------------------------------- /demo/colorpicker/blank.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/blank.gif -------------------------------------------------------------------------------- /demo/colorpicker/slider.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/slider.png -------------------------------------------------------------------------------- /demo/colorpicker/select2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/select2.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_hex.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_hex.png -------------------------------------------------------------------------------- /demo/wiki/coordinate-system.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/wiki/coordinate-system.png -------------------------------------------------------------------------------- /demo/wiki/pathslider-points.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/wiki/pathslider-points.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_hsb_b.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_hsb_b.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_hsb_h.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_hsb_h.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_hsb_s.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_hsb_s.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_indic.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_indic.gif -------------------------------------------------------------------------------- /demo/colorpicker/custom_rgb_b.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_rgb_b.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_rgb_g.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_rgb_g.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_rgb_r.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_rgb_r.png -------------------------------------------------------------------------------- /demo/colorpicker/custom_submit.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_submit.png -------------------------------------------------------------------------------- /demo/colorpicker/colorpicker_overlay.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/colorpicker_overlay.png -------------------------------------------------------------------------------- /demo/colorpicker/colorpicker_select.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/colorpicker_select.gif -------------------------------------------------------------------------------- /demo/colorpicker/custom_background.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Mottie/Pathslider/HEAD/demo/colorpicker/custom_background.png -------------------------------------------------------------------------------- /.gitattributes: -------------------------------------------------------------------------------- 1 | # Auto detect text files and perform LF normalization 2 | * text=auto 3 | 4 | # Custom for Visual Studio 5 | *.cs diff=csharp 6 | *.sln merge=union 7 | *.csproj merge=union 8 | *.vbproj merge=union 9 | *.fsproj merge=union 10 | *.dbproj merge=union 11 | 12 | # Standard to msysgit 13 | *.doc diff=astextplain 14 | *.DOC diff=astextplain 15 | *.docx diff=astextplain 16 | *.DOCX diff=astextplain 17 | *.dot diff=astextplain 18 | *.DOT diff=astextplain 19 | *.pdf diff=astextplain 20 | *.PDF diff=astextplain 21 | *.rtf diff=astextplain 22 | *.RTF diff=astextplain 23 | *.md diss=astextplain -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright 2016 Rob Garrison 2 | 3 | MIT License 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy of 6 | this software and associated documentation files (the "Software"), to deal in 7 | the Software without restriction, including without limitation the rights to use, 8 | copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the 9 | Software, and to permit persons to whom the Software is furnished to do so, 10 | subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS 17 | FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR 18 | COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER 19 | IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 20 | CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "pathslider", 3 | "title": "Pathslider", 4 | "description": "A jQuery numerical slider that follows a bezier path", 5 | "version": "1.0.0-beta", 6 | "author": { 7 | "name": "Rob Garrison", 8 | "url": "https://github.com/Mottie", 9 | "email": "wowmotty@gmail.com" 10 | }, 11 | "license": "MIT", 12 | "dependencies": { 13 | "jquery": ">=1.4.4" 14 | }, 15 | "keywords": [ 16 | "slider", 17 | "number", 18 | "bezier", 19 | "path", 20 | "jquery-plugin" 21 | ], 22 | "homepage": "https://github.com/Mottie/Pathslider", 23 | "demo": "http://mottie.github.io/Pathslider/", 24 | "docs": "https://github.com/Mottie/Pathslider/wiki", 25 | "bugs": "https://github.com/Mottie/Pathslider/issues", 26 | "main": "dist/jquery.pathslider.js", 27 | "files": [ 28 | "dist/*" 29 | ], 30 | "repository": { 31 | "type": "git", 32 | "url": "git://github.com/Mottie/pathslider.git" 33 | }, 34 | "devDependencies": { 35 | "grunt": "^1.0.1", 36 | "grunt-cli": "^1.1.0", 37 | "grunt-contrib-clean": "^1.0.0", 38 | "grunt-contrib-copy": "^1.0.0", 39 | "grunt-contrib-cssmin": "^1.0.1", 40 | "grunt-contrib-jshint": "^1.0.0", 41 | "grunt-contrib-uglify": "^2.0.0" 42 | } 43 | } 44 | -------------------------------------------------------------------------------- /Gruntfile.js: -------------------------------------------------------------------------------- 1 | /*global module:false */ 2 | module.exports = function(grunt) { 3 | 'use strict'; 4 | 5 | // Project configuration. 6 | grunt.initConfig({ 7 | pkg: grunt.file.readJSON('package.json'), 8 | 9 | clean: { 10 | dist: { 11 | src: ['dist/*', 'dist/**/*'] 12 | } 13 | }, 14 | 15 | copy: { 16 | dist: { 17 | files : [{ 18 | expand: true, 19 | dot: true, 20 | flatten: true, 21 | src: [ 'css/pathslider.css', 'js/jquery.pathslider.js' ], 22 | dest: 'dist/' 23 | }] 24 | } 25 | }, 26 | 27 | jshint: { 28 | core: { 29 | options: { 30 | "jquery": true, 31 | "browser": true 32 | }, 33 | src: [ 'js/jquery.*.js' ] 34 | } 35 | }, 36 | 37 | cssmin: { 38 | target: { 39 | files: [{ 40 | expand: true, 41 | flatten: true, 42 | src: ['css/*.css'], 43 | dest: 'dist/', 44 | ext: '.min.css' 45 | }] 46 | } 47 | }, 48 | 49 | uglify: { 50 | options: { 51 | preserveComments: function( node, comment ) { 52 | return /^!/.test( comment.value ); 53 | }, 54 | report: 'gzip' 55 | }, 56 | dist: { 57 | files: [{ 58 | expand: true, 59 | cwd: '', 60 | src: [ 'js/jquery*.js', '!js/*builder.js' ], 61 | dest: 'dist/', 62 | ext: '.min.js', 63 | extDot: 'last', 64 | flatten: true 65 | }] 66 | } 67 | }, 68 | 69 | watch: { 70 | scripts: { 71 | files: ['js/*.js'], 72 | tasks: ['build'] 73 | } 74 | } 75 | 76 | }); 77 | 78 | grunt.loadNpmTasks('grunt-contrib-clean'); 79 | grunt.loadNpmTasks('grunt-contrib-copy'); 80 | grunt.loadNpmTasks('grunt-contrib-cssmin'); 81 | grunt.loadNpmTasks('grunt-contrib-jshint'); 82 | grunt.loadNpmTasks('grunt-contrib-uglify'); 83 | 84 | // Default task. 85 | grunt.registerTask('default', [ 86 | 'clean', 87 | 'jshint', 88 | 'copy', 89 | 'cssmin', 90 | 'uglify' 91 | ]); 92 | 93 | }; 94 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | ################# 2 | ## Eclipse 3 | ################# 4 | 5 | *.pydevproject 6 | .project 7 | .metadata 8 | bin/ 9 | tmp/ 10 | *_test*.* 11 | *_idea*.* 12 | *.tmp 13 | *.bak 14 | *.swp 15 | *~.nib 16 | local.properties 17 | .classpath 18 | .settings/ 19 | .loadpath 20 | 21 | # External tool builders 22 | .externalToolBuilders/ 23 | 24 | # Locally stored "Eclipse launch configurations" 25 | *.launch 26 | 27 | # CDT-specific 28 | .cproject 29 | 30 | # PDT-specific 31 | .buildpath 32 | 33 | 34 | ################# 35 | ## Visual Studio 36 | ################# 37 | 38 | ## Ignore Visual Studio temporary files, build results, and 39 | ## files generated by popular Visual Studio add-ons. 40 | 41 | # User-specific files 42 | *.suo 43 | *.user 44 | *.sln.docstates 45 | 46 | # Build results 47 | [Dd]ebug/ 48 | [Rr]elease/ 49 | node_modules/ 50 | *_i.c 51 | *_p.c 52 | *.ilk 53 | *.log 54 | *.meta 55 | *.obj 56 | *.pch 57 | *.pdb 58 | *.pgc 59 | *.pgd 60 | *.rsp 61 | *.sbr 62 | *.tlb 63 | *.tli 64 | *.tlh 65 | *.tmp 66 | *.vspscc 67 | .builds 68 | *.dotCover 69 | 70 | ## TODO: If you have NuGet Package Restore enabled, uncomment this 71 | #packages/ 72 | 73 | # Visual C++ cache files 74 | ipch/ 75 | *.aps 76 | *.ncb 77 | *.opensdf 78 | *.sdf 79 | 80 | # Visual Studio profiler 81 | *.psess 82 | *.vsp 83 | 84 | # ReSharper is a .NET coding add-in 85 | _ReSharper* 86 | 87 | # Installshield output folder 88 | [Ee]xpress 89 | 90 | # DocProject is a documentation generator add-in 91 | DocProject/buildhelp/ 92 | DocProject/Help/*.HxT 93 | DocProject/Help/*.HxC 94 | DocProject/Help/*.hhc 95 | DocProject/Help/*.hhk 96 | DocProject/Help/*.hhp 97 | DocProject/Help/Html2 98 | DocProject/Help/html 99 | 100 | # Click-Once directory 101 | publish 102 | 103 | # Others 104 | [Bb]in 105 | [Oo]bj 106 | sql 107 | TestResults 108 | *.Cache 109 | ClientBin 110 | stylecop.* 111 | ~$* 112 | *.dbmdl 113 | Generated_Code #added for RIA/Silverlight projects 114 | 115 | # Backup & report files from converting an old project file to a newer 116 | # Visual Studio version. Backup files are not needed, because we have git ;-) 117 | _UpgradeReport_Files/ 118 | Backup*/ 119 | UpgradeLog*.XML 120 | 121 | 122 | 123 | ############ 124 | ## Windows 125 | ############ 126 | 127 | # Windows image file caches 128 | Thumbs.db 129 | 130 | # Folder config file 131 | Desktop.ini 132 | 133 | 134 | ############# 135 | ## Python 136 | ############# 137 | 138 | *.py[co] 139 | 140 | # Packages 141 | *.egg 142 | *.egg-info 143 | build 144 | eggs 145 | parts 146 | bin 147 | var 148 | sdist 149 | develop-eggs 150 | .installed.cfg 151 | 152 | # Installer logs 153 | pip-log.txt 154 | 155 | # Unit test / coverage reports 156 | .coverage 157 | .tox 158 | 159 | #Translations 160 | *.mo 161 | 162 | #Mr Developer 163 | .mr.developer.cfg 164 | 165 | # Mac crap 166 | .DS_Store 167 | -------------------------------------------------------------------------------- /demo/colorpicker/colorpicker.css: -------------------------------------------------------------------------------- 1 | .colorpicker { 2 | width: 356px; 3 | height: 176px; 4 | overflow: hidden; 5 | position: absolute; 6 | background: url(custom_background.png); 7 | font-family: Arial, Helvetica, sans-serif; 8 | display: none; 9 | } 10 | .colorpicker_color { 11 | width: 150px; 12 | height: 150px; 13 | left: 14px; 14 | top: 13px; 15 | position: absolute; 16 | background: #f00; 17 | overflow: hidden; 18 | cursor: crosshair; 19 | } 20 | .colorpicker_color div { 21 | position: absolute; 22 | top: 0; 23 | left: 0; 24 | width: 150px; 25 | height: 150px; 26 | background: url(colorpicker_overlay.png); 27 | } 28 | .colorpicker_color div div { 29 | position: absolute; 30 | top: 0; 31 | left: 0; 32 | width: 11px; 33 | height: 11px; 34 | overflow: hidden; 35 | background: url(colorpicker_select.gif); 36 | margin: -5px 0 0 -5px; 37 | } 38 | .colorpicker_hue { 39 | position: absolute; 40 | top: 13px; 41 | left: 171px; 42 | width: 35px; 43 | height: 150px; 44 | cursor: n-resize; 45 | } 46 | .colorpicker_hue div { 47 | position: absolute; 48 | width: 35px; 49 | height: 9px; 50 | overflow: hidden; 51 | background: url(custom_indic.gif) left top; 52 | margin: -4px 0 0 0; 53 | left: 0px; 54 | } 55 | .colorpicker_new_color { 56 | position: absolute; 57 | width: 60px; 58 | height: 30px; 59 | left: 213px; 60 | top: 13px; 61 | background: #f00; 62 | } 63 | .colorpicker_current_color { 64 | position: absolute; 65 | width: 60px; 66 | height: 30px; 67 | left: 283px; 68 | top: 13px; 69 | background: #f00; 70 | } 71 | .colorpicker input { 72 | background-color: transparent; 73 | border: 1px solid transparent; 74 | position: absolute; 75 | font-size: 10px; 76 | font-family: Arial, Helvetica, sans-serif; 77 | color: #898989; 78 | top: 4px; 79 | right: 11px; 80 | text-align: right; 81 | margin: 0; 82 | padding: 0; 83 | height: 11px; 84 | } 85 | .colorpicker_hex { 86 | position: absolute; 87 | width: 72px; 88 | height: 22px; 89 | background: url(custom_hex.png) top; 90 | left: 212px; 91 | top: 142px; 92 | } 93 | .colorpicker_hex input { 94 | right: 6px; 95 | } 96 | .colorpicker_field { 97 | height: 22px; 98 | width: 62px; 99 | background-position: top; 100 | position: absolute; 101 | } 102 | .colorpicker_field span { 103 | position: absolute; 104 | width: 12px; 105 | height: 22px; 106 | overflow: hidden; 107 | top: 0; 108 | right: 0; 109 | cursor: n-resize; 110 | } 111 | .colorpicker_rgb_r { 112 | background-image: url(custom_rgb_r.png); 113 | top: 52px; 114 | left: 212px; 115 | } 116 | .colorpicker_rgb_g { 117 | background-image: url(custom_rgb_g.png); 118 | top: 82px; 119 | left: 212px; 120 | } 121 | .colorpicker_rgb_b { 122 | background-image: url(custom_rgb_b.png); 123 | top: 112px; 124 | left: 212px; 125 | } 126 | .colorpicker_hsb_h { 127 | background-image: url(custom_hsb_h.png); 128 | top: 52px; 129 | left: 282px; 130 | } 131 | .colorpicker_hsb_s { 132 | background-image: url(custom_hsb_s.png); 133 | top: 82px; 134 | left: 282px; 135 | } 136 | .colorpicker_hsb_b { 137 | background-image: url(custom_hsb_b.png); 138 | top: 112px; 139 | left: 282px; 140 | } 141 | .colorpicker_submit { 142 | position: absolute; 143 | width: 22px; 144 | height: 22px; 145 | background: url(custom_submit.png) top; 146 | left: 322px; 147 | top: 142px; 148 | overflow: hidden; 149 | } 150 | .colorpicker_focus { 151 | background-position: center; 152 | } 153 | .colorpicker_hex.colorpicker_focus { 154 | background-position: bottom; 155 | } 156 | .colorpicker_submit.colorpicker_focus { 157 | background-position: bottom; 158 | } 159 | .colorpicker_slider { 160 | background-position: bottom; 161 | } 162 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | jQuery Pathslider 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 41 | 95 | 96 | 97 | 98 |
99 | 100 |
101 |

jQuery Pathslider

102 | Home 103 | Builder 104 |
105 | 106 |
107 | 108 |
109 |
Basic Curve
110 |
111 |
112 | 113 |
114 |
"P" Curve
115 |
116 |
117 | 118 |
119 |
"S" Curve
120 |
121 |
122 | 123 |
124 | 125 |
126 | 127 |
128 | 129 | 141 | 142 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | A jQuery numerical slider that follows a bezier path. 2 | 3 | [![devDependency Status][david-dev-image]][david-dev-url] [![MIT][license-image]][license-url] 4 | 5 | ## Features 6 | 7 | * Numerical slider similar to the [jQuery UI Slider](http://jqueryui.com/demos/slider/) - currently it only works with percentages (0-100%) 8 | * This slider follows a bezier path. The parameters are set using the pathslider builder. 9 | * Designed to work in older browsers (no canvas support), just add the path as a background image. 10 | * Get and set the slider position dynamically. 11 | * Slider handle (grip) can be set to rotate along with the angle of the curve (uses css3). 12 | * Callback events are available: create, update, start, slide, change, and stop. 13 | * The slider itself works in all browsers: IE6+, Opera, Chrome, Firefox and Safari (known issues below) 14 | * [Pathslider demo](http://mottie.github.com/Pathslider/index.html). 15 | * [Pathslider Builder](http://mottie.github.com/Pathslider/builder.html). 16 | 17 | ## Documentation 18 | 19 | Wiki: 20 | [Home](https://github.com/Mottie/Pathslider/wiki/Home) | 21 | [FAQ](https://github.com/Mottie/Pathslider/wiki/FAQ) | 22 | [Setup](https://github.com/Mottie/Pathslider/wiki/Setup) | 23 | [Options](https://github.com/Mottie/Pathslider/wiki/Options) | 24 | [Events](https://github.com/Mottie/Pathslider/wiki/Events) | 25 | [Theme](https://github.com/Mottie/Pathslider/wiki/Theme) | 26 | [Change](https://github.com/Mottie/Pathslider/wiki/Change) 27 | 28 | ## To Do: 29 | 30 | * Add min, max and step options to use values, other than zero to one hundred percent, with the slider. 31 | * Add enable, disable, destroy methods. 32 | * Smooth out sliding by improving grip position calculation 33 | * The grip jumps around quite a bit with some settings 34 | * This can be minimized by adjusting the tolerance & range settings, but it needs a better method. 35 | * Add more default handle (grip) styles in the css. 36 | * Add ability to make a gradient/patterned stroke style of the curve; I'm not sure it will follow the path of the curve or just be a static background. Needs more testing! 37 | * Add keyboard control to the slider when it has focus - arrows, page up/down, home and end. 38 | * Add a circular path 39 | * I don't plan on making it go continuously around in a circle. It will have a starting and stopping point. 40 | * If you need a continuous circle slider, check out [this one](http://www.eleqtriq.com/2009/12/javascriptdialcontrol/), and [this one](http://www.baijs.nl/tinycircleslider/). 41 | * Combine multiple beziers to make extended shapes - my math skills are a bit lacking so this may take a while, unless someone out there is willing to help ;) 42 | * Consider switching from using canvas to svg. 43 | 44 | ## Dependencies 45 | 46 | * jQuery 1.4.4+ 47 | * A browser that supports canvas (needed for the builder; optional for the pathslider plugin itself). 48 | * A browser that supports css3 transforms (necessary to rotate the grip). 49 | 50 | ## Known Bugs 51 | 52 | * The grip/handle will work if the loop folds back on itself - try these points to see an example: 75,200,200,-125,-200,0,225,200 53 | * The S-Curve demo seems to get stuck near 100% but only in Firefox and on the demo page. It works fine when isolated. The grip isn't staying centered under the cursor like it does in Chrome. 54 | * Safari for Windows likes to start with the grip in the upper left corner of the slider box... most of the time when I hover over it, it jumps to where it should be... wow that is so weird. Also, the "S" curve demo doesn't want to work at all... 55 | 56 | [david-dev-url]: https://david-dm.org/Mottie/pathslider?type=dev 57 | [david-dev-image]: https://img.shields.io/david/dev/Mottie/pathslider.svg 58 | [license-url]: https://github.com/Mottie/Pathslider/blob/master/LICENSE 59 | [license-image]: https://img.shields.io/badge/license-MIT-blue.svg 60 | 61 | ## Change Log 62 | 63 | ### Version 1.0.0-alpha (8/19/2016) (mis-released as beta) 64 | 65 | * Core: 66 | * Use window load event for jQuery v3.0+ compatibility. 67 | * Add `drawCanvas` callback & update curve color options. Fixes [issue #8](https://github.com/Mottie/Pathslider/issues/8). 68 | * Add `finishCurve` function. For use inside of the `drawCanvas` callback. 69 | * Add `redraw` method. See [issue #8](https://github.com/Mottie/Pathslider/issues/8). 70 | * Clean up CSS & use image URI. 71 | * Builder: 72 | * Switch to using `drawCanvas` callback. 73 | * Use `finishCurve` function. 74 | * Add curve shift controls. 75 | * Docs & Readme: 76 | * Update libraries & colors. 77 | * Make MIT license more prominent. 78 | * Optimize png images. 79 | * Extras: 80 | * Add `.git` files. 81 | * Remove BOM from all files. 82 | * Add grunt build script & dist folder. 83 | 84 | ### Version 0.9.1 alpha (12/7/2011) 85 | 86 | * Added touch device compatibility. 87 | * Changed the grip "data-degree" attribute to "data-angle". This attribute contains the angle of transformation to rotate the grip. 88 | * Changed the grip "data-position" attribute to "data-percent". This attribute contains the current percentage distance along the curve of the grip. 89 | * Added some images for the wiki documentation. 90 | 91 | ### Version 0.9 alpha (12/5/2011) 92 | 93 | * Initial commit 94 | -------------------------------------------------------------------------------- /dist/pathslider.min.css: -------------------------------------------------------------------------------- 1 | #slider{width:200px;height:200px}.pathslider-grip{width:30px;height:15px;position:absolute;top:0;left:0;background:#ddd;font-size:1px;z-index:10;cursor:move;border:1px solid;border-radius:7px;-webkit-box-shadow:0 1px 3px rgba(000,000,000,.5),inset 0 0 1px rgba(255,255,255,.6);box-shadow:0 1px 3px rgba(000,000,000,.5),inset 0 0 1px rgba(255,255,255,.6)}.pathslider-grip.sliding,.pathslider-grip:hover{border-color:#8ce;-webkit-box-shadow:0 0 15px #8ce;box-shadow:0 0 15px #8ce}.pathslider{position:relative}.silver{border-color:#949494;background-color:#ebebeb;background-image:-webkit-linear-gradient(top,#fff 0,#ebebeb 50%,#dbdbdb 50%,#b5b5b5);background-image:linear-gradient(top,#fff 0,#ebebeb 50%,#dbdbdb 50%,#b5b5b5)}.black{border-color:#000 2 | background-color: #3b3b3b;background-image:-webkit-linear-gradient(top,#a3a3a3 0,#3b3b3b 50%,#242424 50%,#000);background-image:linear-gradient(top,#a3a3a3 0,#3b3b3b 50%,#242424 50%,#000)}.pathslider-grip.chrome1{width:18px;height:18px;border:0;border-radius:0;background:url() center center no-repeat;-webkit-box-shadow:0 0 0;box-shadow:0 0 0}.pathslider-grip.chrome2{width:57px;height:35px;border:0;border-radius:0;background:url() center center no-repeat;-webkit-box-shadow:0 0 0;box-shadow:0 0 0} -------------------------------------------------------------------------------- /dist/jquery.pathslider.min.js: -------------------------------------------------------------------------------- 1 | !function(a){a.pathslider=function(b,c){var d,e=this;e.$el=a(b).addClass("pathslider"),e.el=b,e.$el.data("pathslider",e),e.init=function(){var b;e.options=d=a.extend(!0,{},a.pathslider.defaults,c),b=document.createElement("canvas"),e.hasCanvas=!(!b.getContext||!b.getContext("2d")),e.hasTouch=document.hasOwnProperty("ontouchend"),e.$grip=a("
").appendTo(e.$el),e.points=[],e.pointsxy=[],e.arrayX=[],e.arrayY=[],e.arrayP=[],e.rad2deg=180/Math.PI,e.sliding=!1,e.lastPercent=e.percent=d.value,a.each("create update start slide change stop".split(" "),function(b,c){a.isFunction(d[c])&&e.$el.bind(c+".pathslider",d[c])}),a(document).bind(e.hasTouch?"touchend.pathslider touchcancel.pathslider":"mouseup.pathslider mouseleave.pathslider",function(a){e.sliding&&(e.$el.trigger("stop.pathslider",[e]),e.lastPercent!==e.percent&&(e.lastPercent=e.percent,e.$el.trigger("change.pathslider",[e]))),e.$grip.removeClass("sliding"),e.sliding=!1}).bind((e.hasTouch?"touchmove":"mousemove")+".pathslider",function(a){e.sliding&&e.setSlider(e.findPos(a),null,!0)}),a(window).bind("resize.pathslider",function(){e.update()}).bind("load",function(){e.sliderDim[0]=e.$el.offset().left,e.sliderDim[1]=e.$el.offset().top}),e.$grip.bind((e.hasTouch?"touchstart":"mousedown")+".pathslider",function(b){return e.sliding=!0,a(this).addClass("sliding"),e.$el.trigger("start.pathslider",[e]),!1}).bind("click",function(){return!1}),e.redraw(),e.$el.trigger("create.pathslider",[e])},e.update=function(){e.$grip.attr("class","pathslider-grip "+d.gripClass),e.ctx&&e.ctx.clearRect(0,0,e.sliderDim[2],e.sliderDim[3]),e.sliderDim=[e.$el.offset().left,e.$el.offset().top,e.$el.width(),e.$el.height()];var b=window.getComputedStyle(e.$grip[0]);e.gripCenter=[parseInt(b.width,10)/2,parseInt(b.height,10)/2],e.dataPoints=d.dataPoints,e.makeArray();var c=a.inArray(e.percent,e.arrayP);e.position=c===-1?Math.round(e.percent/100*e.dataPoints):c,e.setSlider(e.percent,null,!0),e.hasCanvas&&d.useCanvas&&e.drawCurve()},e.setSlider=function(a,b,c){if(!isNaN(a)){a=parseFloat(a,10),a=a>100?100:a<0?0:a;var f,g,h=e.calcBezier(a/100,e.pointsxy),i=a-2>0?e.calcBezier((a-2)/100,e.pointsxy):h,j=a+2<100?e.calcBezier((a+2)/100,e.pointsxy):h,k=j[0]-i[0]===0?90:(j[1]-i[1])/(j[0]-i[0]);e.angle=parseInt(Math.atan(k)*e.rad2deg,10),g="rotate("+e.angle+"deg)",f=d.rotateGrip?{"-webkit-transform":g,transform:g}:{},f.left=h[0]-e.gripCenter[0],f.top=h[1]-e.gripCenter[1],e.$grip.attr({"data-angle":e.angle,"data-percent":a}).css(f),e.percent=a,(a===e.lastPercent||e.sliding)&&c||e.$el.trigger("change.pathslider",[e])}"function"==typeof b&&b(e)},e.mousePos=function(a){return[(a.originalEvent.touches?a.originalEvent.touches[0].pageX:a.pageX)-e.sliderDim[0],(a.originalEvent.touches?a.originalEvent.touches[0].pageY:a.pageY)-e.sliderDim[1]]},e.findPos=function(a){var b,c,f,g,h=[],i=[],j=e.position,k=parseInt(d.tolerance+1,10)||2,l=parseInt(d.range,10)||e.gripCenter[0],m=e.mousePos(a);for(b=0;b1)return e.returnPos(h[0]);if(1===i.length&&h.length>1)return e.returnPos(i[0])}return e.returnPos(j)},e.returnPos=function(a){var b=e.position===a;return e.percent=e.arrayP[a],e.position=a,b||(e.hasCanvas&&d.useCanvas&&e.drawCurve(),e.$el.trigger("slide.pathslider",[e])),e.percent},e.makeArray=function(){var a,b,c=e.pointsxy,d=e.dataPoints;for(a=0;a').appendTo(e.$el),e.$canvas=e.$el.find("canvas").attr({width:e.sliderDim[2],height:e.sliderDim[3]}),e.canvas=e.$canvas[0],e.ctx=e.canvas.getContext("2d")),b=e.ctx,b.clearRect(0,0,e.sliderDim[2],e.sliderDim[3]),b.lineCap=d.curve.cap,b.lineJoin=d.curve.cap,b.lineWidth=d.curve.width,a.isArray(d.curve.color)?(c=b.createLinearGradient(g[0],g[1],g[6],g[7]),f=e.percent/100,c.addColorStop(0,d.curve.color[0]),c.addColorStop(f,d.curve.color[0]),f+.01<=1&&(f+=.01),c.addColorStop(f,d.curve.color[1]),c.addColorStop(1,d.curve.color[1]),b.strokeStyle=c):b.strokeStyle=d.curve.color,f=!0,"function"==typeof d.drawCanvas&&(f=d.drawCanvas(e,b,g)!==!1,b=e.ctx),f===!0&&e.finishCurve(b,g)},e.finishCurve=function(a,b){a=a||e.ctx,b=b||e.pointsxy,a.beginPath(),a.moveTo(b[0],b[1]),a.bezierCurveTo(b[2],b[3],b[4],b[5],b[6],b[7]),a.stroke()},e.init()},a.pathslider.defaults={gripClass:"",rotateGrip:!0,useCanvas:!0,curve:{width:4,color:"#333",cap:"round"},points:[0,50,50,-50,-50,-50,250,50],value:50,dataPoints:100,tolerance:3,range:30},a.fn.pathslider=function(b,c){return this.each(function(){var d,e=a(this).data("pathslider");if((typeof b).match("object|undefined")){if(e)return e.redraw();new a.pathslider(this,b)}else/\d/.test(b)&&!isNaN(b)&&e&&(d="number"==typeof b?b:parseInt(a.trim(b),10),d>=0&&d<=100&&e.setSlider(d,c))})},a.fn.getpathslider=function(){return this.data("pathslider")}}(jQuery); -------------------------------------------------------------------------------- /demo/demo.css: -------------------------------------------------------------------------------- 1 | html,body { margin: 0; height: 100%; } 2 | 3 | body { background: #ddd; color: #333; } 4 | a:link { color: #ddd; text-decoration: none; } 5 | a:hover { color: #fff; } 6 | a:visited, a:active { color: #ccc; } 7 | 8 | h1.hide { display: none; } 9 | .oldie h1.hide { display: block; margin: 50px; color: #000; } 10 | h1.hide, .oldie .builder .wrapper, .oldie .builder .footer { display: none; } 11 | 12 | header, .header { 13 | min-height: 100px; 14 | background: #789; 15 | background: -moz-radial-gradient(center top, ellipse farthest-corner, #789, #677 40%); 16 | background: -webkit-radial-gradient(center top, ellipse farthest-corner, #789, #677 40%); 17 | background: -o-radial-gradient(center top, ellipse farthest-corner, #789, #677 40%); 18 | background: -ms-radial-gradient(center top, ellipse farthest-corner, #789, #677 40%); 19 | box-shadow: inset 0 0 10px #567; 20 | } 21 | 22 | h1 { 23 | color: #ddd; 24 | margin: 0; 25 | text-align: center; 26 | font-size: 3.5em; 27 | font-family: 'Paytone One', sans-serif; 28 | text-shadow: #555 1px 1px, #555 2px 2px; 29 | } 30 | 31 | header a, .header a { 32 | position: absolute; 33 | top: 25px; 34 | width: 50px; 35 | height: 50px; 36 | display: block; 37 | text-indent: -999em; 38 | } 39 | header a.current, .header a.current { 40 | -webkit-box-shadow: inset 0 0 20px #59b; 41 | -moz-box-shadow: inset 0 0 20px #59b; 42 | box-shadow: inset 0 0 20px #59b; 43 | border-radius: 15px; 44 | -moz-border-radius: 15px; 45 | -webkit-border-radius: 15px; 46 | } 47 | 48 | a.home { 49 | left: 20px; 50 | background: url(../demo/pathslider.png) center center no-repeat; 51 | } 52 | a.builder { 53 | left: 75px; 54 | background: url(../demo/pathbuilder.png) center center no-repeat; 55 | } 56 | 57 | .wrapper { 58 | min-height: 100%; 59 | height: auto !important; 60 | height: 100%; 61 | margin: 0 auto -100px; /* the bottom margin is the negative value of the footer's height */ 62 | } 63 | 64 | .content { 65 | width: 50%; 66 | margin: 10px auto; 67 | } 68 | 69 | .builder .content { 70 | width: 80%; 71 | 72 | } 73 | 74 | #slider { 75 | margin: 0 auto; 76 | background: #eee; 77 | width: 300px; 78 | height: 300px; 79 | } 80 | 81 | .builder canvas.pathslider-canvas { 82 | border: #555 1px solid; 83 | } 84 | 85 | /* index.html */ 86 | .demo { 87 | color: #777; 88 | margin: 0; 89 | padding: 20px; 90 | font-size: 2.5em; 91 | font-family: 'Paytone One', sans-serif; 92 | white-space: nowrap; 93 | text-align: center; 94 | text-shadow: #555 1px 1px, #555 2px 2px; 95 | clear: both; 96 | } 97 | 98 | .demo .slider { 99 | margin: 0 auto; 100 | } 101 | 102 | /* builder.html */ 103 | #tabs, footer, div.footer { 104 | background: #677; 105 | margin-top: 10px; 106 | min-width: 565px; /*** make this dynamic in the next version ***/ 107 | } 108 | 109 | #tabs .tab { 110 | color: #eee; 111 | } 112 | 113 | p { 114 | line-height: 2em; 115 | } 116 | 117 | blockquote { 118 | margin: 5px 40px; 119 | } 120 | 121 | h4, .message { 122 | margin: 5px; 123 | } 124 | 125 | h4, strong { 126 | color: #fff; 127 | } 128 | 129 | .message { 130 | min-height: 50px; 131 | } 132 | 133 | img#save { 134 | border: #ff0 1px solid; 135 | width: 15px; 136 | height: 15px; 137 | } 138 | 139 | .moving { 140 | cursor: move; 141 | } 142 | 143 | .corner { 144 | background: url() no-repeat; 145 | width: 10px; 146 | height: 10px; 147 | position: absolute; 148 | bottom: 0; 149 | right: 0; 150 | cursor: nw-resize; 151 | } 152 | 153 | .events { 154 | width: 30%; 155 | position: relative; 156 | margin-bottom: 10px; 157 | float: right; 158 | } 159 | .move-controls { 160 | display: inline-block; 161 | height: 70px; 162 | margin-left: 40px; 163 | } 164 | .move-controls input { 165 | width: 40px; 166 | } 167 | 168 | textarea.code { 169 | background: #ddd; 170 | font-family: 'courier'; 171 | width: 100%; 172 | } 173 | 174 | /* builder inputs */ 175 | .points { width: 20em; } 176 | .position, .angle, .tolerance.text, .range.text, .thickness.text, .grid.text { width: 35px; } 177 | .gripColor, .curveColor { width: 100px; } 178 | .bkgd { width: 90%; } 179 | 180 | /* readme colors */ 181 | .sxy { color: hsl(120,60%,85%); } 182 | .scxy { color: hsl(120,80%,75%); } 183 | .ecxy { color: hsl(0,40%,85%); } 184 | .exy { color: hsl(0,55%,75%); } 185 | 186 | footer, div.footer, .push { 187 | height: 100px; 188 | font-size: 20px; 189 | text-align: center; 190 | clear: both; 191 | padding: 10px 0 30px 0; 192 | } 193 | 194 | footer, div.footer { 195 | background: #789; 196 | background: -moz-radial-gradient(center bottom, ellipse farthest-corner, #789, #677 40%); 197 | background: -webkit-radial-gradient(center bottom, ellipse farthest-corner, #789, #677 40%); 198 | background: -o-radial-gradient(center bottom, ellipse farthest-corner, #789, #677 40%); 199 | background: -ms-radial-gradient(center bottom, ellipse farthest-corner, #789, #677 40%); 200 | box-shadow: inset 0 0 10px #567; 201 | } 202 | 203 | .blog { 204 | font-size: 0.8em; 205 | color: #ccc; 206 | text-shadow: #aaa inset 0px 3px 3px; 207 | } 208 | 209 | #tooltip { 210 | max-width: 400px; 211 | font-size: 1.1em; 212 | color: #222; 213 | background: #ddd; 214 | border: 1px solid #aaa; 215 | padding: 10px; 216 | display: none; 217 | opacity: 0.95; 218 | filter: alpha(opacity=95); 219 | text-align: left; 220 | border-radius: 8px; 221 | -moz-border-radius: 8px; 222 | -webkit-border-radius: 8px; 223 | } 224 | 225 | #tooltip .close { 226 | display: none; 227 | } -------------------------------------------------------------------------------- /css/pathslider.css: -------------------------------------------------------------------------------- 1 | /* set slider dimension here */ 2 | #slider { 3 | width: 200px; 4 | height: 200px; 5 | } 6 | 7 | /* Basic solid color slider grip */ 8 | .pathslider-grip { 9 | width: 30px; 10 | height: 15px; 11 | position: absolute; 12 | top: 0; 13 | left: 0; 14 | background: #ddd; 15 | font-size: 1px; 16 | z-index: 10; 17 | cursor: move; 18 | border: 1px solid; 19 | border-radius: 7px; 20 | -webkit-box-shadow: 0px 1px 3px rgba(000,000,000,0.5), inset 0px 0px 1px rgba(255,255,255,0.6); 21 | box-shadow: 0px 1px 3px rgba(000,000,000,0.5), inset 0px 0px 1px rgba(255,255,255,0.6); 22 | } 23 | 24 | /* grip hover/sliding state */ 25 | .pathslider-grip.sliding, .pathslider-grip:hover { 26 | border-color: #8ce; 27 | -webkit-box-shadow: 0 0 15px #8ce; 28 | box-shadow: 0 0 15px #8ce; 29 | } 30 | 31 | /* slider extra css - same as #slider above */ 32 | .pathslider { 33 | position: relative; 34 | } 35 | 36 | /************* 37 | grip styles 38 | add these using the gripClass option 39 | *************/ 40 | /* shiny silver - sorta */ 41 | .silver { 42 | border-color: #949494; 43 | background-color: #ebebeb; 44 | background-image: -webkit-linear-gradient(top,#ffffff 0%,#ebebeb 50%,#dbdbdb 50%,#b5b5b5); 45 | background-image: linear-gradient(top,#ffffff 0%,#ebebeb 50%,#dbdbdb 50%,#b5b5b5); 46 | } 47 | 48 | /* shiny black */ 49 | .black { 50 | border-color: #000000 51 | background-color: #3b3b3b; 52 | background-image: -webkit-linear-gradient(top,#a3a3a3 0%,#3b3b3b 50%,#242424 50%,#000000); 53 | background-image: linear-gradient(top,#a3a3a3 0%,#3b3b3b 50%,#242424 50%,#000000); 54 | } 55 | 56 | /* Small circular chrome knob 57 | including ".pathslider-grip" to remove border & box shadow 58 | */ 59 | .pathslider-grip.chrome1 { 60 | width: 18px; 61 | height: 18px; 62 | border: 0; 63 | border-radius: 0; 64 | background: url(''); 65 | background-position: center center; 66 | background-repeat: no-repeat; 67 | -webkit-box-shadow: 0 0 0; 68 | box-shadow: 0 0 0; 69 | } 70 | 71 | /* Large curvy chrome slide */ 72 | .pathslider-grip.chrome2 { 73 | width: 57px; 74 | height: 35px; 75 | border: 0; 76 | border-radius: 0; 77 | background: url(''); 78 | background-position: center center; 79 | background-repeat: no-repeat; 80 | -webkit-box-shadow: 0 0 0; 81 | box-shadow: 0 0 0; 82 | } -------------------------------------------------------------------------------- /dist/pathslider.css: -------------------------------------------------------------------------------- 1 | /* set slider dimension here */ 2 | #slider { 3 | width: 200px; 4 | height: 200px; 5 | } 6 | 7 | /* Basic solid color slider grip */ 8 | .pathslider-grip { 9 | width: 30px; 10 | height: 15px; 11 | position: absolute; 12 | top: 0; 13 | left: 0; 14 | background: #ddd; 15 | font-size: 1px; 16 | z-index: 10; 17 | cursor: move; 18 | border: 1px solid; 19 | border-radius: 7px; 20 | -webkit-box-shadow: 0px 1px 3px rgba(000,000,000,0.5), inset 0px 0px 1px rgba(255,255,255,0.6); 21 | box-shadow: 0px 1px 3px rgba(000,000,000,0.5), inset 0px 0px 1px rgba(255,255,255,0.6); 22 | } 23 | 24 | /* grip hover/sliding state */ 25 | .pathslider-grip.sliding, .pathslider-grip:hover { 26 | border-color: #8ce; 27 | -webkit-box-shadow: 0 0 15px #8ce; 28 | box-shadow: 0 0 15px #8ce; 29 | } 30 | 31 | /* slider extra css - same as #slider above */ 32 | .pathslider { 33 | position: relative; 34 | } 35 | 36 | /************* 37 | grip styles 38 | add these using the gripClass option 39 | *************/ 40 | /* shiny silver - sorta */ 41 | .silver { 42 | border-color: #949494; 43 | background-color: #ebebeb; 44 | background-image: -webkit-linear-gradient(top,#ffffff 0%,#ebebeb 50%,#dbdbdb 50%,#b5b5b5); 45 | background-image: linear-gradient(top,#ffffff 0%,#ebebeb 50%,#dbdbdb 50%,#b5b5b5); 46 | } 47 | 48 | /* shiny black */ 49 | .black { 50 | border-color: #000000 51 | background-color: #3b3b3b; 52 | background-image: -webkit-linear-gradient(top,#a3a3a3 0%,#3b3b3b 50%,#242424 50%,#000000); 53 | background-image: linear-gradient(top,#a3a3a3 0%,#3b3b3b 50%,#242424 50%,#000000); 54 | } 55 | 56 | /* Small circular chrome knob 57 | including ".pathslider-grip" to remove border & box shadow 58 | */ 59 | .pathslider-grip.chrome1 { 60 | width: 18px; 61 | height: 18px; 62 | border: 0; 63 | border-radius: 0; 64 | background: url(''); 65 | background-position: center center; 66 | background-repeat: no-repeat; 67 | -webkit-box-shadow: 0 0 0; 68 | box-shadow: 0 0 0; 69 | } 70 | 71 | /* Large curvy chrome slide */ 72 | .pathslider-grip.chrome2 { 73 | width: 57px; 74 | height: 35px; 75 | border: 0; 76 | border-radius: 0; 77 | background: url(''); 78 | background-position: center center; 79 | background-repeat: no-repeat; 80 | -webkit-box-shadow: 0 0 0; 81 | box-shadow: 0 0 0; 82 | } -------------------------------------------------------------------------------- /demo/demo.js: -------------------------------------------------------------------------------- 1 | $(function(){ 2 | 3 | var i, s, t, pathslider = $('#slider'), 4 | points = $('.points'), 5 | position = $('.position'), 6 | angle = $('.angle'), 7 | code = $('textarea.code'), 8 | m = $('.message'), 9 | presets = { 10 | '0' : [ 25,150,50,-50,-50,-50,275,150 ], // arch 11 | '1' : [ 25,275,25,-25,-25,25,275,25 ], // diagonal 12 | '2' : [ 25,275,250,0,-250,0,275,25 ], // s-shape 13 | '3' : [ 100,200,200,25,-100,-275,100,275 ], // p-shape 14 | '4' : [ 75,200,200,-125,-200,-125,225,200 ] // loop 15 | }, 16 | // update code box 17 | updateOptions = function(s){ 18 | var t; 19 | if (s) { 20 | if (s.hasOwnProperty('builderInitialized') && s.builderInitialized) { code.val( s.getCode().replace(/url\(\)/, $('.bkgd').val()) ); } 21 | $('.rotate')[0].checked = s.options.rotateGrip; // I would use prop, but want to make it work with jQuery 1.4.4 22 | $('.tolerance').val(s.options.tolerance).filter('.isRange').next().html( s.options.tolerance ); 23 | $('.range').val( s.options.range ).filter('.isRange').next().html( s.options.range + ' px' ); 24 | $('.edit')[0].checked = s.options.edit; 25 | $('.grid').val( s.options.grid ).filter('.isRange').next().html( s.options.grid === 1 ? 'off' : s.options.grid + ' px' ); 26 | $('.snap')[0].checked = s.options.snap; 27 | $('.thickness').val( s.options.curve.width ).filter('.isRange').next().html( s.options.curve.width + ' px' ); 28 | t = $('.gripColor'); t.val( checkColor(t, t.val()) ); 29 | $('.curveColor').each(function(indx, el){ 30 | var $el = $(el); 31 | $el.val( checkColor($el) ); 32 | }); 33 | } 34 | }, 35 | checkColor = function(t,c){ 36 | var c = c || t.val(), 37 | d = (/^(#[\da-fA-F]{6}$)/.test(c)) ? c : t.attr('data-last'); 38 | t.attr('data-last', d); 39 | return d; 40 | }, 41 | // t = jQuery object, d = default value 42 | checkRange = function(t,d){ 43 | // parseInt($('.tolerance').val(), 10) || 3; 44 | var v = parseInt(t.val(), 10) || d, 45 | min = parseInt(t.attr('min'), 10), 46 | max = parseInt(t.attr('max'), 10); 47 | return (v < min) ? min : (v > max) ? max : v; 48 | }; 49 | 50 | $('#tabs').tabs({ 51 | show: function(e,ui){ 52 | if (ui.tab.innerHTML === "Code") { 53 | var s = pathslider.data('pathslider'); 54 | updateOptions(s); 55 | if (s && s.hasOwnProperty('updateCanvasImage')) { 56 | // update save canvas image 57 | s.updateCanvasImage(); 58 | } 59 | } 60 | } 61 | }); 62 | 63 | // add preset buttons 64 | i = 0; t = ''; 65 | while (presets.hasOwnProperty(i)) { 66 | t += ''; 67 | i++; 68 | } 69 | $('.presets').html(t); 70 | 71 | // check input type support 72 | $('.tolerance, .range, .grid, .thickness').each(function(){ 73 | if (this.type !== 'range') { 74 | $(this) 75 | .addClass('text') 76 | .next() 77 | .html(' range (' + $(this).attr('min') + '-' + $(this).attr('max') + ')'); 78 | } else { 79 | $(this).addClass('isRange'); 80 | } 81 | }); 82 | 83 | // update all pathslider options 84 | $('.set').change(function(){ 85 | var t; 86 | s.options.rotateGrip = $('.rotate')[0].checked; 87 | s.options.tolerance = checkRange( $('.tolerance'), 3); 88 | s.options.range = checkRange( $('.range'), 30); 89 | s.options.edit = $('.edit')[0].checked; 90 | s.options.grid = checkRange( $('.grid'), 25); 91 | s.options.snap = $('.snap')[0].checked; 92 | $('.pathslider-grip').attr('style',''); // clear rotation 93 | s.options.points = points.val().split(','); 94 | s.options.curve.width = checkRange( $('.thickness'), 4); 95 | s.options.curve.cap = $('.capStyle').val().toLowerCase(); 96 | 97 | t = $('.curveColor').map(function(){ 98 | return checkColor($(this)); 99 | }).get(); 100 | if (t[0] === t[1]) { 101 | t = t[0]; 102 | } 103 | // a color array => linear gradient in v1.0.0 104 | s.options.curve.color = t; 105 | 106 | if ($('.gripStyle')[0].selectedIndex === 0) { 107 | $('.gripColor')[0].disabled = false; 108 | t = checkColor( $('.gripColor') ); 109 | s.options.gripClass = ''; // update for code tab 110 | $('.pathslider-grip') 111 | .attr('class', 'pathslider-grip') 112 | .css('background-color', t); 113 | } else { 114 | t = $('.gripStyle').val().toLowerCase().replace(/\s+/g,''); 115 | s.options.gripClass = t; 116 | $('.gripColor')[0].disabled = true; 117 | $('.pathslider-grip') 118 | .attr('class', 'pathslider-grip ' + t) 119 | .css('background-color', ''); 120 | } 121 | 122 | // background - remove ";" because it breaks if you don't 123 | t = $('.bkgd').val().replace(/\;/g,''); 124 | $('.bkgd').val(t); 125 | $('#slider').css('background', t ); 126 | 127 | updateOptions(s); 128 | s.updateBuilder(); 129 | }); 130 | 131 | // set up pathslider 132 | $('#slider') 133 | .pathslider({ 134 | rotateGrip : true, 135 | // points : [ 50,300, 0,250, 180,250, 300,50 ], 136 | points : [ 0,50, 50,-50, -50,-50, 250,50 ], 137 | 138 | create : function(e,slider){ 139 | points.val(slider.points); 140 | position.val(slider.percent); 141 | angle.text( slider.angle );50 142 | }, 143 | update : function(e,slider){ 144 | points.val(slider.points); 145 | updateOptions(slider); 146 | }, 147 | slide : function(e,slider){ 148 | position.val(slider.percent); 149 | angle.text( slider.angle ); 150 | } 151 | }) 152 | .pathbuilder({ 153 | edit : true, 154 | snap : true, 155 | grid : 25, 156 | canvasImage : $('#save') 157 | }); 158 | 159 | // Select all text in the points input 160 | points.mouseup(function(){ 161 | this.select(); 162 | }); 163 | 164 | // shift points 165 | $('.shift').click(function(){ 166 | var x, y, 167 | $el = $(this), 168 | isX = $el.filter('.x').length, 169 | dir = $el.filter('.left, .up').length ? -1 : 1; 170 | amount = parseInt($(isX ? '.movex' : '.movey').val(), 10) || 10; 171 | 172 | x = isX ? dir * amount : 0; 173 | y = isX ? 0 : dir * amount; 174 | s.points = s.options.points = s.shift(x, y); 175 | s.updateBuilder(); 176 | }); 177 | 178 | // "Set" points 179 | $('.setpoints').click(function(){ 180 | var p, pts = points.val().split(','); 181 | if (pts.length === 8) { 182 | s.options.points = pts; 183 | s.updateBuilder(); 184 | } 185 | }); 186 | 187 | // "Set" position 188 | $('.setpos').click(function(){ 189 | s.setSlider( parseFloat(position.val()), function(s){ 190 | // no need to validate, just replace the value 191 | position.val( s.percent ); 192 | } ); 193 | }); 194 | 195 | // presets 196 | $('.presets button').click(function(){ 197 | t = parseInt($(this).text(), 10) - 1; 198 | points.val(s.options.points = presets[t]); 199 | s.updateBuilder(); 200 | }); 201 | 202 | // Add events 203 | pathslider.bind('create.pathslider update.pathslider start.pathslider slide.pathslider change.pathslider stop.pathslider',function(e,s){ 204 | m.append('
  • ' + e.type + ' : ' + s.percent + '
  • '); 205 | var l = m.find('li'); 206 | if (l.length > 10) { l.eq(0).remove(); } 207 | }); 208 | 209 | // Color styles 210 | $('.gripColor, .curveColor').ColorPicker({ 211 | onSubmit: function(hsb, hex, rgb, el) { 212 | $(el).val('#' + hex); 213 | $(el).ColorPickerHide(); 214 | $(el).trigger('change'); 215 | }, 216 | onBeforeShow: function () { 217 | $(this).ColorPickerSetColor(this.value.slice(1)); 218 | } 219 | }) 220 | .bind('keyup', function(){ 221 | $(this).ColorPickerSetColor(this.value.slice(1)); 222 | }); 223 | 224 | // Make info links (tooltips) unclickable 225 | $('a.info').click(function(){ 226 | if (this.target === "") { return false; } 227 | }); 228 | 229 | // set up tooltips 230 | $.jatt({ direction : 'ne' }); 231 | 232 | s = pathslider.data('pathslider'); 233 | 234 | }); -------------------------------------------------------------------------------- /js/jquery.pathslider.builder.js: -------------------------------------------------------------------------------- 1 | /* jQuery Pathslider Builder v0.9.1 alpha 2 | * By Rob Garrison (Mottie) 3 | * MIT License 4 | * 5 | * Code based on the bezier curve demo by Craig Buckler (http://twitter.com/craigbuckler) 6 | * http://blogs.sitepointstatic.com/examples/tech/canvas-curves/bezier-curve.html 7 | */ 8 | (function($){ 9 | 10 | $.pathbuilderdefaults = { 11 | edit : true, // allows user to move bezier curve 12 | snap : true, 13 | grid : 10, 14 | // default canvas styles 15 | // control point line (cpline) & point are used in edit mode 16 | style : { 17 | cpline : { width: 1, color: "#00c" }, 18 | // start point 19 | start : { width: 2, color: "hsl(120,100%,30%)", fill: "hsla(120,100%,30%,0.3)", radius: 10, arc1: 0, arc2: 2 * Math.PI }, 20 | // start control point 21 | cstart : { width: 2, color: "hsl(120,100%,30%)", fill: "hsla(120,100%,30%,0.5)", radius: 10, arc1: 0, arc2: 2 * Math.PI }, 22 | // end control point 23 | cend : { width: 2, color: "hsl(0,100%,30%)", fill: "hsla(0,100%,30%,0.5)", radius: 10, arc1: 0, arc2: 2 * Math.PI }, 24 | // end point 25 | end : { width: 2, color: "hsl(0,100%,30%)", fill: "hsla(0,100%,30%,0.3)", radius: 10, arc1: 0, arc2: 2 * Math.PI }, 26 | grid : { width: 1, color: "hsla(0,0%,60%,0.5)" } 27 | }, 28 | canvasImage : null // jQuery object of img that the canvas is saved into - see base.updateCanvasImage 29 | }; 30 | 31 | $.fn.pathbuilder = function(options) { 32 | return this.each(function(){ 33 | // make sure pathslider is attached 34 | var o, base = $(this).data('pathslider'); 35 | if (!base) { return; } 36 | 37 | base.builderInit = function(){ 38 | base.options = o = $.extend(true, base.options, $.pathbuilderdefaults, options); 39 | base.drag = null; 40 | 41 | base.options.drawCanvas = function(b, ctx, points) { 42 | base.drawControls(ctx, points); 43 | // prevent redrawing the curve 44 | return false; 45 | }; 46 | 47 | if (!base.hasCanvas) { return; } // too bad IE! 48 | 49 | // Add resizable corner to allow resizing of the canvas 50 | $('
    ') 51 | .appendTo(base.$el) 52 | .mousedown(function(e) { 53 | $(document) 54 | .unbind('mousemove.pathbuilder touchmove.pathbuilder') 55 | .bind((base.hasTouch ? 'touchmove' : 'mousemove') + '.pathbuilder', function(e) { 56 | var ww = $(window).width(), 57 | wh = $(window).height(), 58 | l = base.$canvas.offset().left, 59 | t = base.$canvas.offset().top, 60 | w = (e.originalEvent.touches ? e.originalEvent.touches[0].pageX : e.pageX) - l, 61 | h = (e.originalEvent.touches ? e.originalEvent.touches[0].pageY : e.pageY) - t; 62 | w = (w < 100) ? 100 : w > ww-l ? ww-l : w; 63 | h = (h < 100) ? 100 : h > wh-t ? wh-t : h; // min w & h 64 | base.$canvas.attr({ width : w, height: h }); 65 | base.$el.css({ width : w, height: h }); 66 | base.sliderDim = [ base.$el.position().left, base.$el.position().top, w, h ]; 67 | // Add bezier curve & controls 68 | base.drawCurve(); 69 | }) 70 | .bind((base.hasTouch ? 'touchend' : 'mouseup') + '.pathbuilder', function(){ 71 | $(document).unbind('mousemove.pathbuilder touchmove.pathbuilder'); 72 | }); 73 | }); 74 | 75 | base.updateBuilder(true); 76 | base.builderInitialized = true; // initialization flag 77 | base.$el.trigger('update.pathslider', [base] ); 78 | 79 | }; 80 | 81 | base.updateBuilder = function(internal) { 82 | // make sure all of the points are numerical 83 | base.points = o.points = $.map(o.points, function(v,i){ 84 | return parseInt(v*100,10)/100; 85 | }); 86 | // convert control point offsets to x/y positions 87 | base.redraw( (base.builderInitialized) ? base.points : base.shift(25,100) ); 88 | if (o.edit) { 89 | base.$canvas 90 | .unbind('.pathbuilder') 91 | .bind((base.hasTouch ? 'touchstart' : 'mousedown') + '.pathbuilder', function(e){ base.dragStart(e); }) 92 | .bind((base.hasTouch ? 'touchmove' : 'mousemove') + '.pathbuilder', function(e){ base.dragging(e); }) 93 | .bind((base.hasTouch ? 'touchend.pathbuilder touchcancel' : 'mouseup.pathbuilder mouseleave') + '.pathbuilder', function(e){ base.dragEnd(e); }); 94 | } 95 | if (!internal) { 96 | base.$el.trigger('update.pathslider', [base]); 97 | } 98 | }; 99 | 100 | // Make purdy stuff 101 | base.drawControls = function(c, b) { 102 | var i, j, x, y, t, p, s = o.style, 103 | // c = base.ctx, b = base.pointsxy, 104 | w = base.sliderDim[2], h = base.sliderDim[3]; 105 | base.points = base.convert(); 106 | clearTimeout(base.timer); 107 | c.clearRect(0, 0, w, h); 108 | 109 | // finish drawing curve - don't let drawCanvas do this because 110 | // we're changing the colors for the control handles 111 | base.finishCurve(c, b); 112 | c.beginPath(); 113 | c.moveTo(b[0], b[1]); 114 | c.bezierCurveTo(b[2], b[3], b[4], b[5], b[6], b[7]); 115 | c.stroke(); 116 | 117 | // Edit mode = add grid, control points/lines 118 | if (o.edit) { 119 | // make grid 120 | if (o.grid >= 2) { 121 | c.prop({ 122 | lineWidth : s.grid.width, 123 | strokeStyle : s.grid.color 124 | }) 125 | .beginPath(); 126 | for (x = o.grid; x < w; x += o.grid) { 127 | c.moveTo(x, 0).lineTo(x, h); 128 | } 129 | for (y = o.grid; y < h; y += o.grid) { 130 | c.moveTo(0, y).lineTo(w, y); 131 | } 132 | c.stroke(); 133 | } 134 | 135 | j = 0; 136 | base.controls = b; 137 | base.controlNames = ['start', 'cstart', 'cend', 'end']; 138 | 139 | // Add control lines 140 | c.prop({ 141 | lineWidth : s.cpline.width, 142 | strokeStyle : s.cpline.color 143 | }) 144 | .beginPath() 145 | .moveTo(b[0], b[1]) 146 | .lineTo(b[2], b[3]) 147 | .moveTo(b[6], b[7]) 148 | .lineTo(b[4], b[5]) 149 | .stroke(); 150 | // Add control points 151 | for (i=0; i < 8; i++) { 152 | c.prop({ 153 | lineWidth : s[base.controlNames[j]].width, 154 | strokeStyle : s[base.controlNames[j]].color, 155 | fillStyle : s[base.controlNames[j]].fill 156 | }) 157 | .beginPath() 158 | .arc( 159 | b[i++], 160 | b[i], 161 | s[base.controlNames[j]].radius, 162 | s[base.controlNames[j]].arc1, 163 | s[base.controlNames[j]].arc2, 164 | true 165 | ) 166 | .fill() 167 | .stroke(); 168 | j += i%2; 169 | } 170 | } 171 | 172 | // thottle resizing window 173 | base.timer = setTimeout(function(){ 174 | base.makeArray(); 175 | base.setSlider(base.percent, null, true); 176 | }, 100); 177 | 178 | }; 179 | 180 | base.updateCanvasImage = function(){ 181 | if (o.canvasImage.length) { 182 | clearTimeout(base.timer); 183 | base.ctx.clearRect(0, 0, base.sliderDim[2], base.sliderDim[3]); 184 | base.drawCurve(); 185 | o.canvasImage.attr('src', base.canvas.toDataURL()); 186 | base.updateBuilder(); 187 | } 188 | }; 189 | 190 | // start dragging points on canvas 191 | base.dragStart = function(event){ 192 | if (!o.edit) { return; } 193 | var e = base.mousePos(event), 194 | l = base.controls.length, 195 | i, j = 0, r, dx, dy; 196 | for (i=0; i < l; i++) { 197 | dx = base.controls[i++] - e[0]; 198 | dy = base.controls[i] - e[1]; 199 | r = o.style[base.controlNames[j]].radius + o.grid/2; 200 | if ((dx * dx) + (dy * dy) < r * r) { 201 | base.drag = i-1; 202 | base.dPoint = e; 203 | base.$canvas.addClass('moving'); 204 | return; 205 | } 206 | j += i%2; 207 | } 208 | }; 209 | 210 | // dragging points on canvas 211 | base.dragging = function(event){ 212 | if (o.edit && base.drag !== null) { 213 | var i, g = o.grid || 1, 214 | e = base.mousePos(event), 215 | c = base.controls, 216 | l = c.length, 217 | x = e[0] - base.dPoint[0], 218 | y = e[1] - base.dPoint[1]; 219 | // Move whole thing 220 | if (event.shiftKey) { 221 | if (x > o.grid || y > o.grid) { base.dPoint = e; } 222 | for (i=0; i < l; i++) { 223 | // *** SHIFT-snap to grid needs improvement *** 224 | c[i] = (o.snap) ? Math.round((c[i] + x)/g)*g : c[i] + x; 225 | c[++i] = (o.snap) ? Math.round((c[i] + y)/g)*g : c[i] + y; 226 | } 227 | base.dPoint = e; 228 | } else { 229 | // only move the one point 230 | c[base.drag] = (o.snap) ? Math.round((base.dPoint[0] + x)/g)*g : base.dPoint[0] + x; 231 | c[base.drag+1] = (o.snap) ? Math.round((base.dPoint[1] + y)/g)*g : base.dPoint[1] + y; 232 | } 233 | // Add bezier curve & controls 234 | base.drawCurve(); 235 | 236 | base.$el.trigger('update.pathslider', [base]); 237 | } 238 | }; 239 | 240 | // end dragging 241 | base.dragEnd = function(e){ 242 | if (o.edit && base.drag !== null) { 243 | base.drag = null; 244 | base.$canvas.removeClass('moving'); 245 | o.points = base.points; 246 | base.updateBuilder(true); 247 | } 248 | }; 249 | 250 | base.crop = function(){ 251 | o.points = base.points; 252 | base.updateBuilder(true); 253 | // include curve width when cropping 254 | var w = o.curve.width * 2; 255 | return [ 256 | Math.min.apply(this,base.arrayX) - w, // min X 257 | Math.min.apply(this,base.arrayY) - w, // min Y 258 | Math.max.apply(this,base.arrayX) + w, // max X 259 | Math.max.apply(this,base.arrayY) + w // max Y 260 | ]; 261 | }; 262 | 263 | // base.pointsxy = [ sx,sy, csx,csy, cex,cey, ex,ey ] to 264 | // base.points = [ sx,sy, csxo,csyo, cexo,ceyo, ex,ey ] 265 | base.convert = function(p){ 266 | p = p || base.pointsxy; 267 | return [ 268 | p[0], p[1], // start x,y 269 | p[2] - p[0], p[3] - p[1], // start control x,y 270 | p[4] - p[6], p[5] - p[7], // end control x,y 271 | p[6], p[7] // end x,y 272 | ]; 273 | }; 274 | 275 | // shift all points so coords based from origin 276 | // example: base.shift(-10, +10); 277 | base.shift = function(x,y){ 278 | // b = [ sx,sy, csx,csy, cex,cey, ex,ey ] 279 | var b = base.points; 280 | return [ 281 | Math.round(b[0]+x), Math.round(b[1]+y), 282 | b[2], b[3], 283 | b[4], b[5], 284 | Math.round(b[6]+x), Math.round(b[7]+y) 285 | ]; 286 | }; 287 | 288 | base.getCode = function(){ 289 | var dim = base.crop(), 290 | points = base.shift(-dim[0],-dim[1]), 291 | txt = '\n' + 298 | ''; 313 | return txt; 314 | }; 315 | 316 | // Run initializer 317 | base.builderInit(); 318 | 319 | }); 320 | }; 321 | 322 | })(jQuery); 323 | 324 | /* jshint ignore:start */ 325 | /** 326 | * Chainvas: Make APIs chainable 327 | * @author Lea Verou 328 | * MIT license http://www.opensource.org/licenses/mit-license.php 329 | */ 330 | (function(){var e=window.Chainvas={chainable:function(a){return function(){var b=a.apply(this,arguments);return b===void 0?this:b}},chainablizeOne:function(a,b){try{e.utils.hasOwnProperty(a,b)&&e.utils.isFunction(a[b])&&(a[b]=e.chainable(a[b]))}catch(c){}return this},chainablize:function(a,b){var c=a.prototype;if(b)for(var d=b.length;d--;)e.chainablizeOne(c,b[d]);else for(d in c)e.chainablizeOne(c,d);return this},helpers:function(a,b){var c=a.prototype,d;for(d in e.methods)c&&!(d in c)&&(c[d]=e.methods[d]); 331 | e.extend(c,b);return this},extend:function(a,b){return Chainvas.methods.prop.call(a,b)},global:function(a,b,c){typeof a==="string"&&(a=[a]);for(var d=a.length;d--;){var f=window[a[d]];f&&e.chainablize(f,c).helpers(f,b)}},methods:{prop:function(){if(arguments.length===1){var a=arguments[0],b;for(b in a)this[b]=a[b]}else arguments.length===2&&(this[arguments[0]]=arguments[1]);return this}},utils:{isFunction:function(a){var b=Object.prototype.toString.call(a);return b==="[object Function]"||b==="[object Object]"&& 332 | "call"in a&&"apply"in a&&/^\s*\bfunction\s+\w+\([\w,]*\) \{/.test(a+"")},hasOwnProperty:function(a,b){try{return a.hasOwnProperty(b)}catch(c){return b in a&&(!a.prototype||!(b in a.prototype)||a.prototype[b]!==a[b])}}}}})(); 333 | 334 | /** 335 | * Chainvas module: Canvas 336 | */ 337 | Chainvas.global("CanvasRenderingContext2D",{circle:function(a,b,d){return this.beginPath().arc(a,b,d,0,2*Math.PI,!1).closePath()},roundRect:function(a,b,d,e,c){return this.beginPath().moveTo(a+c,b).lineTo(a+d-c,b).quadraticCurveTo(a+d,b,a+d,b+c).lineTo(a+d,b+e-c).quadraticCurveTo(a+d,b+e,a+d-c,b+e).lineTo(a+c,b+e).quadraticCurveTo(a,b+e,a,b+e-c).lineTo(a,b+c).quadraticCurveTo(a,b,a+c,b).closePath()}}); 338 | /* jshint ignore:end */ 339 | -------------------------------------------------------------------------------- /js/jquery.pathslider.js: -------------------------------------------------------------------------------- 1 | /* jQuery Pathslider v1.0.0 alpha 2 | * By Rob Garrison (Mottie) 3 | * MIT License 4 | */ 5 | (function($){ 6 | $.pathslider = function(el, options){ 7 | 8 | // To avoid scope issues, use 'base' instead of 'this' 9 | // to reference this class from internal events and functions. 10 | var base = this, o; 11 | 12 | // Access to jQuery and DOM versions of element 13 | base.$el = $(el).addClass('pathslider'); 14 | base.el = el; 15 | 16 | // Add a reverse reference to the DOM object 17 | base.$el.data("pathslider", base); 18 | 19 | base.init = function(){ 20 | var t; 21 | base.options = o = $.extend(true, {}, $.pathslider.defaults, options); 22 | 23 | // Is there a canvas? 24 | t = document.createElement('canvas'); 25 | base.hasCanvas = !!(t.getContext && t.getContext('2d')); 26 | base.hasTouch = document.hasOwnProperty("ontouchend"); 27 | 28 | // add grip 29 | base.$grip = $('
    ').appendTo(base.$el); 30 | 31 | // store array of x & y positions for cross reference 32 | base.points = []; 33 | base.pointsxy = []; 34 | base.arrayX = []; 35 | base.arrayY = []; 36 | base.arrayP = []; 37 | 38 | base.rad2deg = 180 / Math.PI; // convert radians to degrees (multiply radian by this value) 39 | base.sliding = false; // flag for dragging element 40 | base.lastPercent = base.percent = o.value; 41 | 42 | // Callbacks 43 | // slide triggered on EVERY mouse move; change triggered on slide stop 44 | $.each('create update start slide change stop'.split(' '), function(i,f){ 45 | if ($.isFunction(o[f])){ 46 | base.$el.bind(f + '.pathslider', o[f]); 47 | } 48 | }); 49 | 50 | $(document) 51 | .bind( base.hasTouch ? 'touchend.pathslider touchcancel.pathslider' : 'mouseup.pathslider mouseleave.pathslider', function(e){ 52 | if (base.sliding) { // && ($(e.target).closest('.pathslider').length || e.type === 'mouseleave')) { 53 | base.$el.trigger('stop.pathslider', [base]); 54 | if (base.lastPercent !== base.percent) { 55 | base.lastPercent = base.percent; 56 | base.$el.trigger('change.pathslider', [base]); 57 | } 58 | } 59 | base.$grip.removeClass('sliding'); 60 | base.sliding = false; 61 | }) 62 | .bind( (base.hasTouch ? 'touchmove' : 'mousemove') + '.pathslider', function(e){ 63 | if (base.sliding) { 64 | base.setSlider( base.findPos(e), null, true ); 65 | } 66 | }); 67 | 68 | $(window) 69 | .bind('resize.pathslider', function(){ 70 | base.update(); 71 | }) 72 | .bind('load', function(){ 73 | // needed because loading images/fonts will shift the page 74 | base.sliderDim[0] = base.$el.offset().left; 75 | base.sliderDim[1] = base.$el.offset().top; 76 | }); 77 | 78 | base.$grip 79 | .bind( (base.hasTouch ? 'touchstart' : 'mousedown') + '.pathslider', function(e){ 80 | base.sliding = true; 81 | $(this).addClass('sliding'); 82 | base.$el.trigger('start.pathslider', [base]); 83 | return false; 84 | }) 85 | .bind('click', function(){ 86 | return false; 87 | }); 88 | 89 | base.redraw(); 90 | 91 | base.$el.trigger('create.pathslider', [base]); 92 | 93 | }; 94 | 95 | // update dimensions & grip position 96 | base.update = function(){ 97 | 98 | // using attr to remove other css grip classes when updating 99 | base.$grip.attr('class', 'pathslider-grip ' + o.gripClass); 100 | 101 | if (base.ctx) { 102 | // clear canvas *before* setting new dimensions 103 | // just in case the new size is smaller than the previous 104 | base.ctx.clearRect(0, 0, base.sliderDim[2], base.sliderDim[3]); 105 | } 106 | 107 | base.sliderDim = [ 108 | base.$el.offset().left, 109 | base.$el.offset().top, 110 | base.$el.width(), 111 | base.$el.height() 112 | ]; 113 | 114 | // get grip dimensions; jQuery v3+ width() & height() return the rotated dimensions 115 | // which we don't want! 116 | var computedStyle = window.getComputedStyle(base.$grip[0]); 117 | // for centering grip 118 | base.gripCenter = [ parseInt(computedStyle.width, 10)/2, parseInt(computedStyle.height, 10)/2 ]; 119 | 120 | // number of data points to store - increase to smooth the animation (based on slider size) 121 | base.dataPoints = o.dataPoints; 122 | // in next update add min/max/step 123 | // base.range = o.max - o.min; 124 | // base.dataPoints = base.range * o.step; 125 | 126 | base.makeArray(); 127 | // save the position in the array of the starting value (roughly) 128 | var t = $.inArray(base.percent, base.arrayP); 129 | base.position = (t === -1) ? Math.round(base.percent/100 * base.dataPoints) : t; 130 | 131 | base.setSlider(base.percent, null, true); 132 | if (base.hasCanvas && o.useCanvas) { base.drawCurve(); } 133 | 134 | }; 135 | 136 | // set position of slider 137 | base.setSlider = function(percent, callback, internal) { 138 | if (!isNaN(percent)) { 139 | // find position on bezier curve; p = percent (range 0 - 100) 140 | // set position of slider without using the array (more precision) 141 | percent = parseFloat(percent, 10); 142 | percent = (percent > 100) ? 100 : percent < 0 ? 0 : percent; 143 | var css, angle, 144 | // pos = $.inArray(percent, base.arrayP), 145 | p = base.calcBezier(percent/100, base.pointsxy), 146 | pm1 = (percent - 2 > 0) ? base.calcBezier( (percent-2)/100, base.pointsxy ) : p, 147 | pp1 = (percent + 2 < 100) ? base.calcBezier( (percent+2)/100, base.pointsxy ) : p, 148 | // m = slope of tangent - used to change rotation angle of the grip 149 | // yes, I could have used the cubic derivative, but this is less math 150 | m = (pp1[0] - pm1[0] === 0) ? 90 : (pp1[1] - pm1[1])/(pp1[0] - pm1[0]); 151 | base.angle = parseInt(Math.atan(m) * base.rad2deg, 10); 152 | angle = 'rotate(' + base.angle + 'deg)'; 153 | css = (o.rotateGrip) ? { 154 | '-webkit-transform' : angle, 155 | 'transform' : angle 156 | } : {}; 157 | css.left = p[0] - base.gripCenter[0]; 158 | css.top = p[1] - base.gripCenter[1]; 159 | base.$grip 160 | .attr({ 161 | 'data-angle' : base.angle, 162 | 'data-percent' : percent 163 | }) 164 | .css(css); 165 | // find closest percent in the array - this relies on there being a factor 166 | // of 100 datapoints, so it'll need changing when we have a min/max/step 167 | base.percent = percent; // Math.round(percent*r)/r; 168 | if ((percent !== base.lastPercent && !base.sliding) || !internal) { 169 | base.$el.trigger('change.pathslider', [base]); 170 | } 171 | } 172 | if (typeof callback === 'function') { callback(base); } 173 | }; 174 | 175 | // relative mouse position 176 | base.mousePos = function(e) { 177 | return [ 178 | (e.originalEvent.touches ? e.originalEvent.touches[0].pageX : e.pageX) - base.sliderDim[0], 179 | (e.originalEvent.touches ? e.originalEvent.touches[0].pageY : e.pageY) - base.sliderDim[1] 180 | ]; 181 | }; 182 | 183 | // find percentage given the x,y coordinates 184 | // searching through a set array of points starting from the last known position 185 | // This allows the curve to loop over itself without mixing up intersecting points 186 | // The biggest issue is a very sharp turn 187 | base.findPos = function(event) { 188 | var i, j, dx, dy, px = [], py = [], 189 | last = base.position, //* base.dataPoints / 100, 190 | // check x & y cross ref based on nearby positions (+/- tolerance) 191 | t = parseInt(o.tolerance + 1, 10) || 2, // tolerance of 1 is too small 192 | r = parseInt(o.range, 10) || base.gripCenter[0], // set to 1/2 width of grip 193 | pos = base.mousePos(event); 194 | // save percent 195 | for ( i=0; i < r; i++ ){ 196 | px = []; py = []; 197 | for ( j=0; j < t + 1; j++ ){ 198 | // check positive direction 199 | dx = Math.abs(base.arrayX[last+j] - pos[0]) <= i; 200 | dy = Math.abs(base.arrayY[last+j] - pos[1]) <= i; 201 | if (dx && dy) { return base.returnPos(last+j); } 202 | if (dx) { px.push(last+j); } 203 | if (dy) { py.push(last+j); } 204 | // check in negative direction 205 | dx = Math.abs(base.arrayX[last-j] - pos[0]) <= i; 206 | dy = Math.abs(base.arrayY[last-j] - pos[1]) <= i; 207 | if (dx && dy) { return base.returnPos(last-j); } 208 | if (dx) { px.push(last-j); } 209 | if (dy) { py.push(last-j); } 210 | } 211 | if (px.length === 1 && py.length > 1) { return base.returnPos(px[0]); } 212 | if (py.length === 1 && px.length > 1) { return base.returnPos(py[0]); } 213 | } 214 | return base.returnPos(last); 215 | }; 216 | 217 | // return found position & trigger slide event 218 | base.returnPos = function(p) { 219 | var t = base.position === p; 220 | base.percent = base.arrayP[p]; 221 | base.position = p; 222 | if (!t) { 223 | if (base.hasCanvas && o.useCanvas) { 224 | base.drawCurve(); 225 | } 226 | base.$el.trigger('slide.pathslider', [base] ); 227 | } 228 | return base.percent; 229 | }; 230 | 231 | // build cross-ref array - find position based on x,y coords 232 | base.makeArray = function(){ 233 | var i, t, b = base.pointsxy, 234 | n = base.dataPoints; 235 | for ( i=0; i < n+1; i++ ){ 236 | t = base.calcBezier(i/n, b); 237 | base.arrayX[i] = t[0]; 238 | base.arrayY[i] = t[1]; 239 | base.arrayP[i] = t[2]; 240 | } 241 | }; 242 | 243 | // Calculate bezier x & y based on percentage (p) 244 | // cubic bezier = start(p^3) + cstart(3*p^2*(1−p)) + cend(3*p*(1−p)^2) + end(1−p)^3 245 | // b = [ startx,starty, cstartx,cstarty, cendx,cendy, endx,endy ] 246 | base.calcBezier = function(p,b){ 247 | var p2 = p*p, 248 | omp = (1-p), // omp = one minus p - smart naming ftw! 249 | omp2 = omp*omp, 250 | f1 = omp*omp2, 251 | f2 = 3*p*omp2, 252 | f3 = 3*p2*omp, 253 | f4 = p*p2; 254 | return [ 255 | Math.round(b[0]*f1 + b[2]*f2 + b[4]*f3 + b[6]*f4), // bezier x 256 | Math.round(b[1]*f1 + b[3]*f2 + b[5]*f3 + b[7]*f4), // bezier y 257 | Math.round(p*1000)/10 // percentage with one decimal place 258 | ]; 259 | }; 260 | 261 | // base.points = [ sx,sy, csxo,csyo, cexo,ceyo, ex,ey ] 262 | // sx,sy = start x & y 263 | // csxo,csyo = control start x & y offset from start point 264 | // cexo,ceyo = control end x & y offset from end point 265 | // ex,ey = end x & y 266 | // convert needed for canvas - using the offset just makes the code easier to read 267 | // base.pointsxy = [ sx, sy, csx, csy, cex, cey, ex, ey ] 268 | base.convert2xy = function(p){ 269 | p = p || base.points; 270 | return [ 271 | p[0], p[1], // start x,y 272 | p[0] + p[2], p[1] + p[3], // start control x,y 273 | p[6] + p[4], p[7] + p[5], // end control x,y 274 | p[6], p[7] // end x,y 275 | ]; 276 | }; 277 | 278 | base.redraw = function(points) { 279 | // update from options 280 | base.points = base.options.points = points || base.options.points; 281 | // store array of x & y positions for cross reference 282 | base.pointsxy = base.convert2xy(); 283 | // update grip 284 | base.update(); 285 | // update curve 286 | base.drawCurve(); 287 | base.setSlider(base.percent, null, true); 288 | }; 289 | 290 | // Make purdy curve 291 | base.drawCurve = function() { 292 | var ctx, grad, tmp, 293 | points = base.pointsxy; 294 | if (!base.$el.find('canvas').length) { 295 | $('').appendTo(base.$el); 296 | // size in attribute needed to keep canvas size in proportion 297 | base.$canvas = base.$el.find('canvas').attr({ width: base.sliderDim[2], height: base.sliderDim[3] }); 298 | base.canvas = base.$canvas[0]; 299 | base.ctx = base.canvas.getContext("2d"); 300 | } 301 | ctx = base.ctx; 302 | ctx.clearRect(0, 0, base.sliderDim[2], base.sliderDim[3]); 303 | ctx.lineCap = o.curve.cap; 304 | ctx.lineJoin = o.curve.cap; 305 | ctx.lineWidth = o.curve.width; 306 | // this can be a gradient or image as well. See 307 | // https://developer.mozilla.org/en-US/docs/Web/API/Canvas_API/Tutorial/Applying_styles_and_colors 308 | if ($.isArray(o.curve.color)) { 309 | grad = ctx.createLinearGradient(points[0], points[1], points[6], points[7]); 310 | tmp = base.percent/100; 311 | grad.addColorStop(0, o.curve.color[0]); 312 | grad.addColorStop(tmp, o.curve.color[0]); 313 | if (tmp + 0.01 <= 1) { tmp += 0.01; } 314 | grad.addColorStop(tmp, o.curve.color[1]); 315 | grad.addColorStop(1, o.curve.color[1]); 316 | ctx.strokeStyle = grad; 317 | } else { 318 | ctx.strokeStyle = o.curve.color; 319 | } 320 | tmp = true; 321 | if (typeof o.drawCanvas === 'function') { 322 | // return anything except false to continue drawing the curve 323 | tmp = o.drawCanvas(base, ctx, points) !== false; 324 | ctx = base.ctx; 325 | } 326 | // tmp returned from drawCanvas; if 327 | if (tmp === true) { 328 | base.finishCurve(ctx, points); 329 | } 330 | }; 331 | 332 | base.finishCurve = function(ctx, points) { 333 | ctx = ctx || base.ctx; 334 | points = points || base.pointsxy; 335 | ctx.beginPath(); 336 | ctx.moveTo(points[0], points[1]); 337 | ctx.bezierCurveTo(points[2], points[3], points[4], points[5], points[6], points[7]); 338 | ctx.stroke(); 339 | }; 340 | 341 | // Run initializer 342 | base.init(); 343 | 344 | }; 345 | 346 | $.pathslider.defaults = { 347 | 348 | // Appearance 349 | gripClass : '', // class added to the grip/handle 350 | rotateGrip : true, // when true, the grip will rotate based on the shape of the path 351 | 352 | // canvas curve styling 353 | useCanvas : true, 354 | curve : { width: 4, color: "#333", cap: "round" }, 355 | 356 | // Usability 357 | // sx,sy = start x & y 358 | // csxo,csyo = control start x & y offset from start point 359 | // cexo,ceyo = control end x & y offset from end point 360 | // ex,ey = end x & y 361 | // [ sx,sy, csxo,csyo, cexo,ceyo, ex,ey ] 362 | points : [ 0,50, 50,-50, -50,-50, 250,50 ], 363 | 364 | value : 50, // starting value - range 0 - 100% 365 | // min : 0, // minimum value on the slider 366 | // max : 100, // maximum value on the slider 367 | // step : 1, // step to use between min and max 368 | 369 | // Tweaking 370 | dataPoints: 100, // Total number of points of the curve to save; increase in increments of 100 to smooth out the grip movement, but not more than 500 (it slows everything down) 371 | tolerance : 3, // distance on the curve from the last position to check; increase this to scroll faster 372 | range : 30 // distance, in pixels, from the cursor to a matching x/y on the curve (should be about the same size as the grip) 373 | 374 | }; 375 | 376 | $.fn.pathslider = function(options, callback){ 377 | return this.each(function(){ 378 | var percent, slider = $(this).data('pathslider'); 379 | 380 | // initialize the slider but prevent multiple initializations 381 | if ((typeof(options)).match('object|undefined')){ 382 | if (!slider) { 383 | (new $.pathslider(this, options)); 384 | } else { 385 | return slider.redraw(); 386 | } 387 | // If options is a number, set percentage 388 | } else if (/\d/.test(options) && !isNaN(options) && slider) { 389 | percent = (typeof(options) === "number") ? options : parseInt($.trim(options),10); // accepts " 2 " 390 | // ignore out of bound percentages 391 | if ( percent >= 0 && percent <= 100 ) { 392 | slider.setSlider(percent, callback); // set percent & callback 393 | } 394 | } 395 | }); 396 | }; 397 | 398 | $.fn.getpathslider = function(){ 399 | return this.data('pathslider'); 400 | }; 401 | 402 | })(jQuery); 403 | -------------------------------------------------------------------------------- /dist/jquery.pathslider.js: -------------------------------------------------------------------------------- 1 | /* jQuery Pathslider v1.0.0 alpha 2 | * By Rob Garrison (Mottie) 3 | * MIT License 4 | */ 5 | (function($){ 6 | $.pathslider = function(el, options){ 7 | 8 | // To avoid scope issues, use 'base' instead of 'this' 9 | // to reference this class from internal events and functions. 10 | var base = this, o; 11 | 12 | // Access to jQuery and DOM versions of element 13 | base.$el = $(el).addClass('pathslider'); 14 | base.el = el; 15 | 16 | // Add a reverse reference to the DOM object 17 | base.$el.data("pathslider", base); 18 | 19 | base.init = function(){ 20 | var t; 21 | base.options = o = $.extend(true, {}, $.pathslider.defaults, options); 22 | 23 | // Is there a canvas? 24 | t = document.createElement('canvas'); 25 | base.hasCanvas = !!(t.getContext && t.getContext('2d')); 26 | base.hasTouch = document.hasOwnProperty("ontouchend"); 27 | 28 | // add grip 29 | base.$grip = $('
    ').appendTo(base.$el); 30 | 31 | // store array of x & y positions for cross reference 32 | base.points = []; 33 | base.pointsxy = []; 34 | base.arrayX = []; 35 | base.arrayY = []; 36 | base.arrayP = []; 37 | 38 | base.rad2deg = 180 / Math.PI; // convert radians to degrees (multiply radian by this value) 39 | base.sliding = false; // flag for dragging element 40 | base.lastPercent = base.percent = o.value; 41 | 42 | // Callbacks 43 | // slide triggered on EVERY mouse move; change triggered on slide stop 44 | $.each('create update start slide change stop'.split(' '), function(i,f){ 45 | if ($.isFunction(o[f])){ 46 | base.$el.bind(f + '.pathslider', o[f]); 47 | } 48 | }); 49 | 50 | $(document) 51 | .bind( base.hasTouch ? 'touchend.pathslider touchcancel.pathslider' : 'mouseup.pathslider mouseleave.pathslider', function(e){ 52 | if (base.sliding) { // && ($(e.target).closest('.pathslider').length || e.type === 'mouseleave')) { 53 | base.$el.trigger('stop.pathslider', [base]); 54 | if (base.lastPercent !== base.percent) { 55 | base.lastPercent = base.percent; 56 | base.$el.trigger('change.pathslider', [base]); 57 | } 58 | } 59 | base.$grip.removeClass('sliding'); 60 | base.sliding = false; 61 | }) 62 | .bind( (base.hasTouch ? 'touchmove' : 'mousemove') + '.pathslider', function(e){ 63 | if (base.sliding) { 64 | base.setSlider( base.findPos(e), null, true ); 65 | } 66 | }); 67 | 68 | $(window) 69 | .bind('resize.pathslider', function(){ 70 | base.update(); 71 | }) 72 | .bind('load', function(){ 73 | // needed because loading images/fonts will shift the page 74 | base.sliderDim[0] = base.$el.offset().left; 75 | base.sliderDim[1] = base.$el.offset().top; 76 | }); 77 | 78 | base.$grip 79 | .bind( (base.hasTouch ? 'touchstart' : 'mousedown') + '.pathslider', function(e){ 80 | base.sliding = true; 81 | $(this).addClass('sliding'); 82 | base.$el.trigger('start.pathslider', [base]); 83 | return false; 84 | }) 85 | .bind('click', function(){ 86 | return false; 87 | }); 88 | 89 | base.redraw(); 90 | 91 | base.$el.trigger('create.pathslider', [base]); 92 | 93 | }; 94 | 95 | // update dimensions & grip position 96 | base.update = function(){ 97 | 98 | // using attr to remove other css grip classes when updating 99 | base.$grip.attr('class', 'pathslider-grip ' + o.gripClass); 100 | 101 | if (base.ctx) { 102 | // clear canvas *before* setting new dimensions 103 | // just in case the new size is smaller than the previous 104 | base.ctx.clearRect(0, 0, base.sliderDim[2], base.sliderDim[3]); 105 | } 106 | 107 | base.sliderDim = [ 108 | base.$el.offset().left, 109 | base.$el.offset().top, 110 | base.$el.width(), 111 | base.$el.height() 112 | ]; 113 | 114 | // get grip dimensions; jQuery v3+ width() & height() return the rotated dimensions 115 | // which we don't want! 116 | var computedStyle = window.getComputedStyle(base.$grip[0]); 117 | // for centering grip 118 | base.gripCenter = [ parseInt(computedStyle.width, 10)/2, parseInt(computedStyle.height, 10)/2 ]; 119 | 120 | // number of data points to store - increase to smooth the animation (based on slider size) 121 | base.dataPoints = o.dataPoints; 122 | // in next update add min/max/step 123 | // base.range = o.max - o.min; 124 | // base.dataPoints = base.range * o.step; 125 | 126 | base.makeArray(); 127 | // save the position in the array of the starting value (roughly) 128 | var t = $.inArray(base.percent, base.arrayP); 129 | base.position = (t === -1) ? Math.round(base.percent/100 * base.dataPoints) : t; 130 | 131 | base.setSlider(base.percent, null, true); 132 | if (base.hasCanvas && o.useCanvas) { base.drawCurve(); } 133 | 134 | }; 135 | 136 | // set position of slider 137 | base.setSlider = function(percent, callback, internal) { 138 | if (!isNaN(percent)) { 139 | // find position on bezier curve; p = percent (range 0 - 100) 140 | // set position of slider without using the array (more precision) 141 | percent = parseFloat(percent, 10); 142 | percent = (percent > 100) ? 100 : percent < 0 ? 0 : percent; 143 | var css, angle, 144 | // pos = $.inArray(percent, base.arrayP), 145 | p = base.calcBezier(percent/100, base.pointsxy), 146 | pm1 = (percent - 2 > 0) ? base.calcBezier( (percent-2)/100, base.pointsxy ) : p, 147 | pp1 = (percent + 2 < 100) ? base.calcBezier( (percent+2)/100, base.pointsxy ) : p, 148 | // m = slope of tangent - used to change rotation angle of the grip 149 | // yes, I could have used the cubic derivative, but this is less math 150 | m = (pp1[0] - pm1[0] === 0) ? 90 : (pp1[1] - pm1[1])/(pp1[0] - pm1[0]); 151 | base.angle = parseInt(Math.atan(m) * base.rad2deg, 10); 152 | angle = 'rotate(' + base.angle + 'deg)'; 153 | css = (o.rotateGrip) ? { 154 | '-webkit-transform' : angle, 155 | 'transform' : angle 156 | } : {}; 157 | css.left = p[0] - base.gripCenter[0]; 158 | css.top = p[1] - base.gripCenter[1]; 159 | base.$grip 160 | .attr({ 161 | 'data-angle' : base.angle, 162 | 'data-percent' : percent 163 | }) 164 | .css(css); 165 | // find closest percent in the array - this relies on there being a factor 166 | // of 100 datapoints, so it'll need changing when we have a min/max/step 167 | base.percent = percent; // Math.round(percent*r)/r; 168 | if ((percent !== base.lastPercent && !base.sliding) || !internal) { 169 | base.$el.trigger('change.pathslider', [base]); 170 | } 171 | } 172 | if (typeof callback === 'function') { callback(base); } 173 | }; 174 | 175 | // relative mouse position 176 | base.mousePos = function(e) { 177 | return [ 178 | (e.originalEvent.touches ? e.originalEvent.touches[0].pageX : e.pageX) - base.sliderDim[0], 179 | (e.originalEvent.touches ? e.originalEvent.touches[0].pageY : e.pageY) - base.sliderDim[1] 180 | ]; 181 | }; 182 | 183 | // find percentage given the x,y coordinates 184 | // searching through a set array of points starting from the last known position 185 | // This allows the curve to loop over itself without mixing up intersecting points 186 | // The biggest issue is a very sharp turn 187 | base.findPos = function(event) { 188 | var i, j, dx, dy, px = [], py = [], 189 | last = base.position, //* base.dataPoints / 100, 190 | // check x & y cross ref based on nearby positions (+/- tolerance) 191 | t = parseInt(o.tolerance + 1, 10) || 2, // tolerance of 1 is too small 192 | r = parseInt(o.range, 10) || base.gripCenter[0], // set to 1/2 width of grip 193 | pos = base.mousePos(event); 194 | // save percent 195 | for ( i=0; i < r; i++ ){ 196 | px = []; py = []; 197 | for ( j=0; j < t + 1; j++ ){ 198 | // check positive direction 199 | dx = Math.abs(base.arrayX[last+j] - pos[0]) <= i; 200 | dy = Math.abs(base.arrayY[last+j] - pos[1]) <= i; 201 | if (dx && dy) { return base.returnPos(last+j); } 202 | if (dx) { px.push(last+j); } 203 | if (dy) { py.push(last+j); } 204 | // check in negative direction 205 | dx = Math.abs(base.arrayX[last-j] - pos[0]) <= i; 206 | dy = Math.abs(base.arrayY[last-j] - pos[1]) <= i; 207 | if (dx && dy) { return base.returnPos(last-j); } 208 | if (dx) { px.push(last-j); } 209 | if (dy) { py.push(last-j); } 210 | } 211 | if (px.length === 1 && py.length > 1) { return base.returnPos(px[0]); } 212 | if (py.length === 1 && px.length > 1) { return base.returnPos(py[0]); } 213 | } 214 | return base.returnPos(last); 215 | }; 216 | 217 | // return found position & trigger slide event 218 | base.returnPos = function(p) { 219 | var t = base.position === p; 220 | base.percent = base.arrayP[p]; 221 | base.position = p; 222 | if (!t) { 223 | if (base.hasCanvas && o.useCanvas) { 224 | base.drawCurve(); 225 | } 226 | base.$el.trigger('slide.pathslider', [base] ); 227 | } 228 | return base.percent; 229 | }; 230 | 231 | // build cross-ref array - find position based on x,y coords 232 | base.makeArray = function(){ 233 | var i, t, b = base.pointsxy, 234 | n = base.dataPoints; 235 | for ( i=0; i < n+1; i++ ){ 236 | t = base.calcBezier(i/n, b); 237 | base.arrayX[i] = t[0]; 238 | base.arrayY[i] = t[1]; 239 | base.arrayP[i] = t[2]; 240 | } 241 | }; 242 | 243 | // Calculate bezier x & y based on percentage (p) 244 | // cubic bezier = start(p^3) + cstart(3*p^2*(1−p)) + cend(3*p*(1−p)^2) + end(1−p)^3 245 | // b = [ startx,starty, cstartx,cstarty, cendx,cendy, endx,endy ] 246 | base.calcBezier = function(p,b){ 247 | var p2 = p*p, 248 | omp = (1-p), // omp = one minus p - smart naming ftw! 249 | omp2 = omp*omp, 250 | f1 = omp*omp2, 251 | f2 = 3*p*omp2, 252 | f3 = 3*p2*omp, 253 | f4 = p*p2; 254 | return [ 255 | Math.round(b[0]*f1 + b[2]*f2 + b[4]*f3 + b[6]*f4), // bezier x 256 | Math.round(b[1]*f1 + b[3]*f2 + b[5]*f3 + b[7]*f4), // bezier y 257 | Math.round(p*1000)/10 // percentage with one decimal place 258 | ]; 259 | }; 260 | 261 | // base.points = [ sx,sy, csxo,csyo, cexo,ceyo, ex,ey ] 262 | // sx,sy = start x & y 263 | // csxo,csyo = control start x & y offset from start point 264 | // cexo,ceyo = control end x & y offset from end point 265 | // ex,ey = end x & y 266 | // convert needed for canvas - using the offset just makes the code easier to read 267 | // base.pointsxy = [ sx, sy, csx, csy, cex, cey, ex, ey ] 268 | base.convert2xy = function(p){ 269 | p = p || base.points; 270 | return [ 271 | p[0], p[1], // start x,y 272 | p[0] + p[2], p[1] + p[3], // start control x,y 273 | p[6] + p[4], p[7] + p[5], // end control x,y 274 | p[6], p[7] // end x,y 275 | ]; 276 | }; 277 | 278 | base.redraw = function(points) { 279 | // update from options 280 | base.points = base.options.points = points || base.options.points; 281 | // store array of x & y positions for cross reference 282 | base.pointsxy = base.convert2xy(); 283 | // update grip 284 | base.update(); 285 | // update curve 286 | base.drawCurve(); 287 | base.setSlider(base.percent, null, true); 288 | }; 289 | 290 | // Make purdy curve 291 | base.drawCurve = function() { 292 | var ctx, grad, tmp, 293 | points = base.pointsxy; 294 | if (!base.$el.find('canvas').length) { 295 | $('').appendTo(base.$el); 296 | // size in attribute needed to keep canvas size in proportion 297 | base.$canvas = base.$el.find('canvas').attr({ width: base.sliderDim[2], height: base.sliderDim[3] }); 298 | base.canvas = base.$canvas[0]; 299 | base.ctx = base.canvas.getContext("2d"); 300 | } 301 | ctx = base.ctx; 302 | ctx.clearRect(0, 0, base.sliderDim[2], base.sliderDim[3]); 303 | ctx.lineCap = o.curve.cap; 304 | ctx.lineJoin = o.curve.cap; 305 | ctx.lineWidth = o.curve.width; 306 | // this can be a gradient or image as well. See 307 | // https://developer.mozilla.org/en-US/docs/Web/API/Canvas_API/Tutorial/Applying_styles_and_colors 308 | if ($.isArray(o.curve.color)) { 309 | grad = ctx.createLinearGradient(points[0], points[1], points[6], points[7]); 310 | tmp = base.percent/100; 311 | grad.addColorStop(0, o.curve.color[0]); 312 | grad.addColorStop(tmp, o.curve.color[0]); 313 | if (tmp + 0.01 <= 1) { tmp += 0.01; } 314 | grad.addColorStop(tmp, o.curve.color[1]); 315 | grad.addColorStop(1, o.curve.color[1]); 316 | ctx.strokeStyle = grad; 317 | } else { 318 | ctx.strokeStyle = o.curve.color; 319 | } 320 | tmp = true; 321 | if (typeof o.drawCanvas === 'function') { 322 | // return anything except false to continue drawing the curve 323 | tmp = o.drawCanvas(base, ctx, points) !== false; 324 | ctx = base.ctx; 325 | } 326 | // tmp returned from drawCanvas; if 327 | if (tmp === true) { 328 | base.finishCurve(ctx, points); 329 | } 330 | }; 331 | 332 | base.finishCurve = function(ctx, points) { 333 | ctx = ctx || base.ctx; 334 | points = points || base.pointsxy; 335 | ctx.beginPath(); 336 | ctx.moveTo(points[0], points[1]); 337 | ctx.bezierCurveTo(points[2], points[3], points[4], points[5], points[6], points[7]); 338 | ctx.stroke(); 339 | }; 340 | 341 | // Run initializer 342 | base.init(); 343 | 344 | }; 345 | 346 | $.pathslider.defaults = { 347 | 348 | // Appearance 349 | gripClass : '', // class added to the grip/handle 350 | rotateGrip : true, // when true, the grip will rotate based on the shape of the path 351 | 352 | // canvas curve styling 353 | useCanvas : true, 354 | curve : { width: 4, color: "#333", cap: "round" }, 355 | 356 | // Usability 357 | // sx,sy = start x & y 358 | // csxo,csyo = control start x & y offset from start point 359 | // cexo,ceyo = control end x & y offset from end point 360 | // ex,ey = end x & y 361 | // [ sx,sy, csxo,csyo, cexo,ceyo, ex,ey ] 362 | points : [ 0,50, 50,-50, -50,-50, 250,50 ], 363 | 364 | value : 50, // starting value - range 0 - 100% 365 | // min : 0, // minimum value on the slider 366 | // max : 100, // maximum value on the slider 367 | // step : 1, // step to use between min and max 368 | 369 | // Tweaking 370 | dataPoints: 100, // Total number of points of the curve to save; increase in increments of 100 to smooth out the grip movement, but not more than 500 (it slows everything down) 371 | tolerance : 3, // distance on the curve from the last position to check; increase this to scroll faster 372 | range : 30 // distance, in pixels, from the cursor to a matching x/y on the curve (should be about the same size as the grip) 373 | 374 | }; 375 | 376 | $.fn.pathslider = function(options, callback){ 377 | return this.each(function(){ 378 | var percent, slider = $(this).data('pathslider'); 379 | 380 | // initialize the slider but prevent multiple initializations 381 | if ((typeof(options)).match('object|undefined')){ 382 | if (!slider) { 383 | (new $.pathslider(this, options)); 384 | } else { 385 | return slider.redraw(); 386 | } 387 | // If options is a number, set percentage 388 | } else if (/\d/.test(options) && !isNaN(options) && slider) { 389 | percent = (typeof(options) === "number") ? options : parseInt($.trim(options),10); // accepts " 2 " 390 | // ignore out of bound percentages 391 | if ( percent >= 0 && percent <= 100 ) { 392 | slider.setSlider(percent, callback); // set percent & callback 393 | } 394 | } 395 | }); 396 | }; 397 | 398 | $.fn.getpathslider = function(){ 399 | return this.data('pathslider'); 400 | }; 401 | 402 | })(jQuery); 403 | -------------------------------------------------------------------------------- /demo/colorpicker/colorpicker.js: -------------------------------------------------------------------------------- 1 | /** 2 | * 3 | * Color picker 4 | * Author: Stefan Petre www.eyecon.ro 5 | * 6 | * Dual licensed under the MIT and GPL licenses 7 | * 8 | */ 9 | (function ($) { 10 | var ColorPicker = function () { 11 | var 12 | ids = {}, 13 | inAction, 14 | charMin = 65, 15 | visible, 16 | tpl = '
    ', 17 | defaults = { 18 | eventName: 'click', 19 | onShow: function () {}, 20 | onBeforeShow: function(){}, 21 | onHide: function () {}, 22 | onChange: function () {}, 23 | onSubmit: function () {}, 24 | color: 'ff0000', 25 | livePreview: true, 26 | flat: false 27 | }, 28 | fillRGBFields = function (hsb, cal) { 29 | var rgb = HSBToRGB(hsb); 30 | $(cal).data('colorpicker').fields 31 | .eq(1).val(rgb.r).end() 32 | .eq(2).val(rgb.g).end() 33 | .eq(3).val(rgb.b).end(); 34 | }, 35 | fillHSBFields = function (hsb, cal) { 36 | $(cal).data('colorpicker').fields 37 | .eq(4).val(hsb.h).end() 38 | .eq(5).val(hsb.s).end() 39 | .eq(6).val(hsb.b).end(); 40 | }, 41 | fillHexFields = function (hsb, cal) { 42 | $(cal).data('colorpicker').fields 43 | .eq(0).val(HSBToHex(hsb)).end(); 44 | }, 45 | setSelector = function (hsb, cal) { 46 | $(cal).data('colorpicker').selector.css('backgroundColor', '#' + HSBToHex({h: hsb.h, s: 100, b: 100})); 47 | $(cal).data('colorpicker').selectorIndic.css({ 48 | left: parseInt(150 * hsb.s/100, 10), 49 | top: parseInt(150 * (100-hsb.b)/100, 10) 50 | }); 51 | }, 52 | setHue = function (hsb, cal) { 53 | $(cal).data('colorpicker').hue.css('top', parseInt(150 - 150 * hsb.h/360, 10)); 54 | }, 55 | setCurrentColor = function (hsb, cal) { 56 | $(cal).data('colorpicker').currentColor.css('backgroundColor', '#' + HSBToHex(hsb)); 57 | }, 58 | setNewColor = function (hsb, cal) { 59 | $(cal).data('colorpicker').newColor.css('backgroundColor', '#' + HSBToHex(hsb)); 60 | }, 61 | keyDown = function (ev) { 62 | var pressedKey = ev.charCode || ev.keyCode || -1; 63 | if ((pressedKey > charMin && pressedKey <= 90) || pressedKey == 32) { 64 | return false; 65 | } 66 | var cal = $(this).parent().parent(); 67 | if (cal.data('colorpicker').livePreview === true) { 68 | change.apply(this); 69 | } 70 | }, 71 | change = function (ev) { 72 | var cal = $(this).parent().parent(), col; 73 | if (this.parentNode.className.indexOf('_hex') > 0) { 74 | cal.data('colorpicker').color = col = HexToHSB(fixHex(this.value)); 75 | } else if (this.parentNode.className.indexOf('_hsb') > 0) { 76 | cal.data('colorpicker').color = col = fixHSB({ 77 | h: parseInt(cal.data('colorpicker').fields.eq(4).val(), 10), 78 | s: parseInt(cal.data('colorpicker').fields.eq(5).val(), 10), 79 | b: parseInt(cal.data('colorpicker').fields.eq(6).val(), 10) 80 | }); 81 | } else { 82 | cal.data('colorpicker').color = col = RGBToHSB(fixRGB({ 83 | r: parseInt(cal.data('colorpicker').fields.eq(1).val(), 10), 84 | g: parseInt(cal.data('colorpicker').fields.eq(2).val(), 10), 85 | b: parseInt(cal.data('colorpicker').fields.eq(3).val(), 10) 86 | })); 87 | } 88 | if (ev) { 89 | fillRGBFields(col, cal.get(0)); 90 | fillHexFields(col, cal.get(0)); 91 | fillHSBFields(col, cal.get(0)); 92 | } 93 | setSelector(col, cal.get(0)); 94 | setHue(col, cal.get(0)); 95 | setNewColor(col, cal.get(0)); 96 | cal.data('colorpicker').onChange.apply(cal, [col, HSBToHex(col), HSBToRGB(col)]); 97 | }, 98 | blur = function (ev) { 99 | var cal = $(this).parent().parent(); 100 | cal.data('colorpicker').fields.parent().removeClass('colorpicker_focus'); 101 | }, 102 | focus = function () { 103 | charMin = this.parentNode.className.indexOf('_hex') > 0 ? 70 : 65; 104 | $(this).parent().parent().data('colorpicker').fields.parent().removeClass('colorpicker_focus'); 105 | $(this).parent().addClass('colorpicker_focus'); 106 | }, 107 | downIncrement = function (ev) { 108 | var field = $(this).parent().find('input').focus(); 109 | var current = { 110 | el: $(this).parent().addClass('colorpicker_slider'), 111 | max: this.parentNode.className.indexOf('_hsb_h') > 0 ? 360 : (this.parentNode.className.indexOf('_hsb') > 0 ? 100 : 255), 112 | y: ev.pageY, 113 | field: field, 114 | val: parseInt(field.val(), 10), 115 | preview: $(this).parent().parent().data('colorpicker').livePreview 116 | }; 117 | $(document).bind('mouseup', current, upIncrement); 118 | $(document).bind('mousemove', current, moveIncrement); 119 | }, 120 | moveIncrement = function (ev) { 121 | ev.data.field.val(Math.max(0, Math.min(ev.data.max, parseInt(ev.data.val + ev.pageY - ev.data.y, 10)))); 122 | if (ev.data.preview) { 123 | change.apply(ev.data.field.get(0), [true]); 124 | } 125 | return false; 126 | }, 127 | upIncrement = function (ev) { 128 | change.apply(ev.data.field.get(0), [true]); 129 | ev.data.el.removeClass('colorpicker_slider').find('input').focus(); 130 | $(document).unbind('mouseup', upIncrement); 131 | $(document).unbind('mousemove', moveIncrement); 132 | return false; 133 | }, 134 | downHue = function (ev) { 135 | var current = { 136 | cal: $(this).parent(), 137 | y: $(this).offset().top 138 | }; 139 | current.preview = current.cal.data('colorpicker').livePreview; 140 | $(document).bind('mouseup', current, upHue); 141 | $(document).bind('mousemove', current, moveHue); 142 | }, 143 | moveHue = function (ev) { 144 | change.apply( 145 | ev.data.cal.data('colorpicker') 146 | .fields 147 | .eq(4) 148 | .val(parseInt(360*(150 - Math.max(0,Math.min(150,(ev.pageY - ev.data.y))))/150, 10)) 149 | .get(0), 150 | [ev.data.preview] 151 | ); 152 | return false; 153 | }, 154 | upHue = function (ev) { 155 | fillRGBFields(ev.data.cal.data('colorpicker').color, ev.data.cal.get(0)); 156 | fillHexFields(ev.data.cal.data('colorpicker').color, ev.data.cal.get(0)); 157 | $(document).unbind('mouseup', upHue); 158 | $(document).unbind('mousemove', moveHue); 159 | return false; 160 | }, 161 | downSelector = function (ev) { 162 | var current = { 163 | cal: $(this).parent(), 164 | pos: $(this).offset() 165 | }; 166 | current.preview = current.cal.data('colorpicker').livePreview; 167 | $(document).bind('mouseup', current, upSelector); 168 | $(document).bind('mousemove', current, moveSelector); 169 | }, 170 | moveSelector = function (ev) { 171 | change.apply( 172 | ev.data.cal.data('colorpicker') 173 | .fields 174 | .eq(6) 175 | .val(parseInt(100*(150 - Math.max(0,Math.min(150,(ev.pageY - ev.data.pos.top))))/150, 10)) 176 | .end() 177 | .eq(5) 178 | .val(parseInt(100*(Math.max(0,Math.min(150,(ev.pageX - ev.data.pos.left))))/150, 10)) 179 | .get(0), 180 | [ev.data.preview] 181 | ); 182 | return false; 183 | }, 184 | upSelector = function (ev) { 185 | fillRGBFields(ev.data.cal.data('colorpicker').color, ev.data.cal.get(0)); 186 | fillHexFields(ev.data.cal.data('colorpicker').color, ev.data.cal.get(0)); 187 | $(document).unbind('mouseup', upSelector); 188 | $(document).unbind('mousemove', moveSelector); 189 | return false; 190 | }, 191 | enterSubmit = function (ev) { 192 | $(this).addClass('colorpicker_focus'); 193 | }, 194 | leaveSubmit = function (ev) { 195 | $(this).removeClass('colorpicker_focus'); 196 | }, 197 | clickSubmit = function (ev) { 198 | var cal = $(this).parent(); 199 | var col = cal.data('colorpicker').color; 200 | cal.data('colorpicker').origColor = col; 201 | setCurrentColor(col, cal.get(0)); 202 | cal.data('colorpicker').onSubmit(col, HSBToHex(col), HSBToRGB(col), cal.data('colorpicker').el); 203 | }, 204 | show = function (ev) { 205 | var cal = $('#' + $(this).data('colorpickerId')); 206 | cal.data('colorpicker').onBeforeShow.apply(this, [cal.get(0)]); 207 | var pos = $(this).offset(); 208 | var viewPort = getViewport(); 209 | var top = pos.top + this.offsetHeight; 210 | var left = pos.left; 211 | if (top + 176 > viewPort.t + viewPort.h) { 212 | top -= this.offsetHeight + 176; 213 | } 214 | if (left + 356 > viewPort.l + viewPort.w) { 215 | left -= 356; 216 | } 217 | cal.css({left: left + 'px', top: top + 'px'}); 218 | if (cal.data('colorpicker').onShow.apply(this, [cal.get(0)]) != false) { 219 | cal.show(); 220 | } 221 | $(document).bind('mousedown', {cal: cal}, hide); 222 | return false; 223 | }, 224 | hide = function (ev) { 225 | if (!isChildOf(ev.data.cal.get(0), ev.target, ev.data.cal.get(0))) { 226 | if (ev.data.cal.data('colorpicker').onHide.apply(this, [ev.data.cal.get(0)]) != false) { 227 | ev.data.cal.hide(); 228 | } 229 | $(document).unbind('mousedown', hide); 230 | } 231 | }, 232 | isChildOf = function(parentEl, el, container) { 233 | if (parentEl == el) { 234 | return true; 235 | } 236 | if (parentEl.contains) { 237 | return parentEl.contains(el); 238 | } 239 | if ( parentEl.compareDocumentPosition ) { 240 | return !!(parentEl.compareDocumentPosition(el) & 16); 241 | } 242 | var prEl = el.parentNode; 243 | while(prEl && prEl != container) { 244 | if (prEl == parentEl) 245 | return true; 246 | prEl = prEl.parentNode; 247 | } 248 | return false; 249 | }, 250 | getViewport = function () { 251 | var m = document.compatMode == 'CSS1Compat'; 252 | return { 253 | l : window.pageXOffset || (m ? document.documentElement.scrollLeft : document.body.scrollLeft), 254 | t : window.pageYOffset || (m ? document.documentElement.scrollTop : document.body.scrollTop), 255 | w : window.innerWidth || (m ? document.documentElement.clientWidth : document.body.clientWidth), 256 | h : window.innerHeight || (m ? document.documentElement.clientHeight : document.body.clientHeight) 257 | }; 258 | }, 259 | fixHSB = function (hsb) { 260 | return { 261 | h: Math.min(360, Math.max(0, hsb.h)), 262 | s: Math.min(100, Math.max(0, hsb.s)), 263 | b: Math.min(100, Math.max(0, hsb.b)) 264 | }; 265 | }, 266 | fixRGB = function (rgb) { 267 | return { 268 | r: Math.min(255, Math.max(0, rgb.r)), 269 | g: Math.min(255, Math.max(0, rgb.g)), 270 | b: Math.min(255, Math.max(0, rgb.b)) 271 | }; 272 | }, 273 | fixHex = function (hex) { 274 | var len = 6 - hex.length; 275 | if (len > 0) { 276 | var o = []; 277 | for (var i=0; i -1) ? hex.substring(1) : hex), 16); 287 | return {r: hex >> 16, g: (hex & 0x00FF00) >> 8, b: (hex & 0x0000FF)}; 288 | }, 289 | HexToHSB = function (hex) { 290 | return RGBToHSB(HexToRGB(hex)); 291 | }, 292 | RGBToHSB = function (rgb) { 293 | var hsb = { 294 | h: 0, 295 | s: 0, 296 | b: 0 297 | }; 298 | var min = Math.min(rgb.r, rgb.g, rgb.b); 299 | var max = Math.max(rgb.r, rgb.g, rgb.b); 300 | var delta = max - min; 301 | hsb.b = max; 302 | if (max != 0) { 303 | 304 | } 305 | hsb.s = max != 0 ? 255 * delta / max : 0; 306 | if (hsb.s != 0) { 307 | if (rgb.r == max) { 308 | hsb.h = (rgb.g - rgb.b) / delta; 309 | } else if (rgb.g == max) { 310 | hsb.h = 2 + (rgb.b - rgb.r) / delta; 311 | } else { 312 | hsb.h = 4 + (rgb.r - rgb.g) / delta; 313 | } 314 | } else { 315 | hsb.h = -1; 316 | } 317 | hsb.h *= 60; 318 | if (hsb.h < 0) { 319 | hsb.h += 360; 320 | } 321 | hsb.s *= 100/255; 322 | hsb.b *= 100/255; 323 | return hsb; 324 | }, 325 | HSBToRGB = function (hsb) { 326 | var rgb = {}; 327 | var h = Math.round(hsb.h); 328 | var s = Math.round(hsb.s*255/100); 329 | var v = Math.round(hsb.b*255/100); 330 | if(s == 0) { 331 | rgb.r = rgb.g = rgb.b = v; 332 | } else { 333 | var t1 = v; 334 | var t2 = (255-s)*v/255; 335 | var t3 = (t1-t2)*(h%60)/60; 336 | if(h==360) h = 0; 337 | if(h<60) {rgb.r=t1; rgb.b=t2; rgb.g=t2+t3} 338 | else if(h<120) {rgb.g=t1; rgb.b=t2; rgb.r=t1-t3} 339 | else if(h<180) {rgb.g=t1; rgb.r=t2; rgb.b=t2+t3} 340 | else if(h<240) {rgb.b=t1; rgb.r=t2; rgb.g=t1-t3} 341 | else if(h<300) {rgb.b=t1; rgb.g=t2; rgb.r=t2+t3} 342 | else if(h<360) {rgb.r=t1; rgb.g=t2; rgb.b=t1-t3} 343 | else {rgb.r=0; rgb.g=0; rgb.b=0} 344 | } 345 | return {r:Math.round(rgb.r), g:Math.round(rgb.g), b:Math.round(rgb.b)}; 346 | }, 347 | RGBToHex = function (rgb) { 348 | var hex = [ 349 | rgb.r.toString(16), 350 | rgb.g.toString(16), 351 | rgb.b.toString(16) 352 | ]; 353 | $.each(hex, function (nr, val) { 354 | if (val.length == 1) { 355 | hex[nr] = '0' + val; 356 | } 357 | }); 358 | return hex.join(''); 359 | }, 360 | HSBToHex = function (hsb) { 361 | return RGBToHex(HSBToRGB(hsb)); 362 | }, 363 | restoreOriginal = function () { 364 | var cal = $(this).parent(); 365 | var col = cal.data('colorpicker').origColor; 366 | cal.data('colorpicker').color = col; 367 | fillRGBFields(col, cal.get(0)); 368 | fillHexFields(col, cal.get(0)); 369 | fillHSBFields(col, cal.get(0)); 370 | setSelector(col, cal.get(0)); 371 | setHue(col, cal.get(0)); 372 | setNewColor(col, cal.get(0)); 373 | }; 374 | return { 375 | init: function (opt) { 376 | opt = $.extend({}, defaults, opt||{}); 377 | if (typeof opt.color == 'string') { 378 | opt.color = HexToHSB(opt.color); 379 | } else if (opt.color.r != undefined && opt.color.g != undefined && opt.color.b != undefined) { 380 | opt.color = RGBToHSB(opt.color); 381 | } else if (opt.color.h != undefined && opt.color.s != undefined && opt.color.b != undefined) { 382 | opt.color = fixHSB(opt.color); 383 | } else { 384 | return this; 385 | } 386 | return this.each(function () { 387 | if (!$(this).data('colorpickerId')) { 388 | var options = $.extend({}, opt); 389 | options.origColor = opt.color; 390 | var id = 'collorpicker_' + parseInt(Math.random() * 1000); 391 | $(this).data('colorpickerId', id); 392 | var cal = $(tpl).attr('id', id); 393 | if (options.flat) { 394 | cal.appendTo(this).show(); 395 | } else { 396 | cal.appendTo(document.body); 397 | } 398 | options.fields = cal 399 | .find('input') 400 | .bind('keyup', keyDown) 401 | .bind('change', change) 402 | .bind('blur', blur) 403 | .bind('focus', focus); 404 | cal 405 | .find('span').bind('mousedown', downIncrement).end() 406 | .find('>div.colorpicker_current_color').bind('click', restoreOriginal); 407 | options.selector = cal.find('div.colorpicker_color').bind('mousedown', downSelector); 408 | options.selectorIndic = options.selector.find('div div'); 409 | options.el = this; 410 | options.hue = cal.find('div.colorpicker_hue div'); 411 | cal.find('div.colorpicker_hue').bind('mousedown', downHue); 412 | options.newColor = cal.find('div.colorpicker_new_color'); 413 | options.currentColor = cal.find('div.colorpicker_current_color'); 414 | cal.data('colorpicker', options); 415 | cal.find('div.colorpicker_submit') 416 | .bind('mouseenter', enterSubmit) 417 | .bind('mouseleave', leaveSubmit) 418 | .bind('click', clickSubmit); 419 | fillRGBFields(options.color, cal.get(0)); 420 | fillHSBFields(options.color, cal.get(0)); 421 | fillHexFields(options.color, cal.get(0)); 422 | setHue(options.color, cal.get(0)); 423 | setSelector(options.color, cal.get(0)); 424 | setCurrentColor(options.color, cal.get(0)); 425 | setNewColor(options.color, cal.get(0)); 426 | if (options.flat) { 427 | cal.css({ 428 | position: 'relative', 429 | display: 'block' 430 | }); 431 | } else { 432 | $(this).bind(options.eventName, show); 433 | } 434 | } 435 | }); 436 | }, 437 | showPicker: function() { 438 | return this.each( function () { 439 | if ($(this).data('colorpickerId')) { 440 | show.apply(this); 441 | } 442 | }); 443 | }, 444 | hidePicker: function() { 445 | return this.each( function () { 446 | if ($(this).data('colorpickerId')) { 447 | $('#' + $(this).data('colorpickerId')).hide(); 448 | } 449 | }); 450 | }, 451 | setColor: function(col) { 452 | if (typeof col == 'string') { 453 | col = HexToHSB(col); 454 | } else if (col.r != undefined && col.g != undefined && col.b != undefined) { 455 | col = RGBToHSB(col); 456 | } else if (col.h != undefined && col.s != undefined && col.b != undefined) { 457 | col = fixHSB(col); 458 | } else { 459 | return this; 460 | } 461 | return this.each(function(){ 462 | if ($(this).data('colorpickerId')) { 463 | var cal = $('#' + $(this).data('colorpickerId')); 464 | cal.data('colorpicker').color = col; 465 | cal.data('colorpicker').origColor = col; 466 | fillRGBFields(col, cal.get(0)); 467 | fillHSBFields(col, cal.get(0)); 468 | fillHexFields(col, cal.get(0)); 469 | setHue(col, cal.get(0)); 470 | setSelector(col, cal.get(0)); 471 | setCurrentColor(col, cal.get(0)); 472 | setNewColor(col, cal.get(0)); 473 | } 474 | }); 475 | } 476 | }; 477 | }(); 478 | $.fn.extend({ 479 | ColorPicker: ColorPicker.init, 480 | ColorPickerHide: ColorPicker.hidePicker, 481 | ColorPickerShow: ColorPicker.showPicker, 482 | ColorPickerSetColor: ColorPicker.setColor 483 | }); 484 | })(jQuery) --------------------------------------------------------------------------------