portal html css js resource

jquery.Jcrop.js 44KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724
  1. /**
  2. * jquery.Jcrop.js v0.9.12
  3. * jQuery Image Cropping Plugin - released under MIT License
  4. * Author: Kelly Hallman <khallman@gmail.com>
  5. * http://github.com/tapmodo/Jcrop
  6. * Copyright (c) 2008-2013 Tapmodo Interactive LLC {{{
  7. *
  8. * Permission is hereby granted, free of charge, to any person
  9. * obtaining a copy of this software and associated documentation
  10. * files (the "Software"), to deal in the Software without
  11. * restriction, including without limitation the rights to use,
  12. * copy, modify, merge, publish, distribute, sublicense, and/or sell
  13. * copies of the Software, and to permit persons to whom the
  14. * Software is furnished to do so, subject to the following
  15. * conditions:
  16. *
  17. * The above copyright notice and this permission notice shall be
  18. * included in all copies or substantial portions of the Software.
  19. *
  20. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  21. * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
  22. * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  23. * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
  24. * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  25. * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  26. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
  27. * OTHER DEALINGS IN THE SOFTWARE.
  28. *
  29. * }}}
  30. */
  31. (function ($) {
  32. $.Jcrop = function (obj, opt) {
  33. var options = $.extend({}, $.Jcrop.defaults),
  34. docOffset,
  35. _ua = navigator.userAgent.toLowerCase(),
  36. is_msie = /msie/.test(_ua),
  37. ie6mode = /msie [1-6]\./.test(_ua);
  38. // Internal Methods {{{
  39. function px(n) {
  40. return Math.round(n) + 'px';
  41. }
  42. function cssClass(cl) {
  43. return options.baseClass + '-' + cl;
  44. }
  45. function supportsColorFade() {
  46. return $.fx.step.hasOwnProperty('backgroundColor');
  47. }
  48. function getPos(obj) //{{{
  49. {
  50. var pos = $(obj).offset();
  51. return [pos.left, pos.top];
  52. }
  53. //}}}
  54. function mouseAbs(e) //{{{
  55. {
  56. return [(e.pageX - docOffset[0]), (e.pageY - docOffset[1])];
  57. }
  58. //}}}
  59. function setOptions(opt) //{{{
  60. {
  61. if (typeof(opt) !== 'object') opt = {};
  62. options = $.extend(options, opt);
  63. $.each(['onChange','onSelect','onRelease','onDblClick'],function(i,e) {
  64. if (typeof(options[e]) !== 'function') options[e] = function () {};
  65. });
  66. }
  67. //}}}
  68. function startDragMode(mode, pos, touch) //{{{
  69. {
  70. docOffset = getPos($img);
  71. Tracker.setCursor(mode === 'move' ? mode : mode + '-resize');
  72. if (mode === 'move') {
  73. return Tracker.activateHandlers(createMover(pos), doneSelect, touch);
  74. }
  75. var fc = Coords.getFixed();
  76. var opp = oppLockCorner(mode);
  77. var opc = Coords.getCorner(oppLockCorner(opp));
  78. Coords.setPressed(Coords.getCorner(opp));
  79. Coords.setCurrent(opc);
  80. Tracker.activateHandlers(dragmodeHandler(mode, fc), doneSelect, touch);
  81. }
  82. //}}}
  83. function dragmodeHandler(mode, f) //{{{
  84. {
  85. return function (pos) {
  86. if (!options.aspectRatio) {
  87. switch (mode) {
  88. case 'e':
  89. pos[1] = f.y2;
  90. break;
  91. case 'w':
  92. pos[1] = f.y2;
  93. break;
  94. case 'n':
  95. pos[0] = f.x2;
  96. break;
  97. case 's':
  98. pos[0] = f.x2;
  99. break;
  100. }
  101. } else {
  102. switch (mode) {
  103. case 'e':
  104. pos[1] = f.y + 1;
  105. break;
  106. case 'w':
  107. pos[1] = f.y + 1;
  108. break;
  109. case 'n':
  110. pos[0] = f.x + 1;
  111. break;
  112. case 's':
  113. pos[0] = f.x + 1;
  114. break;
  115. }
  116. }
  117. Coords.setCurrent(pos);
  118. Selection.update();
  119. };
  120. }
  121. //}}}
  122. function createMover(pos) //{{{
  123. {
  124. var lloc = pos;
  125. KeyManager.watchKeys();
  126. return function (pos) {
  127. Coords.moveOffset([pos[0] - lloc[0], pos[1] - lloc[1]]);
  128. lloc = pos;
  129. Selection.update();
  130. };
  131. }
  132. //}}}
  133. function oppLockCorner(ord) //{{{
  134. {
  135. switch (ord) {
  136. case 'n':
  137. return 'sw';
  138. case 's':
  139. return 'nw';
  140. case 'e':
  141. return 'nw';
  142. case 'w':
  143. return 'ne';
  144. case 'ne':
  145. return 'sw';
  146. case 'nw':
  147. return 'se';
  148. case 'se':
  149. return 'nw';
  150. case 'sw':
  151. return 'ne';
  152. }
  153. }
  154. //}}}
  155. function createDragger(ord) //{{{
  156. {
  157. return function (e) {
  158. if (options.disabled) {
  159. return false;
  160. }
  161. if ((ord === 'move') && !options.allowMove) {
  162. return false;
  163. }
  164. // Fix position of crop area when dragged the very first time.
  165. // Necessary when crop image is in a hidden element when page is loaded.
  166. docOffset = getPos($img);
  167. btndown = true;
  168. startDragMode(ord, mouseAbs(e));
  169. e.stopPropagation();
  170. e.preventDefault();
  171. return false;
  172. };
  173. }
  174. //}}}
  175. function presize($obj, w, h) //{{{
  176. {
  177. var nw = $obj.width(),
  178. nh = $obj.height();
  179. if ((nw > w) && w > 0) {
  180. nw = w;
  181. nh = (w / $obj.width()) * $obj.height();
  182. }
  183. if ((nh > h) && h > 0) {
  184. nh = h;
  185. nw = (h / $obj.height()) * $obj.width();
  186. }
  187. xscale = $obj.width() / nw;
  188. yscale = $obj.height() / nh;
  189. $obj.width(nw).height(nh);
  190. }
  191. //}}}
  192. function unscale(c) //{{{
  193. {
  194. return {
  195. x: parseInt(c.x * xscale),
  196. y: parseInt(c.y * yscale),
  197. x2: parseInt(c.x2 * xscale),
  198. y2: parseInt(c.y2 * yscale),
  199. w: parseInt(c.w * xscale),
  200. h: parseInt(c.h * yscale)
  201. };
  202. }
  203. //}}}
  204. function doneSelect(pos) //{{{
  205. {
  206. var c = Coords.getFixed();
  207. if ((c.w > options.minSelect[0]) && (c.h > options.minSelect[1])) {
  208. Selection.enableHandles();
  209. Selection.done();
  210. } else {
  211. Selection.release();
  212. }
  213. Tracker.setCursor(options.allowSelect ? 'crosshair' : 'default');
  214. }
  215. //}}}
  216. function newSelection(e) //{{{
  217. {
  218. if (options.disabled) {
  219. return false;
  220. }
  221. if (!options.allowSelect) {
  222. return false;
  223. }
  224. btndown = true;
  225. docOffset = getPos($img);
  226. Selection.disableHandles();
  227. Tracker.setCursor('crosshair');
  228. var pos = mouseAbs(e);
  229. Coords.setPressed(pos);
  230. Selection.update();
  231. Tracker.activateHandlers(selectDrag, doneSelect, e.type.substring(0,5)==='touch');
  232. KeyManager.watchKeys();
  233. e.stopPropagation();
  234. e.preventDefault();
  235. return false;
  236. }
  237. //}}}
  238. function selectDrag(pos) //{{{
  239. {
  240. Coords.setCurrent(pos);
  241. Selection.update();
  242. }
  243. //}}}
  244. function newTracker() //{{{
  245. {
  246. var trk = $('<div></div>').addClass(cssClass('tracker'));
  247. if (is_msie) {
  248. trk.css({
  249. opacity: 0,
  250. backgroundColor: 'white'
  251. });
  252. }
  253. return trk;
  254. }
  255. //}}}
  256. // }}}
  257. // Initialization {{{
  258. // Sanitize some options {{{
  259. if (typeof(obj) !== 'object') {
  260. obj = $(obj)[0];
  261. }
  262. if (typeof(opt) !== 'object') {
  263. opt = {};
  264. }
  265. // }}}
  266. setOptions(opt);
  267. // Initialize some jQuery objects {{{
  268. // The values are SET on the image(s) for the interface
  269. // If the original image has any of these set, they will be reset
  270. // However, if you destroy() the Jcrop instance the original image's
  271. // character in the DOM will be as you left it.
  272. var img_css = {
  273. border: 'none',
  274. visibility: 'visible',
  275. margin: 0,
  276. padding: 0,
  277. position: 'absolute',
  278. top: 0,
  279. left: 0
  280. };
  281. var $origimg = $(obj),
  282. img_mode = true;
  283. if (obj.tagName == 'IMG') {
  284. // Fix size of crop image.
  285. // Necessary when crop image is within a hidden element when page is loaded.
  286. if ($origimg[0].width != 0 && $origimg[0].height != 0) {
  287. // Obtain dimensions from contained img element.
  288. $origimg.width($origimg[0].width);
  289. $origimg.height($origimg[0].height);
  290. } else {
  291. // Obtain dimensions from temporary image in case the original is not loaded yet (e.g. IE 7.0).
  292. var tempImage = new Image();
  293. tempImage.src = $origimg[0].src;
  294. $origimg.width(tempImage.width);
  295. $origimg.height(tempImage.height);
  296. }
  297. var $img = $origimg.clone().removeAttr('id').css(img_css).show();
  298. $img.width($origimg.width());
  299. $img.height($origimg.height());
  300. $origimg.after($img).hide();
  301. } else {
  302. $img = $origimg.css(img_css).show();
  303. img_mode = false;
  304. if (options.shade === null) { options.shade = true; }
  305. }
  306. presize($img, options.boxWidth, options.boxHeight);
  307. var boundx = $img.width(),
  308. boundy = $img.height(),
  309. $div = $('<div />').width(boundx).height(boundy).addClass(cssClass('holder')).css({
  310. position: 'relative',
  311. backgroundColor: options.bgColor
  312. }).insertAfter($origimg).append($img);
  313. if (options.addClass) {
  314. $div.addClass(options.addClass);
  315. }
  316. var $img2 = $('<div />'),
  317. $img_holder = $('<div />')
  318. .width('100%').height('100%').css({
  319. zIndex: 310,
  320. position: 'absolute',
  321. overflow: 'hidden'
  322. }),
  323. $hdl_holder = $('<div />')
  324. .width('100%').height('100%').css({
  325. 'zIndex':320
  326. }),
  327. $sel = $('<div id="small" />')
  328. .css({
  329. position: 'absolute',
  330. zIndex: 600
  331. }).dblclick(function(){
  332. var c = Coords.getFixed();
  333. options.onDblClick.call(api,c);
  334. }).insertBefore($img).append($img_holder, $hdl_holder);
  335. if (img_mode) {
  336. $img2 = $('<img />')
  337. .attr('src', $img.attr('src')).css(img_css).width(boundx).height(boundy),
  338. $img_holder.append($img2);
  339. }
  340. /*
  341. if (ie6mode) {
  342. $sel.css({
  343. overflowY: 'hidden'
  344. });
  345. }
  346. */
  347. var bound = options.boundary;
  348. var $trk = newTracker().width(boundx + (bound * 2)).height(boundy + (bound * 2)).css({
  349. position: 'absolute',
  350. top: px(-bound),
  351. left: px(-bound),
  352. zIndex: 290
  353. }).mousedown(newSelection);
  354. /* }}} */
  355. // Set more variables {{{
  356. var bgcolor = options.bgColor,
  357. bgopacity = options.bgOpacity,
  358. xlimit, ylimit, xmin, ymin, xscale, yscale, enabled = true,
  359. btndown, animating, shift_down;
  360. docOffset = getPos($img);
  361. // }}}
  362. // }}}
  363. // Internal Modules {{{
  364. // Touch Module {{{
  365. var Touch = (function () {
  366. // Touch support detection function adapted (under MIT License)
  367. // from code by Jeffrey Sambells - http://github.com/iamamused/
  368. function hasTouchSupport() {
  369. var support = {}, events = ['touchstart', 'touchmove', 'touchend'],
  370. el = document.createElement('div'), i;
  371. try {
  372. for(i=0; i<events.length; i++) {
  373. var eventName = events[i];
  374. eventName = 'on' + eventName;
  375. var isSupported = (eventName in el);
  376. if (!isSupported) {
  377. el.setAttribute(eventName, 'return;');
  378. isSupported = typeof el[eventName] == 'function';
  379. }
  380. support[events[i]] = isSupported;
  381. }
  382. return support.touchstart && support.touchend && support.touchmove;
  383. }
  384. catch(err) {
  385. return false;
  386. }
  387. }
  388. function detectSupport() {
  389. if ((options.touchSupport === true) || (options.touchSupport === false)) return options.touchSupport;
  390. else return hasTouchSupport();
  391. }
  392. return {
  393. createDragger: function (ord) {
  394. return function (e) {
  395. if (options.disabled) {
  396. return false;
  397. }
  398. if ((ord === 'move') && !options.allowMove) {
  399. return false;
  400. }
  401. docOffset = getPos($img);
  402. btndown = true;
  403. startDragMode(ord, mouseAbs(Touch.cfilter(e)), true);
  404. e.stopPropagation();
  405. e.preventDefault();
  406. return false;
  407. };
  408. },
  409. newSelection: function (e) {
  410. return newSelection(Touch.cfilter(e));
  411. },
  412. cfilter: function (e){
  413. e.pageX = e.originalEvent.changedTouches[0].pageX;
  414. e.pageY = e.originalEvent.changedTouches[0].pageY;
  415. return e;
  416. },
  417. isSupported: hasTouchSupport,
  418. support: detectSupport()
  419. };
  420. }());
  421. // }}}
  422. // Coords Module {{{
  423. var Coords = (function () {
  424. var x1 = 0,
  425. y1 = 0,
  426. x2 = 0,
  427. y2 = 0,
  428. ox, oy;
  429. function setPressed(pos) //{{{
  430. {
  431. pos = rebound(pos);
  432. x2 = x1 = pos[0];
  433. y2 = y1 = pos[1];
  434. }
  435. //}}}
  436. function setCurrent(pos) //{{{
  437. {
  438. pos = rebound(pos);
  439. ox = pos[0] - x2;
  440. oy = pos[1] - y2;
  441. x2 = pos[0];
  442. y2 = pos[1];
  443. }
  444. //}}}
  445. function getOffset() //{{{
  446. {
  447. return [ox, oy];
  448. }
  449. //}}}
  450. function moveOffset(offset) //{{{
  451. {
  452. var ox = offset[0],
  453. oy = offset[1];
  454. if (0 > x1 + ox) {
  455. ox -= ox + x1;
  456. }
  457. if (0 > y1 + oy) {
  458. oy -= oy + y1;
  459. }
  460. if (boundy < y2 + oy) {
  461. oy += boundy - (y2 + oy);
  462. }
  463. if (boundx < x2 + ox) {
  464. ox += boundx - (x2 + ox);
  465. }
  466. x1 += ox;
  467. x2 += ox;
  468. y1 += oy;
  469. y2 += oy;
  470. }
  471. //}}}
  472. function getCorner(ord) //{{{
  473. {
  474. var c = getFixed();
  475. switch (ord) {
  476. case 'ne':
  477. return [c.x2, c.y];
  478. case 'nw':
  479. return [c.x, c.y];
  480. case 'se':
  481. return [c.x2, c.y2];
  482. case 'sw':
  483. return [c.x, c.y2];
  484. }
  485. }
  486. //}}}
  487. function getFixed() //{{{
  488. {
  489. if (!options.aspectRatio) {
  490. return getRect();
  491. }
  492. // This function could use some optimization I think...
  493. var aspect = options.aspectRatio,
  494. min_x = options.minSize[0] / xscale,
  495. min_y = options.minSize[1]/yscale,
  496. max_x = options.maxSize[0] / xscale,
  497. max_y = options.maxSize[1] / yscale,
  498. rw = x2 - x1,
  499. rh = y2 - y1,
  500. rwa = Math.abs(rw),
  501. rha = Math.abs(rh),
  502. real_ratio = rwa / rha,
  503. xx, yy, w, h;
  504. if (max_x === 0) {
  505. max_x = boundx * 10;
  506. }
  507. if (max_y === 0) {
  508. max_y = boundy * 10;
  509. }
  510. if (real_ratio < aspect) {
  511. yy = y2;
  512. w = rha * aspect;
  513. xx = rw < 0 ? x1 - w : w + x1;
  514. if (xx < 0) {
  515. xx = 0;
  516. h = Math.abs((xx - x1) / aspect);
  517. yy = rh < 0 ? y1 - h : h + y1;
  518. } else if (xx > boundx) {
  519. xx = boundx;
  520. h = Math.abs((xx - x1) / aspect);
  521. yy = rh < 0 ? y1 - h : h + y1;
  522. }
  523. } else {
  524. xx = x2;
  525. h = rwa / aspect;
  526. yy = rh < 0 ? y1 - h : y1 + h;
  527. if (yy < 0) {
  528. yy = 0;
  529. w = Math.abs((yy - y1) * aspect);
  530. xx = rw < 0 ? x1 - w : w + x1;
  531. } else if (yy > boundy) {
  532. yy = boundy;
  533. w = Math.abs(yy - y1) * aspect;
  534. xx = rw < 0 ? x1 - w : w + x1;
  535. }
  536. }
  537. // Magic %-)
  538. if (xx > x1) { // right side
  539. if (xx - x1 < min_x) {
  540. xx = x1 + min_x;
  541. } else if (xx - x1 > max_x) {
  542. xx = x1 + max_x;
  543. }
  544. if (yy > y1) {
  545. yy = y1 + (xx - x1) / aspect;
  546. } else {
  547. yy = y1 - (xx - x1) / aspect;
  548. }
  549. } else if (xx < x1) { // left side
  550. if (x1 - xx < min_x) {
  551. xx = x1 - min_x;
  552. } else if (x1 - xx > max_x) {
  553. xx = x1 - max_x;
  554. }
  555. if (yy > y1) {
  556. yy = y1 + (x1 - xx) / aspect;
  557. } else {
  558. yy = y1 - (x1 - xx) / aspect;
  559. }
  560. }
  561. if (xx < 0) {
  562. x1 -= xx;
  563. xx = 0;
  564. } else if (xx > boundx) {
  565. x1 -= xx - boundx;
  566. xx = boundx;
  567. }
  568. if (yy < 0) {
  569. y1 -= yy;
  570. yy = 0;
  571. } else if (yy > boundy) {
  572. y1 -= yy - boundy;
  573. yy = boundy;
  574. }
  575. return makeObj(flipCoords(x1, y1, xx, yy));
  576. }
  577. //}}}
  578. function rebound(p) //{{{
  579. {
  580. if (p[0] < 0) p[0] = 0;
  581. if (p[1] < 0) p[1] = 0;
  582. if (p[0] > boundx) p[0] = boundx;
  583. if (p[1] > boundy) p[1] = boundy;
  584. return [Math.round(p[0]), Math.round(p[1])];
  585. }
  586. //}}}
  587. function flipCoords(x1, y1, x2, y2) //{{{
  588. {
  589. var xa = x1,
  590. xb = x2,
  591. ya = y1,
  592. yb = y2;
  593. if (x2 < x1) {
  594. xa = x2;
  595. xb = x1;
  596. }
  597. if (y2 < y1) {
  598. ya = y2;
  599. yb = y1;
  600. }
  601. return [xa, ya, xb, yb];
  602. }
  603. //}}}
  604. function getRect() //{{{
  605. {
  606. var xsize = x2 - x1,
  607. ysize = y2 - y1,
  608. delta;
  609. if (xlimit && (Math.abs(xsize) > xlimit)) {
  610. x2 = (xsize > 0) ? (x1 + xlimit) : (x1 - xlimit);
  611. }
  612. if (ylimit && (Math.abs(ysize) > ylimit)) {
  613. y2 = (ysize > 0) ? (y1 + ylimit) : (y1 - ylimit);
  614. }
  615. if (ymin / yscale && (Math.abs(ysize) < ymin / yscale)) {
  616. y2 = (ysize > 0) ? (y1 + ymin / yscale) : (y1 - ymin / yscale);
  617. }
  618. if (xmin / xscale && (Math.abs(xsize) < xmin / xscale)) {
  619. x2 = (xsize > 0) ? (x1 + xmin / xscale) : (x1 - xmin / xscale);
  620. }
  621. if (x1 < 0) {
  622. x2 -= x1;
  623. x1 -= x1;
  624. }
  625. if (y1 < 0) {
  626. y2 -= y1;
  627. y1 -= y1;
  628. }
  629. if (x2 < 0) {
  630. x1 -= x2;
  631. x2 -= x2;
  632. }
  633. if (y2 < 0) {
  634. y1 -= y2;
  635. y2 -= y2;
  636. }
  637. if (x2 > boundx) {
  638. delta = x2 - boundx;
  639. x1 -= delta;
  640. x2 -= delta;
  641. }
  642. if (y2 > boundy) {
  643. delta = y2 - boundy;
  644. y1 -= delta;
  645. y2 -= delta;
  646. }
  647. if (x1 > boundx) {
  648. delta = x1 - boundy;
  649. y2 -= delta;
  650. y1 -= delta;
  651. }
  652. if (y1 > boundy) {
  653. delta = y1 - boundy;
  654. y2 -= delta;
  655. y1 -= delta;
  656. }
  657. return makeObj(flipCoords(x1, y1, x2, y2));
  658. }
  659. //}}}
  660. function makeObj(a) //{{{
  661. {
  662. return {
  663. x: a[0],
  664. y: a[1],
  665. x2: a[2],
  666. y2: a[3],
  667. w: a[2] - a[0],
  668. h: a[3] - a[1]
  669. };
  670. }
  671. //}}}
  672. return {
  673. flipCoords: flipCoords,
  674. setPressed: setPressed,
  675. setCurrent: setCurrent,
  676. getOffset: getOffset,
  677. moveOffset: moveOffset,
  678. getCorner: getCorner,
  679. getFixed: getFixed
  680. };
  681. }());
  682. //}}}
  683. // Shade Module {{{
  684. var Shade = (function() {
  685. var enabled = false,
  686. holder = $('<div id="opa" />').css({
  687. position: 'absolute',
  688. zIndex: 240,
  689. opacity: 0,
  690. top: 0,
  691. left: 0,
  692. width: "100%",
  693. height: "100%"
  694. }),
  695. shades = {
  696. top: createShade(),
  697. left: createShade().height(boundy),
  698. right: createShade().height(boundy),
  699. bottom: createShade()
  700. };
  701. function resizeShades(w,h) {
  702. shades.left.css({ height: px(h) });
  703. shades.right.css({ height: px(h) });
  704. }
  705. function updateAuto()
  706. {
  707. return updateShade(Coords.getFixed());
  708. }
  709. function updateShade(c)
  710. {
  711. shades.top.css({
  712. left: px(c.x),
  713. width: px(c.w),
  714. height: px(c.y),
  715. top: 0
  716. });
  717. shades.bottom.css({
  718. top: px(c.y2),
  719. left: px(c.x),
  720. width: px(c.w),
  721. height: px(boundy-c.y2)
  722. });
  723. shades.right.css({
  724. left: px(c.x2),
  725. width: px(boundx-c.x2),
  726. top: 0
  727. });
  728. shades.left.css({
  729. width: px(c.x),
  730. top: 0,
  731. left: 0
  732. });
  733. }
  734. function createShade() {
  735. return $('<div />').css({
  736. position: 'absolute',
  737. backgroundColor: options.shadeColor||options.bgColor
  738. }).appendTo(holder);
  739. }
  740. function enableShade() {
  741. if (!enabled) {
  742. enabled = true;
  743. holder.insertBefore($img);
  744. updateAuto();
  745. Selection.setBgOpacity(1,0,1);
  746. $img2.hide();
  747. setBgColor(options.shadeColor||options.bgColor,1);
  748. if (Selection.isAwake())
  749. {
  750. setOpacity(options.bgOpacity,1);
  751. }
  752. else setOpacity(1,1);
  753. }
  754. }
  755. function setBgColor(color,now) {
  756. colorChangeMacro(getShades(),color,now);
  757. }
  758. function disableShade() {
  759. if (enabled) {
  760. holder.remove();
  761. $img2.show();
  762. enabled = false;
  763. if (Selection.isAwake()) {
  764. Selection.setBgOpacity(options.bgOpacity,1,1);
  765. } else {
  766. Selection.setBgOpacity(1,1,1);
  767. Selection.disableHandles();
  768. }
  769. colorChangeMacro($div,0,1);
  770. }
  771. }
  772. function setOpacity(opacity,now) {
  773. if (enabled) {
  774. if (options.bgFade && !now) {
  775. holder.animate({
  776. opacity: 1-opacity
  777. },{
  778. queue: false,
  779. duration: options.fadeTime
  780. });
  781. }
  782. else holder.css({opacity:1-opacity});
  783. }
  784. }
  785. function refreshAll() {
  786. options.shade ? enableShade() : disableShade();
  787. if (Selection.isAwake()) setOpacity(options.bgOpacity);
  788. }
  789. function getShades() {
  790. return holder.children();
  791. }
  792. return {
  793. update: updateAuto,
  794. updateRaw: updateShade,
  795. getShades: getShades,
  796. setBgColor: setBgColor,
  797. enable: enableShade,
  798. disable: disableShade,
  799. resize: resizeShades,
  800. refresh: refreshAll,
  801. opacity: setOpacity
  802. };
  803. }());
  804. // }}}
  805. // Selection Module {{{
  806. var Selection = (function () {
  807. var awake,
  808. hdep = 370,
  809. borders = {},
  810. handle = {},
  811. dragbar = {},
  812. seehandles = false;
  813. // Private Methods
  814. function insertBorder(type) //{{{
  815. {
  816. var jq = $('<div />').css({
  817. position: 'absolute',
  818. opacity: options.borderOpacity
  819. }).addClass(cssClass(type));
  820. $img_holder.append(jq);
  821. return jq;
  822. }
  823. //}}}
  824. function dragDiv(ord, zi) //{{{
  825. {
  826. var jq = $('<div />').mousedown(createDragger(ord)).css({
  827. cursor: ord + '-resize',
  828. position: 'absolute',
  829. zIndex: zi
  830. }).addClass('ord-'+ord);
  831. if (Touch.support) {
  832. jq.bind('touchstart.jcrop', Touch.createDragger(ord));
  833. }
  834. $hdl_holder.append(jq);
  835. return jq;
  836. }
  837. function dragDiv2(ord, zi) //{{{
  838. {
  839. var jq = $('<div />').mousedown(createDragger(ord)).css({
  840. cursor: ord + '-resize',
  841. position: 'absolute',
  842. zIndex: zi
  843. }).addClass('ord2-'+ord);
  844. if (Touch.support) {
  845. jq.bind('touchstart.jcrop', Touch.createDragger(ord));
  846. }
  847. $hdl_holder.append(jq);
  848. return jq;
  849. }
  850. //}}}
  851. function insertHandle(ord) //{{{
  852. {
  853. var hs = options.handleSize,
  854. div = dragDiv(ord, hdep++).css({
  855. opacity: options.handleOpacity
  856. }).addClass(cssClass('handle'));
  857. if (hs) { div.width(hs).height(hs);}
  858. return div;
  859. }
  860. //}}}
  861. function insertDragbar(ord) //{{{
  862. {
  863. return dragDiv2(ord, hdep++).addClass('jcrop-dragbar');
  864. }
  865. //}}}
  866. function createDragbars(li) //{{{
  867. {
  868. var i;
  869. for (i = 0; i < li.length; i++) {
  870. dragbar[li[i]] = insertDragbar(li[i]);
  871. }
  872. }
  873. //}}}
  874. function createBorders(li) //{{{
  875. {
  876. var cl,i;
  877. for (i = 0; i < li.length; i++) {
  878. switch(li[i]){
  879. case'n': cl='hline'; break;
  880. case's': cl='hline bottom'; break;
  881. case'e': cl='vline right'; break;
  882. case'w': cl='vline'; break;
  883. }
  884. borders[li[i]] = insertBorder(cl);
  885. }
  886. }
  887. //}}}
  888. //createHandles: ['n','s','e','w','nw','ne','se','sw'],
  889. function createHandles(li) //{{{
  890. {
  891. var i;
  892. for (i = 0; i < li.length; i++) {
  893. handle[li[i]] = insertHandle(li[i]);
  894. }
  895. }
  896. //}}}
  897. function moveto(x, y) //{{{
  898. {
  899. if (!options.shade) {
  900. $img2.css({
  901. top: px(-y),
  902. left: px(-x)
  903. });
  904. }
  905. $sel.css({
  906. top: px(y),
  907. left: px(x)
  908. });
  909. }
  910. //}}}
  911. function resize(w, h) //{{{
  912. {
  913. $sel.width(Math.round(w)).height(Math.round(h));
  914. }
  915. //}}}
  916. function refresh() //{{{
  917. {
  918. var c = Coords.getFixed();
  919. Coords.setPressed([c.x, c.y]);
  920. Coords.setCurrent([c.x2, c.y2]);
  921. updateVisible();
  922. }
  923. //}}}
  924. // Internal Methods
  925. function updateVisible(select) //{{{
  926. {
  927. if (awake) {
  928. return update(select);
  929. }
  930. }
  931. //}}}
  932. function update(select) //{{{
  933. {
  934. var c = Coords.getFixed();
  935. resize(c.w, c.h);
  936. moveto(c.x, c.y);
  937. if (options.shade) Shade.updateRaw(c);
  938. awake || show();
  939. if (select) {
  940. options.onSelect.call(api, unscale(c));
  941. } else {
  942. options.onChange.call(api, unscale(c));
  943. }
  944. }
  945. //}}}
  946. function setBgOpacity(opacity,force,now) //{{{
  947. {
  948. if (!awake && !force) return;
  949. if (options.bgFade && !now) {
  950. $img.animate({
  951. opacity: opacity
  952. },{
  953. queue: false,
  954. duration: options.fadeTime
  955. });
  956. } else {
  957. $img.css('opacity', opacity);
  958. }
  959. }
  960. //}}}
  961. function show() //{{{
  962. {
  963. $sel.show();
  964. if (options.shade) Shade.opacity(bgopacity);
  965. else setBgOpacity(bgopacity,true);
  966. awake = true;
  967. }
  968. //}}}
  969. function release() //{{{
  970. {
  971. disableHandles();
  972. $sel.hide();
  973. if (options.shade) Shade.opacity(1);
  974. else setBgOpacity(1);
  975. awake = false;
  976. options.onRelease.call(api);
  977. }
  978. //}}}
  979. function showHandles() //{{{
  980. {
  981. if (seehandles) {
  982. $hdl_holder.show();
  983. }
  984. }
  985. //}}}
  986. function enableHandles() //{{{
  987. {
  988. seehandles = true;
  989. if (options.allowResize) {
  990. $hdl_holder.show();
  991. return true;
  992. }
  993. }
  994. //}}}
  995. function disableHandles() //{{{
  996. {
  997. seehandles = false;
  998. $hdl_holder.hide();
  999. }
  1000. //}}}
  1001. function animMode(v) //{{{
  1002. {
  1003. if (v) {
  1004. animating = true;
  1005. disableHandles();
  1006. } else {
  1007. animating = false;
  1008. enableHandles();
  1009. }
  1010. }
  1011. //}}}
  1012. function done() //{{{
  1013. {
  1014. animMode(false);
  1015. refresh();
  1016. }
  1017. //}}}
  1018. // Insert draggable elements {{{
  1019. // Insert border divs for outline
  1020. if ($.isArray(options.createDragbars))
  1021. createDragbars(options.createDragbars);
  1022. if ($.isArray(options.createHandles))
  1023. createHandles(options.createHandles);
  1024. if (options.drawBorders && $.isArray(options.createBorders))
  1025. createBorders(options.createBorders);
  1026. //}}}
  1027. // This is a hack for iOS5 to support drag/move touch functionality
  1028. $(document).bind('touchstart.jcrop-ios',function(e) {
  1029. if ($(e.currentTarget).hasClass('jcrop-tracker')) e.stopPropagation();
  1030. });
  1031. var $track = newTracker().mousedown(createDragger('move')).css({
  1032. cursor: 'move',
  1033. position: 'absolute',
  1034. zIndex: 360
  1035. });
  1036. if (Touch.support) {
  1037. $track.bind('touchstart.jcrop', Touch.createDragger('move'));
  1038. }
  1039. $img_holder.append($track);
  1040. disableHandles();
  1041. return {
  1042. updateVisible: updateVisible,
  1043. update: update,
  1044. release: release,
  1045. refresh: refresh,
  1046. isAwake: function () {
  1047. return awake;
  1048. },
  1049. setCursor: function (cursor) {
  1050. $track.css('cursor', cursor);
  1051. },
  1052. enableHandles: enableHandles,
  1053. enableOnly: function () {
  1054. seehandles = true;
  1055. },
  1056. showHandles: showHandles,
  1057. disableHandles: disableHandles,
  1058. animMode: animMode,
  1059. setBgOpacity: setBgOpacity,
  1060. done: done
  1061. };
  1062. }());
  1063. //}}}
  1064. // Tracker Module {{{
  1065. var Tracker = (function () {
  1066. var onMove = function () {},
  1067. onDone = function () {},
  1068. trackDoc = options.trackDocument;
  1069. function toFront(touch) //{{{
  1070. {
  1071. $trk.css({
  1072. zIndex: 450
  1073. });
  1074. if (touch)
  1075. $(document)
  1076. .bind('touchmove.jcrop', trackTouchMove)
  1077. .bind('touchend.jcrop', trackTouchEnd);
  1078. else if (trackDoc)
  1079. $(document)
  1080. .bind('mousemove.jcrop',trackMove)
  1081. .bind('mouseup.jcrop',trackUp);
  1082. }
  1083. //}}}
  1084. function toBack() //{{{
  1085. {
  1086. $trk.css({
  1087. zIndex: 290
  1088. });
  1089. $(document).unbind('.jcrop');
  1090. }
  1091. //}}}
  1092. function trackMove(e) //{{{
  1093. {
  1094. onMove(mouseAbs(e));
  1095. return false;
  1096. }
  1097. //}}}
  1098. function trackUp(e) //{{{
  1099. {
  1100. e.preventDefault();
  1101. e.stopPropagation();
  1102. if (btndown) {
  1103. btndown = false;
  1104. onDone(mouseAbs(e));
  1105. if (Selection.isAwake()) {
  1106. options.onSelect.call(api, unscale(Coords.getFixed()));
  1107. }
  1108. toBack();
  1109. onMove = function () {};
  1110. onDone = function () {};
  1111. }
  1112. return false;
  1113. }
  1114. //}}}
  1115. function activateHandlers(move, done, touch) //{{{
  1116. {
  1117. btndown = true;
  1118. onMove = move;
  1119. onDone = done;
  1120. toFront(touch);
  1121. return false;
  1122. }
  1123. //}}}
  1124. function trackTouchMove(e) //{{{
  1125. {
  1126. onMove(mouseAbs(Touch.cfilter(e)));
  1127. return false;
  1128. }
  1129. //}}}
  1130. function trackTouchEnd(e) //{{{
  1131. {
  1132. return trackUp(Touch.cfilter(e));
  1133. }
  1134. //}}}
  1135. function setCursor(t) //{{{
  1136. {
  1137. $trk.css('cursor', t);
  1138. }
  1139. //}}}
  1140. if (!trackDoc) {
  1141. $trk.mousemove(trackMove).mouseup(trackUp).mouseout(trackUp);
  1142. }
  1143. $img.before($trk);
  1144. return {
  1145. activateHandlers: activateHandlers,
  1146. setCursor: setCursor
  1147. };
  1148. }());
  1149. //}}}
  1150. // KeyManager Module {{{
  1151. var KeyManager = (function () {
  1152. var $keymgr = $('<input type="radio" />').css({
  1153. position: 'fixed',
  1154. left: '-120px',
  1155. width: '12px'
  1156. }).addClass('jcrop-keymgr'),
  1157. $keywrap = $('<div />').css({
  1158. position: 'absolute',
  1159. overflow: 'hidden'
  1160. }).append($keymgr);
  1161. function watchKeys() //{{{
  1162. {
  1163. if (options.keySupport) {
  1164. $keymgr.show();
  1165. $keymgr.focus();
  1166. }
  1167. }
  1168. //}}}
  1169. function onBlur(e) //{{{
  1170. {
  1171. $keymgr.hide();
  1172. }
  1173. //}}}
  1174. function doNudge(e, x, y) //{{{
  1175. {
  1176. if (options.allowMove) {
  1177. Coords.moveOffset([x, y]);
  1178. Selection.updateVisible(true);
  1179. }
  1180. e.preventDefault();
  1181. e.stopPropagation();
  1182. }
  1183. //}}}
  1184. function parseKey(e) //{{{
  1185. {
  1186. if (e.ctrlKey || e.metaKey) {
  1187. return true;
  1188. }
  1189. shift_down = e.shiftKey ? true : false;
  1190. var nudge = shift_down ? 10 : 1;
  1191. switch (e.keyCode) {
  1192. case 37:
  1193. doNudge(e, -nudge, 0);
  1194. break;
  1195. case 39:
  1196. doNudge(e, nudge, 0);
  1197. break;
  1198. case 38:
  1199. doNudge(e, 0, -nudge);
  1200. break;
  1201. case 40:
  1202. doNudge(e, 0, nudge);
  1203. break;
  1204. case 27:
  1205. if (options.allowSelect) Selection.release();
  1206. break;
  1207. case 9:
  1208. return true;
  1209. }
  1210. return false;
  1211. }
  1212. //}}}
  1213. if (options.keySupport) {
  1214. $keymgr.keydown(parseKey).blur(onBlur);
  1215. if (ie6mode || !options.fixedSupport) {
  1216. $keymgr.css({
  1217. position: 'absolute',
  1218. left: '-20px'
  1219. });
  1220. $keywrap.append($keymgr).insertBefore($img);
  1221. } else {
  1222. $keymgr.insertBefore($img);
  1223. }
  1224. }
  1225. return {
  1226. watchKeys: watchKeys
  1227. };
  1228. }());
  1229. //}}}
  1230. // }}}
  1231. // API methods {{{
  1232. function setClass(cname) //{{{
  1233. {
  1234. $div.removeClass().addClass(cssClass('holder')).addClass(cname);
  1235. }
  1236. //}}}
  1237. function animateTo(a, callback) //{{{
  1238. {
  1239. var x1 = a[0] / xscale,
  1240. y1 = a[1] / yscale,
  1241. x2 = a[2] / xscale,
  1242. y2 = a[3] / yscale;
  1243. if (animating) {
  1244. return;
  1245. }
  1246. var animto = Coords.flipCoords(x1, y1, x2, y2),
  1247. c = Coords.getFixed(),
  1248. initcr = [c.x, c.y, c.x2, c.y2],
  1249. animat = initcr,
  1250. interv = options.animationDelay,
  1251. ix1 = animto[0] - initcr[0],
  1252. iy1 = animto[1] - initcr[1],
  1253. ix2 = animto[2] - initcr[2],
  1254. iy2 = animto[3] - initcr[3],
  1255. pcent = 0,
  1256. velocity = options.swingSpeed;
  1257. x1 = animat[0];
  1258. y1 = animat[1];
  1259. x2 = animat[2];
  1260. y2 = animat[3];
  1261. Selection.animMode(true);
  1262. var anim_timer;
  1263. function queueAnimator() {
  1264. window.setTimeout(animator, interv);
  1265. }
  1266. var animator = (function () {
  1267. return function () {
  1268. pcent += (100 - pcent) / velocity;
  1269. animat[0] = Math.round(x1 + ((pcent / 100) * ix1));
  1270. animat[1] = Math.round(y1 + ((pcent / 100) * iy1));
  1271. animat[2] = Math.round(x2 + ((pcent / 100) * ix2));
  1272. animat[3] = Math.round(y2 + ((pcent / 100) * iy2));
  1273. if (pcent >= 99.8) {
  1274. pcent = 100;
  1275. }
  1276. if (pcent < 100) {
  1277. setSelectRaw(animat);
  1278. queueAnimator();
  1279. } else {
  1280. Selection.done();
  1281. Selection.animMode(false);
  1282. if (typeof(callback) === 'function') {
  1283. callback.call(api);
  1284. }
  1285. }
  1286. };
  1287. }());
  1288. queueAnimator();
  1289. }
  1290. //}}}
  1291. function setSelect(rect) //{{{
  1292. {
  1293. setSelectRaw([rect[0] / xscale, rect[1] / yscale, rect[2] / xscale, rect[3] / yscale]);
  1294. options.onSelect.call(api, unscale(Coords.getFixed()));
  1295. Selection.enableHandles();
  1296. }
  1297. //}}}
  1298. function setSelectRaw(l) //{{{
  1299. {
  1300. Coords.setPressed([l[0], l[1]]);
  1301. Coords.setCurrent([l[2], l[3]]);
  1302. Selection.update();
  1303. }
  1304. //}}}
  1305. function tellSelect() //{{{
  1306. {
  1307. return unscale(Coords.getFixed());
  1308. }
  1309. //}}}
  1310. function tellScaled() //{{{
  1311. {
  1312. return Coords.getFixed();
  1313. }
  1314. //}}}
  1315. function setOptionsNew(opt) //{{{
  1316. {
  1317. setOptions(opt);
  1318. interfaceUpdate();
  1319. }
  1320. //}}}
  1321. function disableCrop() //{{{
  1322. {
  1323. options.disabled = true;
  1324. Selection.disableHandles();
  1325. Selection.setCursor('default');
  1326. Tracker.setCursor('default');
  1327. }
  1328. //}}}
  1329. function enableCrop() //{{{
  1330. {
  1331. options.disabled = false;
  1332. interfaceUpdate();
  1333. }
  1334. //}}}
  1335. function cancelCrop() //{{{
  1336. {
  1337. Selection.done();
  1338. Tracker.activateHandlers(null, null);
  1339. }
  1340. //}}}
  1341. function destroy() //{{{
  1342. {
  1343. $div.remove();
  1344. $origimg.show();
  1345. $origimg.css('visibility','visible');
  1346. $(obj).removeData('Jcrop');
  1347. }
  1348. //}}}
  1349. function setImage(src, callback) //{{{
  1350. {
  1351. Selection.release();
  1352. disableCrop();
  1353. var img = new Image();
  1354. img.onload = function () {
  1355. var iw = img.width;
  1356. var ih = img.height;
  1357. var bw = options.boxWidth;
  1358. var bh = options.boxHeight;
  1359. $img.width(iw).height(ih);
  1360. $img.attr('src', src);
  1361. $img2.attr('src', src);
  1362. presize($img, bw, bh);
  1363. boundx = $img.width();
  1364. boundy = $img.height();
  1365. $img2.width(boundx).height(boundy);
  1366. $trk.width(boundx + (bound * 2)).height(boundy + (bound * 2));
  1367. $div.width(boundx).height(boundy);
  1368. Shade.resize(boundx,boundy);
  1369. enableCrop();
  1370. if (typeof(callback) === 'function') {
  1371. callback.call(api);
  1372. }
  1373. };
  1374. img.src = src;
  1375. }
  1376. //}}}
  1377. function colorChangeMacro($obj,color,now) {
  1378. var mycolor = color || options.bgColor;
  1379. if (options.bgFade && supportsColorFade() && options.fadeTime && !now) {
  1380. $obj.animate({
  1381. backgroundColor: mycolor
  1382. }, {
  1383. queue: false,
  1384. duration: options.fadeTime
  1385. });
  1386. } else {
  1387. $obj.css('backgroundColor', mycolor);
  1388. }
  1389. }
  1390. function interfaceUpdate(alt) //{{{
  1391. // This method tweaks the interface based on options object.
  1392. // Called when options are changed and at end of initialization.
  1393. {
  1394. if (options.allowResize) {
  1395. if (alt) {
  1396. Selection.enableOnly();
  1397. } else {
  1398. Selection.enableHandles();
  1399. }
  1400. } else {
  1401. Selection.disableHandles();
  1402. }
  1403. Tracker.setCursor(options.allowSelect ? 'crosshair' : 'default');
  1404. Selection.setCursor(options.allowMove ? 'move' : 'default');
  1405. if (options.hasOwnProperty('trueSize')) {
  1406. xscale = options.trueSize[0] / boundx;
  1407. yscale = options.trueSize[1] / boundy;
  1408. }
  1409. if (options.hasOwnProperty('setSelect')) {
  1410. setSelect(options.setSelect);
  1411. Selection.done();
  1412. delete(options.setSelect);
  1413. }
  1414. Shade.refresh();
  1415. if (options.bgColor != bgcolor) {
  1416. colorChangeMacro(
  1417. options.shade? Shade.getShades(): $div,
  1418. options.shade?
  1419. (options.shadeColor || options.bgColor):
  1420. options.bgColor
  1421. );
  1422. bgcolor = options.bgColor;
  1423. }
  1424. if (bgopacity != options.bgOpacity) {
  1425. bgopacity = options.bgOpacity;
  1426. if (options.shade) Shade.refresh();
  1427. else Selection.setBgOpacity(bgopacity);
  1428. }
  1429. xlimit = options.maxSize[0] || 0;
  1430. ylimit = options.maxSize[1] || 0;
  1431. xmin = options.minSize[0] || 0;
  1432. ymin = options.minSize[1] || 0;
  1433. if (options.hasOwnProperty('outerImage')) {
  1434. $img.attr('src', options.outerImage);
  1435. delete(options.outerImage);
  1436. }
  1437. Selection.refresh();
  1438. }
  1439. //}}}
  1440. //}}}
  1441. if (Touch.support) $trk.bind('touchstart.jcrop', Touch.newSelection);
  1442. $hdl_holder.hide();
  1443. interfaceUpdate(true);
  1444. var api = {
  1445. setImage: setImage,
  1446. animateTo: animateTo,
  1447. setSelect: setSelect,
  1448. setOptions: setOptionsNew,
  1449. tellSelect: tellSelect,
  1450. tellScaled: tellScaled,
  1451. setClass: setClass,
  1452. disable: disableCrop,
  1453. enable: enableCrop,
  1454. cancel: cancelCrop,
  1455. release: Selection.release,
  1456. destroy: destroy,
  1457. focus: KeyManager.watchKeys,
  1458. getBounds: function () {
  1459. return [boundx * xscale, boundy * yscale];
  1460. },
  1461. getWidgetSize: function () {
  1462. return [boundx, boundy];
  1463. },
  1464. getScaleFactor: function () {
  1465. return [xscale, yscale];
  1466. },
  1467. getOptions: function() {
  1468. // careful: internal values are returned
  1469. return options;
  1470. },
  1471. ui: {
  1472. holder: $div,
  1473. selection: $sel
  1474. }
  1475. };
  1476. if (is_msie) $div.bind('selectstart', function () { return false; });
  1477. $origimg.data('Jcrop', api);
  1478. return api;
  1479. };
  1480. $.fn.Jcrop = function (options, callback) //{{{
  1481. {
  1482. var api;
  1483. // Iterate over each object, attach Jcrop
  1484. this.each(function () {
  1485. // If we've already attached to this object
  1486. if ($(this).data('Jcrop')) {
  1487. // The API can be requested this way (undocumented)
  1488. if (options === 'api') return $(this).data('Jcrop');
  1489. // Otherwise, we just reset the options...
  1490. else $(this).data('Jcrop').setOptions(options);
  1491. }
  1492. // If we haven't been attached, preload and attach
  1493. else {
  1494. if (this.tagName == 'IMG')
  1495. $.Jcrop.Loader(this,function(){
  1496. $(this).css({display:'block',visibility:'hidden'});
  1497. api = $.Jcrop(this, options);
  1498. if ($.isFunction(callback)) callback.call(api);
  1499. });
  1500. else {
  1501. $(this).css({display:'block',visibility:'hidden'});
  1502. api = $.Jcrop(this, options);
  1503. if ($.isFunction(callback)) callback.call(api);
  1504. }
  1505. }
  1506. });
  1507. // Return "this" so the object is chainable (jQuery-style)
  1508. return this;
  1509. };
  1510. //}}}
  1511. // $.Jcrop.Loader - basic image loader {{{
  1512. $.Jcrop.Loader = function(imgobj,success,error){
  1513. var $img = $(imgobj), img = $img[0];
  1514. function completeCheck(){
  1515. if (img.complete) {
  1516. $img.unbind('.jcloader');
  1517. if ($.isFunction(success)) success.call(img);
  1518. }
  1519. else window.setTimeout(completeCheck,50);
  1520. }
  1521. $img
  1522. .bind('load.jcloader',completeCheck)
  1523. .bind('error.jcloader',function(e){
  1524. $img.unbind('.jcloader');
  1525. if ($.isFunction(error)) error.call(img);
  1526. });
  1527. if (img.complete && $.isFunction(success)){
  1528. $img.unbind('.jcloader');
  1529. success.call(img);
  1530. }
  1531. };
  1532. //}}}
  1533. // Global Defaults {{{
  1534. $.Jcrop.defaults = {
  1535. // Basic Settings
  1536. allowSelect: true,
  1537. allowMove: true,
  1538. allowResize: true,
  1539. trackDocument: true,
  1540. // Styling Options
  1541. baseClass: 'jcrop',
  1542. addClass: null,
  1543. bgColor: 'black',
  1544. bgOpacity: 0.6,
  1545. bgFade: false,
  1546. borderOpacity: 0.4,
  1547. handleOpacity: 0.5,
  1548. handleSize: null,
  1549. aspectRatio: 0,
  1550. keySupport: false,
  1551. createHandles: ['n','s','e','w','nw','ne','se','sw'],
  1552. createDragbars: ['n','s','e','w'],
  1553. createBorders: ['n','s','e','w'],
  1554. drawBorders: true,
  1555. dragEdges: true,
  1556. fixedSupport: true,
  1557. touchSupport: null,
  1558. shade: null,
  1559. boxWidth: 0,
  1560. boxHeight: 0,
  1561. boundary: 0,
  1562. fadeTime: 400,
  1563. animationDelay: 20,
  1564. swingSpeed: 3,
  1565. minSelect: [0, 0],
  1566. maxSize: [0, 0],
  1567. minSize: [0, 0],
  1568. // Callbacks / Event Handlers
  1569. onChange: function () {},
  1570. onSelect: function () {},
  1571. onDblClick: function () {},
  1572. onRelease: function () {}
  1573. };
  1574. // }}}
  1575. }(jQuery));