sofa_gallery 0.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (66) hide show
  1. data/.document +5 -0
  2. data/Gemfile +9 -0
  3. data/Gemfile.lock +106 -0
  4. data/LICENSE +20 -0
  5. data/README.md +30 -0
  6. data/Rakefile +21 -0
  7. data/VERSION +1 -0
  8. data/app/assets/images/sofa_gallery/jcrop.gif +0 -0
  9. data/app/assets/javascripts/sofa_gallery/application.js +15 -0
  10. data/app/assets/javascripts/sofa_gallery/jquery.jcrop.js +246 -0
  11. data/app/assets/javascripts/sofa_gallery/jquery.js +18 -0
  12. data/app/assets/javascripts/sofa_gallery/jquery_ui.js +248 -0
  13. data/app/assets/javascripts/sofa_gallery/rails.js +315 -0
  14. data/app/assets/stylesheets/sofa_gallery/application.css +68 -0
  15. data/app/assets/stylesheets/sofa_gallery/jquery.jcrop.css +32 -0
  16. data/app/assets/stylesheets/sofa_gallery/reset.css +1 -0
  17. data/app/controllers/application_controller.rb +6 -0
  18. data/app/controllers/gallery_admin/base_controller.rb +3 -0
  19. data/app/controllers/gallery_admin/galleries_controller.rb +59 -0
  20. data/app/controllers/gallery_admin/photos_controller.rb +76 -0
  21. data/app/helpers/sofa_gallery_helper.rb +11 -0
  22. data/app/models/sofa/gallery.rb +17 -0
  23. data/app/models/sofa/photo.rb +64 -0
  24. data/app/views/gallery_admin/_navigation.html.erb +1 -0
  25. data/app/views/gallery_admin/galleries/_form.html.erb +11 -0
  26. data/app/views/gallery_admin/galleries/edit.html.erb +5 -0
  27. data/app/views/gallery_admin/galleries/index.html.erb +28 -0
  28. data/app/views/gallery_admin/galleries/new.html.erb +5 -0
  29. data/app/views/gallery_admin/photos/_form.html.erb +13 -0
  30. data/app/views/gallery_admin/photos/crop.html.erb +32 -0
  31. data/app/views/gallery_admin/photos/edit.html.erb +5 -0
  32. data/app/views/gallery_admin/photos/index.html.erb +24 -0
  33. data/app/views/gallery_admin/photos/new.html.erb +5 -0
  34. data/app/views/layouts/gallery_admin/application.html.erb +16 -0
  35. data/config.ru +4 -0
  36. data/config/application.rb +51 -0
  37. data/config/boot.rb +13 -0
  38. data/config/database.yml +16 -0
  39. data/config/environment.rb +5 -0
  40. data/config/environments/development.rb +25 -0
  41. data/config/environments/production.rb +52 -0
  42. data/config/environments/test.rb +39 -0
  43. data/config/initializers/paperclip.rb +3 -0
  44. data/config/initializers/sofa_gallery.rb +15 -0
  45. data/config/routes.rb +12 -0
  46. data/db/migrate/01_create_sofa_gallery.rb +35 -0
  47. data/lib/generators/README +10 -0
  48. data/lib/generators/sofa_gallery_generator.rb +33 -0
  49. data/lib/paperclip_processors/cropper.rb +18 -0
  50. data/lib/sofa_gallery.rb +23 -0
  51. data/lib/sofa_gallery/configuration.rb +26 -0
  52. data/lib/sofa_gallery/engine.rb +21 -0
  53. data/lib/sofa_gallery/form_builder.rb +50 -0
  54. data/script/rails +6 -0
  55. data/sofa_gallery.gemspec +113 -0
  56. data/test/fixtures/files/default.jpg +0 -0
  57. data/test/fixtures/files/default.txt +1 -0
  58. data/test/fixtures/sofa/galleries.yml +10 -0
  59. data/test/fixtures/sofa/photos.yml +9 -0
  60. data/test/functional/galleries_controller_test.rb +87 -0
  61. data/test/functional/photos_controller_test.rb +123 -0
  62. data/test/test_helper.rb +36 -0
  63. data/test/unit/configuration_test.rb +18 -0
  64. data/test/unit/gallery_test.rb +33 -0
  65. data/test/unit/photo_test.rb +29 -0
  66. metadata +167 -0
data/.document ADDED
@@ -0,0 +1,5 @@
1
+ lib/**/*.rb
2
+ bin/*
3
+ -
4
+ features/**/*.feature
5
+ LICENSE.txt
data/Gemfile ADDED
@@ -0,0 +1,9 @@
1
+ source 'http://rubygems.org'
2
+
3
+ gem 'rails', '>=3.1.0.rc4'
4
+ gem 'paperclip', :git => 'https://github.com/thoughtbot/paperclip.git'
5
+
6
+ group :development do
7
+ gem 'sqlite3'
8
+ gem 'jeweler'
9
+ end
data/Gemfile.lock ADDED
@@ -0,0 +1,106 @@
1
+ GIT
2
+ remote: https://github.com/thoughtbot/paperclip.git
3
+ revision: cbba34c2b9abb112bb9bae269edb2c8ab01adf51
4
+ specs:
5
+ paperclip (2.3.10)
6
+ activerecord (>= 2.3.0)
7
+ activesupport (>= 2.3.2)
8
+ cocaine (>= 0.0.2)
9
+
10
+ GEM
11
+ remote: http://rubygems.org/
12
+ specs:
13
+ actionmailer (3.1.0.rc4)
14
+ actionpack (= 3.1.0.rc4)
15
+ mail (~> 2.3.0)
16
+ actionpack (3.1.0.rc4)
17
+ activemodel (= 3.1.0.rc4)
18
+ activesupport (= 3.1.0.rc4)
19
+ builder (~> 3.0.0)
20
+ erubis (~> 2.7.0)
21
+ i18n (~> 0.6)
22
+ rack (~> 1.3.0)
23
+ rack-cache (~> 1.0.1)
24
+ rack-mount (~> 0.8.1)
25
+ rack-test (~> 0.6.0)
26
+ sprockets (~> 2.0.0.beta.10)
27
+ tzinfo (~> 0.3.27)
28
+ activemodel (3.1.0.rc4)
29
+ activesupport (= 3.1.0.rc4)
30
+ bcrypt-ruby (~> 2.1.4)
31
+ builder (~> 3.0.0)
32
+ i18n (~> 0.6)
33
+ activerecord (3.1.0.rc4)
34
+ activemodel (= 3.1.0.rc4)
35
+ activesupport (= 3.1.0.rc4)
36
+ arel (~> 2.1.1)
37
+ tzinfo (~> 0.3.27)
38
+ activeresource (3.1.0.rc4)
39
+ activemodel (= 3.1.0.rc4)
40
+ activesupport (= 3.1.0.rc4)
41
+ activesupport (3.1.0.rc4)
42
+ multi_json (~> 1.0)
43
+ arel (2.1.1)
44
+ bcrypt-ruby (2.1.4)
45
+ builder (3.0.0)
46
+ cocaine (0.1.0)
47
+ erubis (2.7.0)
48
+ git (1.2.5)
49
+ hike (1.0.0)
50
+ i18n (0.6.0)
51
+ jeweler (1.6.2)
52
+ bundler (~> 1.0)
53
+ git (>= 1.2.5)
54
+ rake
55
+ mail (2.3.0)
56
+ i18n (>= 0.4.0)
57
+ mime-types (~> 1.16)
58
+ treetop (~> 1.4.8)
59
+ mime-types (1.16)
60
+ multi_json (1.0.3)
61
+ polyglot (0.3.1)
62
+ rack (1.3.0)
63
+ rack-cache (1.0.2)
64
+ rack (>= 0.4)
65
+ rack-mount (0.8.1)
66
+ rack (>= 1.0.0)
67
+ rack-ssl (1.3.2)
68
+ rack
69
+ rack-test (0.6.0)
70
+ rack (>= 1.0)
71
+ rails (3.1.0.rc4)
72
+ actionmailer (= 3.1.0.rc4)
73
+ actionpack (= 3.1.0.rc4)
74
+ activerecord (= 3.1.0.rc4)
75
+ activeresource (= 3.1.0.rc4)
76
+ activesupport (= 3.1.0.rc4)
77
+ bundler (~> 1.0)
78
+ railties (= 3.1.0.rc4)
79
+ railties (3.1.0.rc4)
80
+ actionpack (= 3.1.0.rc4)
81
+ activesupport (= 3.1.0.rc4)
82
+ rack-ssl (~> 1.3.2)
83
+ rake (>= 0.8.7)
84
+ rdoc (~> 3.4)
85
+ thor (~> 0.14.6)
86
+ rake (0.9.2)
87
+ rdoc (3.6.1)
88
+ sprockets (2.0.0.beta.10)
89
+ hike (~> 1.0)
90
+ rack (~> 1.0)
91
+ tilt (!= 1.3.0, ~> 1.1)
92
+ sqlite3 (1.3.3)
93
+ thor (0.14.6)
94
+ tilt (1.3.2)
95
+ treetop (1.4.9)
96
+ polyglot (>= 0.3.1)
97
+ tzinfo (0.3.27)
98
+
99
+ PLATFORMS
100
+ ruby
101
+
102
+ DEPENDENCIES
103
+ jeweler
104
+ paperclip!
105
+ rails (>= 3.1.0.rc4)
106
+ sqlite3
data/LICENSE ADDED
@@ -0,0 +1,20 @@
1
+ Copyright (c) 2011 The Working Group
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,30 @@
1
+ SofaGallery
2
+ ===========
3
+
4
+ SofaGallery is an image gallery engine for Rails 3.1 apps. As a bonus it integrates seamlessly with ComfortableMexicanSofa (microCMS)
5
+
6
+ Installation
7
+ ------------
8
+ Add gem definition to your Gemfile:
9
+
10
+ gem 'sofa_gallery'
11
+
12
+ Then from the Rails project's root run:
13
+
14
+ bundle install
15
+ rails generate sofa_gallery
16
+ rake db:migrate
17
+
18
+ Usage
19
+ -----
20
+ You can immediately access admin area by going to /admin/galleries.
21
+
22
+ If you are using SofaGallery on it's own take a look in the initializer: [/config/initializers/sofa\_gallery.rb](https://github.com/twg/sofa-gallery/blob/master/config/initializers/sofa_gallery.rb)
23
+ You probably want to set the admin controller to be something that handles user authentication within your app. Same goes for the admin\_route\_prefix.
24
+
25
+ If you are using SofaGallery in conjunction with ComfortableMexicanSofa everything will be configured automatically.
26
+
27
+
28
+ CMS Gallery is released under the [MIT license](https://github.com/twg/sofa-gallery/raw/master/LICENSE)
29
+
30
+ Copyright 2011 The Working Group
data/Rakefile ADDED
@@ -0,0 +1,21 @@
1
+ require File.expand_path('../config/application', __FILE__)
2
+ require 'rubygems'
3
+ require 'rake'
4
+
5
+ SofaGallery::Application.load_tasks
6
+
7
+ begin
8
+ require 'jeweler'
9
+ Jeweler::Tasks.new do |gem|
10
+ gem.name = 'sofa_gallery'
11
+ gem.homepage = 'http://github.com/twg/sofa-gallery'
12
+ gem.license = 'MIT'
13
+ gem.summary = 'SofaGallery is an image gallery engine for Rails 3.1 apps (and ComfortableMexicanSofa)'
14
+ gem.description = ''
15
+ gem.email = 'stephen@theworkinggroup.ca'
16
+ gem.authors = ['Oleg Khabarov', 'Stephen McLeod', 'The Working Group Inc.']
17
+ end
18
+ Jeweler::GemcutterTasks.new
19
+ rescue LoadError
20
+ puts "Jeweler (or a dependency) not available. Install it with: gem install jeweler"
21
+ end
data/VERSION ADDED
@@ -0,0 +1 @@
1
+ 0.0.0
@@ -0,0 +1,15 @@
1
+ //= require sofa_gallery/jquery.js
2
+ //= require sofa_gallery/jquery_ui.js
3
+ //= require sofa_gallery/rails.js
4
+
5
+ $.SofaGallery = function(){
6
+ $(function(){
7
+ $('.sortable').sortable({
8
+ handle: '.dragger',
9
+ axis: 'y',
10
+ update: function(){
11
+ $.post(window.location.pathname + '/reorder', '_method=put&'+$(this).sortable('serialize'));
12
+ }
13
+ });
14
+ });
15
+ }();
@@ -0,0 +1,246 @@
1
+ /**
2
+ * jquery.Jcrop.min.js v0.9.9 (build:20110607)
3
+ * jQuery Image Cropping Plugin
4
+ * @author Kelly Hallman <khallman@gmail.com>
5
+ * Copyright (c) 2008-2011 Kelly Hallman - released under MIT License
6
+ * https://github.com/tapmodo/Jcrop
7
+ */
8
+
9
+ (function($){$.Jcrop=function(obj,opt){var options=$.extend({},$.Jcrop.defaults),docOffset,lastcurs,ie6mode=false;function px(n){return parseInt(n,10)+'px';}
10
+ function pct(n){return parseInt(n,10)+'%';}
11
+ function cssClass(cl){return options.baseClass+'-'+cl;}
12
+ function supportsColorFade(){return $.fx.step.hasOwnProperty('backgroundColor');}
13
+ function getPos(obj)
14
+ {var pos=$(obj).offset();return[pos.left,pos.top];}
15
+ function mouseAbs(e)
16
+ {return[(e.pageX-docOffset[0]),(e.pageY-docOffset[1])];}
17
+ function setOptions(opt)
18
+ {if(typeof(opt)!=='object'){opt={};}
19
+ options=$.extend(options,opt);if(typeof(options.onChange)!=='function'){options.onChange=function(){};}
20
+ if(typeof(options.onSelect)!=='function'){options.onSelect=function(){};}
21
+ if(typeof(options.onRelease)!=='function'){options.onRelease=function(){};}}
22
+ function myCursor(type)
23
+ {if(type!==lastcurs){Tracker.setCursor(type);lastcurs=type;}}
24
+ function startDragMode(mode,pos)
25
+ {docOffset=getPos($img);Tracker.setCursor(mode==='move'?mode:mode+'-resize');if(mode==='move'){return Tracker.activateHandlers(createMover(pos),doneSelect);}
26
+ var fc=Coords.getFixed();var opp=oppLockCorner(mode);var opc=Coords.getCorner(oppLockCorner(opp));Coords.setPressed(Coords.getCorner(opp));Coords.setCurrent(opc);Tracker.activateHandlers(dragmodeHandler(mode,fc),doneSelect);}
27
+ function dragmodeHandler(mode,f)
28
+ {return function(pos){if(!options.aspectRatio){switch(mode){case'e':pos[1]=f.y2;break;case'w':pos[1]=f.y2;break;case'n':pos[0]=f.x2;break;case's':pos[0]=f.x2;break;}}else{switch(mode){case'e':pos[1]=f.y+1;break;case'w':pos[1]=f.y+1;break;case'n':pos[0]=f.x+1;break;case's':pos[0]=f.x+1;break;}}
29
+ Coords.setCurrent(pos);Selection.update();};}
30
+ function createMover(pos)
31
+ {var lloc=pos;KeyManager.watchKeys();return function(pos){Coords.moveOffset([pos[0]-lloc[0],pos[1]-lloc[1]]);lloc=pos;Selection.update();};}
32
+ function oppLockCorner(ord)
33
+ {switch(ord){case'n':return'sw';case's':return'nw';case'e':return'nw';case'w':return'ne';case'ne':return'sw';case'nw':return'se';case'se':return'nw';case'sw':return'ne';}}
34
+ function createDragger(ord)
35
+ {return function(e){if(options.disabled){return false;}
36
+ if((ord==='move')&&!options.allowMove){return false;}
37
+ btndown=true;startDragMode(ord,mouseAbs(e));e.stopPropagation();e.preventDefault();return false;};}
38
+ function presize($obj,w,h)
39
+ {var nw=$obj.width(),nh=$obj.height();if((nw>w)&&w>0){nw=w;nh=(w/$obj.width())*$obj.height();}
40
+ if((nh>h)&&h>0){nh=h;nw=(h/$obj.height())*$obj.width();}
41
+ xscale=$obj.width()/nw;yscale=$obj.height()/nh;$obj.width(nw).height(nh);}
42
+ function unscale(c)
43
+ {return{x:parseInt(c.x*xscale,10),y:parseInt(c.y*yscale,10),x2:parseInt(c.x2*xscale,10),y2:parseInt(c.y2*yscale,10),w:parseInt(c.w*xscale,10),h:parseInt(c.h*yscale,10)};}
44
+ function doneSelect(pos)
45
+ {var c=Coords.getFixed();if((c.w>options.minSelect[0])&&(c.h>options.minSelect[1])){Selection.enableHandles();Selection.done();}else{Selection.release();}
46
+ Tracker.setCursor(options.allowSelect?'crosshair':'default');}
47
+ function newSelection(e)
48
+ {if(options.disabled){return false;}
49
+ if(!options.allowSelect){return false;}
50
+ btndown=true;docOffset=getPos($img);Selection.disableHandles();myCursor('crosshair');var pos=mouseAbs(e);Coords.setPressed(pos);Selection.update();Tracker.activateHandlers(selectDrag,doneSelect);KeyManager.watchKeys();e.stopPropagation();e.preventDefault();return false;}
51
+ function selectDrag(pos)
52
+ {Coords.setCurrent(pos);Selection.update();}
53
+ function newTracker()
54
+ {var trk=$('<div></div>').addClass(cssClass('tracker'));if($.browser.msie){trk.css({opacity:0,backgroundColor:'white'});}
55
+ return trk;}
56
+ if($.browser.msie&&($.browser.version.split('.')[0]==='6')){ie6mode=true;}
57
+ if(typeof(obj)!=='object'){obj=$(obj)[0];}
58
+ if(typeof(opt)!=='object'){opt={};}
59
+ setOptions(opt);var img_css={border:'none',margin:0,padding:0,position:'absolute'};var $origimg=$(obj);var $img=$origimg.clone().removeAttr('id').css(img_css);$img.width($origimg.width());$img.height($origimg.height());$origimg.after($img).hide();presize($img,options.boxWidth,options.boxHeight);var boundx=$img.width(),boundy=$img.height(),$div=$('<div />').width(boundx).height(boundy).addClass(cssClass('holder')).css({position:'relative',backgroundColor:options.bgColor}).insertAfter($origimg).append($img);delete(options.bgColor);if(options.addClass){$div.addClass(options.addClass);}
60
+ var $img2=$('<img />').attr('src',$img.attr('src')).css(img_css).width(boundx).height(boundy),$img_holder=$('<div />').width(pct(100)).height(pct(100)).css({zIndex:310,position:'absolute',overflow:'hidden'}).append($img2),$hdl_holder=$('<div />').width(pct(100)).height(pct(100)).css('zIndex',320),$sel=$('<div />').css({position:'absolute',zIndex:300}).insertBefore($img).append($img_holder,$hdl_holder);if(ie6mode){$sel.css({overflowY:'hidden'});}
61
+ var bound=options.boundary;var $trk=newTracker().width(boundx+(bound*2)).height(boundy+(bound*2)).css({position:'absolute',top:px(-bound),left:px(-bound),zIndex:290}).mousedown(newSelection);var bgopacity=options.bgOpacity,xlimit,ylimit,xmin,ymin,xscale,yscale,enabled=true,btndown,animating,shift_down;docOffset=getPos($img);var Touch=(function(){function hasTouchSupport(){var support={},events=['touchstart','touchmove','touchend'],el=document.createElement('div'),i;try{for(i=0;i<events.length;i++){var eventName=events[i];eventName='on'+eventName;var isSupported=(eventName in el);if(!isSupported){el.setAttribute(eventName,'return;');isSupported=typeof el[eventName]=='function';}
62
+ support[events[i]]=isSupported;}
63
+ return support.touchstart&&support.touchend&&support.touchmove;}
64
+ catch(err){return false;}}
65
+ function detectSupport(){if((options.touchSupport===true)||(options.touchSupport===false))return options.touchSupport;else return hasTouchSupport();}
66
+ return{createDragger:function(ord){return function(e){e.pageX=e.originalEvent.changedTouches[0].pageX;e.pageY=e.originalEvent.changedTouches[0].pageY;if(options.disabled){return false;}
67
+ if((ord==='move')&&!options.allowMove){return false;}
68
+ btndown=true;startDragMode(ord,mouseAbs(e));e.stopPropagation();e.preventDefault();return false;};},newSelection:function(e){e.pageX=e.originalEvent.changedTouches[0].pageX;e.pageY=e.originalEvent.changedTouches[0].pageY;return newSelection(e);},isSupported:hasTouchSupport,support:detectSupport()};}());var Coords=(function(){var x1=0,y1=0,x2=0,y2=0,ox,oy;function setPressed(pos)
69
+ {pos=rebound(pos);x2=x1=pos[0];y2=y1=pos[1];}
70
+ function setCurrent(pos)
71
+ {pos=rebound(pos);ox=pos[0]-x2;oy=pos[1]-y2;x2=pos[0];y2=pos[1];}
72
+ function getOffset()
73
+ {return[ox,oy];}
74
+ function moveOffset(offset)
75
+ {var ox=offset[0],oy=offset[1];if(0>x1+ox){ox-=ox+x1;}
76
+ if(0>y1+oy){oy-=oy+y1;}
77
+ if(boundy<y2+oy){oy+=boundy-(y2+oy);}
78
+ if(boundx<x2+ox){ox+=boundx-(x2+ox);}
79
+ x1+=ox;x2+=ox;y1+=oy;y2+=oy;}
80
+ function getCorner(ord)
81
+ {var c=getFixed();switch(ord){case'ne':return[c.x2,c.y];case'nw':return[c.x,c.y];case'se':return[c.x2,c.y2];case'sw':return[c.x,c.y2];}}
82
+ function getFixed()
83
+ {if(!options.aspectRatio){return getRect();}
84
+ var aspect=options.aspectRatio,min_x=options.minSize[0]/xscale,max_x=options.maxSize[0]/xscale,max_y=options.maxSize[1]/yscale,rw=x2-x1,rh=y2-y1,rwa=Math.abs(rw),rha=Math.abs(rh),real_ratio=rwa/rha,xx,yy;if(max_x===0){max_x=boundx*10;}
85
+ if(max_y===0){max_y=boundy*10;}
86
+ if(real_ratio<aspect){yy=y2;w=rha*aspect;xx=rw<0?x1-w:w+x1;if(xx<0){xx=0;h=Math.abs((xx-x1)/aspect);yy=rh<0?y1-h:h+y1;}else if(xx>boundx){xx=boundx;h=Math.abs((xx-x1)/aspect);yy=rh<0?y1-h:h+y1;}}else{xx=x2;h=rwa/aspect;yy=rh<0?y1-h:y1+h;if(yy<0){yy=0;w=Math.abs((yy-y1)*aspect);xx=rw<0?x1-w:w+x1;}else if(yy>boundy){yy=boundy;w=Math.abs(yy-y1)*aspect;xx=rw<0?x1-w:w+x1;}}
87
+ if(xx>x1){if(xx-x1<min_x){xx=x1+min_x;}else if(xx-x1>max_x){xx=x1+max_x;}
88
+ if(yy>y1){yy=y1+(xx-x1)/aspect;}else{yy=y1-(xx-x1)/aspect;}}else if(xx<x1){if(x1-xx<min_x){xx=x1-min_x;}else if(x1-xx>max_x){xx=x1-max_x;}
89
+ if(yy>y1){yy=y1+(x1-xx)/aspect;}else{yy=y1-(x1-xx)/aspect;}}
90
+ if(xx<0){x1-=xx;xx=0;}else if(xx>boundx){x1-=xx-boundx;xx=boundx;}
91
+ if(yy<0){y1-=yy;yy=0;}else if(yy>boundy){y1-=yy-boundy;yy=boundy;}
92
+ return makeObj(flipCoords(x1,y1,xx,yy));}
93
+ function rebound(p)
94
+ {if(p[0]<0){p[0]=0;}
95
+ if(p[1]<0){p[1]=0;}
96
+ if(p[0]>boundx){p[0]=boundx;}
97
+ if(p[1]>boundy){p[1]=boundy;}
98
+ return[p[0],p[1]];}
99
+ function flipCoords(x1,y1,x2,y2)
100
+ {var xa=x1,xb=x2,ya=y1,yb=y2;if(x2<x1){xa=x2;xb=x1;}
101
+ if(y2<y1){ya=y2;yb=y1;}
102
+ return[Math.round(xa),Math.round(ya),Math.round(xb),Math.round(yb)];}
103
+ function getRect()
104
+ {var xsize=x2-x1,ysize=y2-y1,delta;if(xlimit&&(Math.abs(xsize)>xlimit)){x2=(xsize>0)?(x1+xlimit):(x1-xlimit);}
105
+ if(ylimit&&(Math.abs(ysize)>ylimit)){y2=(ysize>0)?(y1+ylimit):(y1-ylimit);}
106
+ if(ymin/yscale&&(Math.abs(ysize)<ymin/yscale)){y2=(ysize>0)?(y1+ymin/yscale):(y1-ymin/yscale);}
107
+ if(xmin/xscale&&(Math.abs(xsize)<xmin/xscale)){x2=(xsize>0)?(x1+xmin/xscale):(x1-xmin/xscale);}
108
+ if(x1<0){x2-=x1;x1-=x1;}
109
+ if(y1<0){y2-=y1;y1-=y1;}
110
+ if(x2<0){x1-=x2;x2-=x2;}
111
+ if(y2<0){y1-=y2;y2-=y2;}
112
+ if(x2>boundx){delta=x2-boundx;x1-=delta;x2-=delta;}
113
+ if(y2>boundy){delta=y2-boundy;y1-=delta;y2-=delta;}
114
+ if(x1>boundx){delta=x1-boundy;y2-=delta;y1-=delta;}
115
+ if(y1>boundy){delta=y1-boundy;y2-=delta;y1-=delta;}
116
+ return makeObj(flipCoords(x1,y1,x2,y2));}
117
+ function makeObj(a)
118
+ {return{x:a[0],y:a[1],x2:a[2],y2:a[3],w:a[2]-a[0],h:a[3]-a[1]};}
119
+ return{flipCoords:flipCoords,setPressed:setPressed,setCurrent:setCurrent,getOffset:getOffset,moveOffset:moveOffset,getCorner:getCorner,getFixed:getFixed};}());var Selection=(function(){var awake,hdep=370;var borders={};var handle={};var seehandles=false;var hhs=options.handleOffset;function insertBorder(type)
120
+ {var jq=$('<div />').css({position:'absolute',opacity:options.borderOpacity}).addClass(cssClass(type));$img_holder.append(jq);return jq;}
121
+ function dragDiv(ord,zi)
122
+ {var jq=$('<div />').mousedown(createDragger(ord)).css({cursor:ord+'-resize',position:'absolute',zIndex:zi});if(Touch.support){jq.bind('touchstart',Touch.createDragger(ord));}
123
+ $hdl_holder.append(jq);return jq;}
124
+ function insertHandle(ord)
125
+ {return dragDiv(ord,hdep++).css({top:px(-hhs+1),left:px(-hhs+1),opacity:options.handleOpacity}).addClass(cssClass('handle'));}
126
+ function insertDragbar(ord)
127
+ {var s=options.handleSize,h=s,w=s,t=hhs,l=hhs;switch(ord){case'n':case's':w=pct(100);break;case'e':case'w':h=pct(100);break;}
128
+ return dragDiv(ord,hdep++).width(w).height(h).css({top:px(-t+1),left:px(-l+1)});}
129
+ function createHandles(li)
130
+ {var i;for(i=0;i<li.length;i++){handle[li[i]]=insertHandle(li[i]);}}
131
+ function moveHandles(c)
132
+ {var midvert=Math.round((c.h/2)-hhs),midhoriz=Math.round((c.w/2)-hhs),north=-hhs+1,west=-hhs+1,east=c.w-hhs,south=c.h-hhs,x,y;if(handle.e){handle.e.css({top:px(midvert),left:px(east)});handle.w.css({top:px(midvert)});handle.s.css({top:px(south),left:px(midhoriz)});handle.n.css({left:px(midhoriz)});}
133
+ if(handle.ne){handle.ne.css({left:px(east)});handle.se.css({top:px(south),left:px(east)});handle.sw.css({top:px(south)});}
134
+ if(handle.b){handle.b.css({top:px(south)});handle.r.css({left:px(east)});}}
135
+ function moveto(x,y)
136
+ {$img2.css({top:px(-y),left:px(-x)});$sel.css({top:px(y),left:px(x)});}
137
+ function resize(w,h)
138
+ {$sel.width(w).height(h);}
139
+ function refresh()
140
+ {var c=Coords.getFixed();Coords.setPressed([c.x,c.y]);Coords.setCurrent([c.x2,c.y2]);updateVisible();}
141
+ function updateVisible()
142
+ {if(awake){return update();}}
143
+ function update()
144
+ {var c=Coords.getFixed();resize(c.w,c.h);moveto(c.x,c.y);if(seehandles){moveHandles(c);}
145
+ if(!awake){show();}
146
+ options.onChange.call(api,unscale(c));}
147
+ function show()
148
+ {$sel.show();if(options.bgFade){$img.fadeTo(options.fadeTime,bgopacity);}else{$img.css('opacity',bgopacity);}
149
+ awake=true;}
150
+ function release()
151
+ {disableHandles();$sel.hide();if(options.bgFade){$img.fadeTo(options.fadeTime,1);}else{$img.css('opacity',1);}
152
+ awake=false;options.onRelease.call(api);}
153
+ function showHandles()
154
+ {if(seehandles){moveHandles(Coords.getFixed());$hdl_holder.show();}}
155
+ function enableHandles()
156
+ {seehandles=true;if(options.allowResize){moveHandles(Coords.getFixed());$hdl_holder.show();return true;}}
157
+ function disableHandles()
158
+ {seehandles=false;$hdl_holder.hide();}
159
+ function animMode(v)
160
+ {if(animating===v){disableHandles();}else{enableHandles();}}
161
+ function done()
162
+ {animMode(false);refresh();}
163
+ if(options.drawBorders){borders={top:insertBorder('hline'),bottom:insertBorder('hline bottom'),left:insertBorder('vline'),right:insertBorder('vline right')};}
164
+ if(options.dragEdges){handle.t=insertDragbar('n');handle.b=insertDragbar('s');handle.r=insertDragbar('e');handle.l=insertDragbar('w');}
165
+ if(options.sideHandles){createHandles(['n','s','e','w']);}
166
+ if(options.cornerHandles){createHandles(['sw','nw','ne','se']);}
167
+ var $track=newTracker().mousedown(createDragger('move')).css({cursor:'move',position:'absolute',zIndex:360});if(Touch.support){$track.bind('touchstart.jcrop',Touch.createDragger('move'));}
168
+ $img_holder.append($track);disableHandles();return{updateVisible:updateVisible,update:update,release:release,refresh:refresh,isAwake:function(){return awake;},setCursor:function(cursor){$track.css('cursor',cursor);},enableHandles:enableHandles,enableOnly:function(){seehandles=true;},showHandles:showHandles,disableHandles:disableHandles,animMode:animMode,done:done};}());var Tracker=(function(){var onMove=function(){},onDone=function(){},trackDoc=options.trackDocument;function toFront()
169
+ {$trk.css({zIndex:450});if(trackDoc){$(document).bind('mousemove',trackMove).bind('mouseup',trackUp);}}
170
+ function toBack()
171
+ {$trk.css({zIndex:290});if(trackDoc){$(document).unbind('mousemove',trackMove).unbind('mouseup',trackUp);}}
172
+ function trackMove(e)
173
+ {onMove(mouseAbs(e));return false;}
174
+ function trackUp(e)
175
+ {e.preventDefault();e.stopPropagation();if(btndown){btndown=false;onDone(mouseAbs(e));if(Selection.isAwake()){options.onSelect.call(api,unscale(Coords.getFixed()));}
176
+ toBack();onMove=function(){};onDone=function(){};}
177
+ return false;}
178
+ function activateHandlers(move,done)
179
+ {btndown=true;onMove=move;onDone=done;toFront();return false;}
180
+ function trackTouchMove(e)
181
+ {e.pageX=e.originalEvent.changedTouches[0].pageX;e.pageY=e.originalEvent.changedTouches[0].pageY;return trackMove(e);}
182
+ function trackTouchEnd(e)
183
+ {e.pageX=e.originalEvent.changedTouches[0].pageX;e.pageY=e.originalEvent.changedTouches[0].pageY;return trackUp(e);}
184
+ function setCursor(t)
185
+ {$trk.css('cursor',t);}
186
+ if(Touch.support){$(document).bind('touchmove',trackTouchMove).bind('touchend',trackTouchEnd);}
187
+ if(!trackDoc){$trk.mousemove(trackMove).mouseup(trackUp).mouseout(trackUp);}
188
+ $img.before($trk);return{activateHandlers:activateHandlers,setCursor:setCursor};}());var KeyManager=(function(){var $keymgr=$('<input type="radio" />').css({position:'fixed',left:'-120px',width:'12px'}),$keywrap=$('<div />').css({position:'absolute',overflow:'hidden'}).append($keymgr);function watchKeys()
189
+ {if(options.keySupport){$keymgr.show();$keymgr.focus();}}
190
+ function onBlur(e)
191
+ {$keymgr.hide();}
192
+ function doNudge(e,x,y)
193
+ {if(options.allowMove){Coords.moveOffset([x,y]);Selection.updateVisible();}
194
+ e.preventDefault();e.stopPropagation();}
195
+ function parseKey(e)
196
+ {if(e.ctrlKey){return true;}
197
+ shift_down=e.shiftKey?true:false;var nudge=shift_down?10:1;switch(e.keyCode){case 37:doNudge(e,-nudge,0);break;case 39:doNudge(e,nudge,0);break;case 38:doNudge(e,0,-nudge);break;case 40:doNudge(e,0,nudge);break;case 27:Selection.release();break;case 9:return true;}
198
+ return false;}
199
+ if(options.keySupport){$keymgr.keydown(parseKey).blur(onBlur);if(ie6mode||!options.fixedSupport){$keymgr.css({position:'absolute',left:'-20px'});$keywrap.append($keymgr).insertBefore($img);}else{$keymgr.insertBefore($img);}}
200
+ return{watchKeys:watchKeys};}());function setClass(cname)
201
+ {$div.removeClass().addClass(cssClass('holder')).addClass(cname);}
202
+ function animateTo(a,callback)
203
+ {var x1=parseInt(a[0],10)/xscale,y1=parseInt(a[1],10)/yscale,x2=parseInt(a[2],10)/xscale,y2=parseInt(a[3],10)/yscale;if(animating){return;}
204
+ var animto=Coords.flipCoords(x1,y1,x2,y2),c=Coords.getFixed(),initcr=[c.x,c.y,c.x2,c.y2],animat=initcr,interv=options.animationDelay,ix1=animto[0]-initcr[0],iy1=animto[1]-initcr[1],ix2=animto[2]-initcr[2],iy2=animto[3]-initcr[3],pcent=0,velocity=options.swingSpeed;x=animat[0];y=animat[1];x2=animat[2];y2=animat[3];Selection.animMode(true);var anim_timer;function queueAnimator(){window.setTimeout(animator,interv);}
205
+ var animator=(function(){return function(){pcent+=(100-pcent)/velocity;animat[0]=x+((pcent/100)*ix1);animat[1]=y+((pcent/100)*iy1);animat[2]=x2+((pcent/100)*ix2);animat[3]=y2+((pcent/100)*iy2);if(pcent>=99.8){pcent=100;}
206
+ if(pcent<100){setSelectRaw(animat);queueAnimator();}else{Selection.done();if(typeof(callback)==='function'){callback.call(api);}}};}());queueAnimator();}
207
+ function setSelect(rect)
208
+ {setSelectRaw([parseInt(rect[0],10)/xscale,parseInt(rect[1],10)/yscale,parseInt(rect[2],10)/xscale,parseInt(rect[3],10)/yscale]);}
209
+ function setSelectRaw(l)
210
+ {Coords.setPressed([l[0],l[1]]);Coords.setCurrent([l[2],l[3]]);Selection.update();}
211
+ function tellSelect()
212
+ {return unscale(Coords.getFixed());}
213
+ function tellScaled()
214
+ {return Coords.getFixed();}
215
+ function setOptionsNew(opt)
216
+ {setOptions(opt);interfaceUpdate();}
217
+ function disableCrop()
218
+ {options.disabled=true;Selection.disableHandles();Selection.setCursor('default');Tracker.setCursor('default');}
219
+ function enableCrop()
220
+ {options.disabled=false;interfaceUpdate();}
221
+ function cancelCrop()
222
+ {Selection.done();Tracker.activateHandlers(null,null);}
223
+ function destroy()
224
+ {$div.remove();$origimg.show();$(obj).removeData('Jcrop');}
225
+ function setImage(src,callback)
226
+ {Selection.release();disableCrop();var img=new Image();img.onload=function(){var iw=img.width;var ih=img.height;var bw=options.boxWidth;var bh=options.boxHeight;$img.width(iw).height(ih);$img.attr('src',src);$img2.attr('src',src);presize($img,bw,bh);boundx=$img.width();boundy=$img.height();$img2.width(boundx).height(boundy);$trk.width(boundx+(bound*2)).height(boundy+(bound*2));$div.width(boundx).height(boundy);enableCrop();if(typeof(callback)==='function'){callback.call(api);}};img.src=src;}
227
+ function interfaceUpdate(alt)
228
+ {if(options.allowResize){if(alt){Selection.enableOnly();}else{Selection.enableHandles();}}else{Selection.disableHandles();}
229
+ Tracker.setCursor(options.allowSelect?'crosshair':'default');Selection.setCursor(options.allowMove?'move':'default');if(options.hasOwnProperty('setSelect')){setSelect(options.setSelect);Selection.done();delete(options.setSelect);}
230
+ if(options.hasOwnProperty('trueSize')){xscale=options.trueSize[0]/boundx;yscale=options.trueSize[1]/boundy;}
231
+ if(options.hasOwnProperty('bgColor')){if(supportsColorFade()&&options.fadeTime){$div.animate({backgroundColor:options.bgColor},{queue:false,duration:options.fadeTime});}else{$div.css('backgroundColor',options.bgColor);}
232
+ delete(options.bgColor);}
233
+ if(options.hasOwnProperty('bgOpacity')){bgopacity=options.bgOpacity;if(Selection.isAwake()){if(options.fadeTime){$img.fadeTo(options.fadeTime,bgopacity);}else{$div.css('opacity',options.opacity);}}
234
+ delete(options.bgOpacity);}
235
+ xlimit=options.maxSize[0]||0;ylimit=options.maxSize[1]||0;xmin=options.minSize[0]||0;ymin=options.minSize[1]||0;if(options.hasOwnProperty('outerImage')){$img.attr('src',options.outerImage);delete(options.outerImage);}
236
+ Selection.refresh();}
237
+ if(Touch.support){$trk.bind('touchstart',Touch.newSelection);}
238
+ $hdl_holder.hide();interfaceUpdate(true);var api={setImage:setImage,animateTo:animateTo,setSelect:setSelect,setOptions:setOptionsNew,tellSelect:tellSelect,tellScaled:tellScaled,setClass:setClass,disable:disableCrop,enable:enableCrop,cancel:cancelCrop,release:Selection.release,destroy:destroy,focus:KeyManager.watchKeys,getBounds:function(){return[boundx*xscale,boundy*yscale];},getWidgetSize:function(){return[boundx,boundy];},getScaleFactor:function(){return[xscale,yscale];},ui:{holder:$div,selection:$sel}};if($.browser.msie){$div.bind('selectstart',function(){return false;});}
239
+ $origimg.data('Jcrop',api);return api;};$.fn.Jcrop=function(options,callback)
240
+ {function attachWhenDone(from)
241
+ {var opt=(typeof(options)==='object')?options:{};var loadsrc=opt.useImg||from.src;var img=new Image();img.onload=function(){function attachJcrop(){var api=$.Jcrop(from,opt);if(typeof(callback)==='function'){callback.call(api);}}
242
+ function attachAttempt(){if(!img.width||!img.height){window.setTimeout(attachAttempt,50);}else{attachJcrop();}}
243
+ window.setTimeout(attachAttempt,50);};img.src=loadsrc;}
244
+ this.each(function(){if($(this).data('Jcrop')){if(options==='api'){return $(this).data('Jcrop');}
245
+ else{$(this).data('Jcrop').setOptions(options);}}
246
+ else{attachWhenDone(this);}});return this;};$.Jcrop.defaults={allowSelect:true,allowMove:true,allowResize:true,trackDocument:true,baseClass:'jcrop',addClass:null,bgColor:'black',bgOpacity:0.6,bgFade:false,borderOpacity:0.4,handleOpacity:0.5,handleSize:9,handleOffset:5,aspectRatio:0,keySupport:true,cornerHandles:true,sideHandles:true,drawBorders:true,dragEdges:true,fixedSupport:true,touchSupport:null,boxWidth:0,boxHeight:0,boundary:2,fadeTime:400,animationDelay:20,swingSpeed:3,minSelect:[0,0],maxSize:[0,0],minSize:[0,0],onChange:function(){},onSelect:function(){},onRelease:function(){}};}(jQuery));