2010-01-31 07:11:57 +00:00
/ *
2009-11-25 13:51:42 +00:00
* svg - editor . js
*
2012-09-16 18:53:27 +00:00
* Licensed under the MIT License
2009-11-25 13:51:42 +00:00
*
2010-01-13 21:03:00 +00:00
* Copyright ( c ) 2010 Alexis Deveria
* Copyright ( c ) 2010 Pavol Rusnak
* Copyright ( c ) 2010 Jeff Schiller
2010-04-02 12:47:00 +00:00
* Copyright ( c ) 2010 Narendra Sisodiya
2009-11-25 13:51:42 +00:00
*
* /
2009-06-01 15:55:21 +00:00
2010-11-06 00:45:21 +00:00
// Dependencies:
2010-11-07 05:20:03 +00:00
// 1) units.js
2011-01-13 07:47:21 +00:00
// 2) browser.js
// 3) svgcanvas.js
2010-11-06 00:45:21 +00:00
2012-10-22 10:47:50 +00:00
( function ( ) {
2013-02-15 21:57:35 +00:00
if ( ! window . svgEditor ) window . svgEditor = function ( $ ) {
2013-02-15 13:38:55 +00:00
var svgCanvas ,
Editor = { } ,
2013-02-15 21:57:35 +00:00
isReady = false ,
2013-02-15 13:38:55 +00:00
defaultPrefs = {
lang : 'en' ,
iconsize : 'm' ,
bkgd _color : '#FFF' ,
bkgd _url : '' ,
img _save : 'embed'
2010-03-11 21:07:13 +00:00
} ,
curPrefs = { } ,
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Note: Difference between Prefs and Config is that Prefs can be
2010-03-12 20:43:33 +00:00
// changed in the UI and are stored in the browser, config can not
2010-03-11 21:07:13 +00:00
curConfig = {
2013-01-09 23:14:35 +00:00
canvasName : 'default' ,
2010-03-12 20:43:33 +00:00
canvas _expansion : 3 ,
dimensions : [ 640 , 480 ] ,
initFill : {
2013-10-13 23:23:08 +00:00
color : 'FF0000' , // solid red
2010-03-12 20:43:33 +00:00
opacity : 1
} ,
initStroke : {
2010-03-18 14:37:09 +00:00
width : 5 ,
2013-10-13 23:23:08 +00:00
color : '000000' , // solid black
2010-03-12 20:43:33 +00:00
opacity : 1
} ,
2010-03-16 19:28:57 +00:00
initOpacity : 1 ,
imgPath : 'images/' ,
langPath : 'locale/' ,
2010-03-24 20:13:13 +00:00
extPath : 'extensions/' ,
2010-07-27 20:25:50 +00:00
jGraduatePath : 'jgraduate/images/' ,
2013-02-15 21:57:35 +00:00
extensions : [
'ext-markers.js' ,
'ext-connector.js' ,
'ext-eyedropper.js' ,
'ext-shapes.js' ,
'ext-imagelib.js' ,
'ext-grid.js' ,
'ext-polygon.js' ,
'ext-star.js'
] ,
2010-03-18 14:37:09 +00:00
initTool : 'select' ,
2010-07-27 20:25:50 +00:00
wireframe : false ,
2010-09-09 12:46:34 +00:00
colorPickerCSS : null ,
2010-09-10 13:23:31 +00:00
gridSnapping : false ,
2013-02-15 21:57:35 +00:00
gridColor : '#000' ,
2010-10-05 17:13:00 +00:00
baseUnit : 'px' ,
2010-10-01 18:59:12 +00:00
snappingStep : 10 ,
showRulers : true
2010-03-12 20:43:33 +00:00
} ,
2010-07-09 19:09:37 +00:00
uiStrings = Editor . uiStrings = {
2011-01-03 18:45:36 +00:00
common : {
2013-02-15 21:57:35 +00:00
ok : 'OK' ,
cancel : 'Cancel' ,
key _up : 'Up' ,
key _down : 'Down' ,
key _backspace : 'Backspace' ,
key _del : 'Del'
2011-01-03 18:45:36 +00:00
} ,
2011-01-16 20:25:57 +00:00
// This is needed if the locale is English, since the locale strings are not read in that instance.
layers : {
2013-02-15 21:57:35 +00:00
layer : 'Layer'
2011-01-16 20:25:57 +00:00
} ,
2011-01-03 18:45:36 +00:00
notification : {
2013-02-15 21:57:35 +00:00
invalidAttrValGiven : 'Invalid value given' ,
noContentToFitTo : 'No content to fit to' ,
dupeLayerName : 'There is already a layer named that!' ,
enterUniqueLayerName : 'Please enter a unique layer name' ,
enterNewLayerName : 'Please enter the new layer name' ,
layerHasThatName : 'Layer already has that name' ,
QmoveElemsToLayer : 'Move selected elements to layer \'%s\'?' ,
QwantToClear : 'Do you want to clear the drawing?\nThis will also erase your undo history!' ,
QwantToOpen : 'Do you want to open a new file?\nThis will also erase your undo history!' ,
QerrorsRevertToSource : 'There were parsing errors in your SVG source.\nRevert back to original SVG source?' ,
QignoreSourceChanges : 'Ignore changes made to SVG source?' ,
featNotSupported : 'Feature not supported' ,
enterNewImgURL : 'Enter the new image URL' ,
defsFailOnSave : 'NOTE: Due to a bug in your browser, this image may appear wrong (missing gradients or elements). It will however appear correct once actually saved.' ,
loadingImage : 'Loading image, please wait...' ,
saveFromBrowser : 'Select \'Save As...\' in your browser to save this image as a %s file.' ,
noteTheseIssues : 'Also note the following issues: ' ,
unsavedChanges : 'There are unsaved changes.' ,
enterNewLinkURL : 'Enter the new hyperlink URL' ,
errorLoadingSVG : 'Error: Unable to load SVG data' ,
URLloadFail : 'Unable to load from URL' ,
retrieving : 'Retrieving \'%s\' ...'
2011-01-03 18:45:36 +00:00
}
2013-02-15 13:38:55 +00:00
} ,
customHandlers = { } ;
2012-10-22 10:47:50 +00:00
2010-03-16 19:28:57 +00:00
Editor . curConfig = curConfig ;
2010-07-27 13:37:52 +00:00
Editor . tool _scale = 1 ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Store and retrieve preferences
$ . pref = function ( key , val ) {
2013-02-15 21:57:35 +00:00
if ( val ) curPrefs [ key ] = val ;
2010-03-11 21:07:13 +00:00
key = 'svg-edit-' + key ;
var host = location . hostname ,
2013-02-15 21:57:35 +00:00
onWeb = host && host . indexOf ( '.' ) >= 0 ,
2010-03-11 21:07:13 +00:00
store = ( val != undefined ) ,
storage = false ;
// Some FF versions throw security errors here
2012-10-22 10:47:50 +00:00
try {
2013-02-15 21:57:35 +00:00
if ( window . localStorage ) { // && onWeb removed so Webkit works locally
2010-03-11 21:07:13 +00:00
storage = localStorage ;
}
} catch ( e ) { }
2012-10-22 10:47:50 +00:00
try {
2013-02-15 21:57:35 +00:00
if ( window . globalStorage && onWeb ) {
2010-03-11 21:07:13 +00:00
storage = globalStorage [ host ] ;
}
} catch ( e ) { }
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( storage ) {
if ( store ) {
storage . setItem ( key , val ) ;
} else if ( storage . getItem ( key ) ) {
return storage . getItem ( key ) + '' ; // Convert to string for FF (.value fails in Webkit)
}
} else if ( window . widget ) {
if ( store ) {
widget . setPreferenceForKey ( val , key ) ;
} else {
return widget . preferenceForKey ( key ) ;
}
2010-03-11 21:07:13 +00:00
} else {
2013-02-15 21:57:35 +00:00
if ( store ) {
2010-03-11 21:07:13 +00:00
var d = new Date ( ) ;
d . setTime ( d . getTime ( ) + 31536000000 ) ;
val = encodeURIComponent ( val ) ;
document . cookie = key + '=' + val + '; expires=' + d . toUTCString ( ) ;
} else {
2013-02-15 21:57:35 +00:00
var result = document . cookie . match ( new RegExp ( key + '=([^;]+)' ) ) ;
return result ? decodeURIComponent ( result [ 1 ] ) : '' ;
2010-03-11 21:07:13 +00:00
}
2010-01-21 17:38:37 +00:00
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
Editor . setConfig = function ( opts ) {
$ . each ( opts , function ( key , val ) {
2010-03-12 20:43:33 +00:00
// Only allow prefs defined in defaultPrefs
2013-02-15 21:57:35 +00:00
if ( key in defaultPrefs ) {
2010-03-11 21:07:13 +00:00
$ . pref ( key , val ) ;
2010-02-12 20:29:59 +00:00
}
2010-03-11 21:07:13 +00:00
} ) ;
2010-03-12 20:43:33 +00:00
$ . extend ( true , curConfig , opts ) ;
2013-02-15 21:57:35 +00:00
if ( opts . extensions ) {
2010-03-24 20:13:13 +00:00
curConfig . extensions = opts . extensions ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Extension mechanisms must call setCustomHandlers with two functions: opts.open and opts.save
// opts.open's responsibilities are:
// - invoke a file chooser dialog in 'open' mode
// - let user pick a SVG file
// - calls setCanvas.setSvgString() with the string contents of that file
// opts.save's responsibilities are:
2012-10-22 10:47:50 +00:00
// - accept the string contents of the current document
2010-03-11 21:07:13 +00:00
// - invoke a file chooser dialog in 'save' mode
// - save the file to location chosen by the user
Editor . setCustomHandlers = function ( opts ) {
2010-07-16 15:46:54 +00:00
Editor . ready ( function ( ) {
2013-02-15 21:57:35 +00:00
if ( opts . open ) {
2010-08-13 15:30:00 +00:00
$ ( '#tool_open > input[type="file"]' ) . remove ( ) ;
2010-07-16 15:46:54 +00:00
$ ( '#tool_open' ) . show ( ) ;
svgCanvas . open = opts . open ;
}
2013-02-15 21:57:35 +00:00
if ( opts . save ) {
2013-02-16 15:19:20 +00:00
Editor . showSaveWarning = false ;
2013-02-15 21:57:35 +00:00
svgCanvas . bind ( 'saved' , opts . save ) ;
2010-07-16 15:46:54 +00:00
}
2013-02-15 21:57:35 +00:00
if ( opts . pngsave ) {
svgCanvas . bind ( 'exported' , opts . pngsave ) ;
2010-07-16 15:46:54 +00:00
}
customHandlers = opts ;
} ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-12 09:41:50 +00:00
Editor . randomizeIds = function ( ) {
2013-02-15 13:38:55 +00:00
svgCanvas . randomizeIds ( arguments ) ;
} ;
2010-03-12 09:41:50 +00:00
2010-03-11 21:07:13 +00:00
Editor . init = function ( ) {
2010-09-22 19:09:21 +00:00
// For external openers
( function ( ) {
// let the opener know SVG Edit is ready
var w = window . opener ;
if ( w ) {
2013-02-15 13:38:55 +00:00
try {
2013-02-15 21:57:35 +00:00
var svgEditorReadyEvent = w . document . createEvent ( 'Event' ) ;
svgEditorReadyEvent . initEvent ( 'svgEditorReady' , true , true ) ;
2010-09-22 19:09:21 +00:00
w . document . documentElement . dispatchEvent ( svgEditorReadyEvent ) ;
2013-02-15 13:38:55 +00:00
}
2010-09-22 19:09:21 +00:00
catch ( e ) { }
}
} ) ( ) ;
2010-03-12 20:43:33 +00:00
( function ( ) {
// Load config/data from URL if given
2010-03-16 19:28:57 +00:00
var urldata = $ . deparam . querystring ( true ) ;
2013-02-15 21:57:35 +00:00
if ( ! $ . isEmptyObject ( urldata ) ) {
if ( urldata . dimensions ) {
2010-03-12 20:43:33 +00:00
urldata . dimensions = urldata . dimensions . split ( ',' ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( urldata . extensions ) {
2010-03-24 20:13:13 +00:00
urldata . extensions = urldata . extensions . split ( ',' ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( urldata . bkgd _color ) {
2010-03-16 19:28:57 +00:00
urldata . bkgd _color = '#' + urldata . bkgd _color ;
}
2010-03-12 20:43:33 +00:00
svgEditor . setConfig ( urldata ) ;
2012-10-22 10:47:50 +00:00
2010-03-12 20:43:33 +00:00
var src = urldata . source ;
2010-04-15 16:05:34 +00:00
var qstr = $ . param . querystring ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! src ) { // urldata.source may have been null if it ended with '='
if ( qstr . indexOf ( 'source=data:' ) >= 0 ) {
2010-10-08 17:54:45 +00:00
src = qstr . match ( /source=(data:[^&]*)/ ) [ 1 ] ;
}
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( src ) {
if ( src . indexOf ( 'data:' ) === 0 ) {
2010-03-12 20:43:33 +00:00
// plusses get replaced by spaces, so re-insert
2013-02-15 21:57:35 +00:00
src = src . replace ( / /g , '+' ) ;
2010-03-12 20:43:33 +00:00
Editor . loadFromDataURI ( src ) ;
} else {
Editor . loadFromString ( src ) ;
}
2013-02-15 21:57:35 +00:00
} else if ( qstr . indexOf ( 'paramurl=' ) !== - 1 ) {
2010-04-15 15:50:55 +00:00
// Get paramater URL (use full length of remaining location.href)
svgEditor . loadFromURL ( qstr . substr ( 9 ) ) ;
2013-02-15 21:57:35 +00:00
} else if ( urldata . url ) {
2010-03-12 20:43:33 +00:00
svgEditor . loadFromURL ( urldata . url ) ;
}
2013-01-09 23:14:35 +00:00
} else {
var name = 'svgedit-' + Editor . curConfig . canvasName ;
var cached = window . localStorage . getItem ( name ) ;
if ( cached ) {
Editor . loadFromString ( cached ) ;
}
2010-03-12 20:43:33 +00:00
}
} ) ( ) ;
2012-10-22 10:47:50 +00:00
2010-03-29 18:06:51 +00:00
var extFunc = function ( ) {
$ . each ( curConfig . extensions , function ( ) {
2010-07-22 19:10:51 +00:00
var extname = this ;
$ . getScript ( curConfig . extPath + extname , function ( d ) {
// Fails locally in Chrome 5
2013-02-15 21:57:35 +00:00
if ( ! d ) {
2010-07-22 19:10:51 +00:00
var s = document . createElement ( 'script' ) ;
s . src = curConfig . extPath + extname ;
document . querySelector ( 'head' ) . appendChild ( s ) ;
}
} ) ;
2010-03-29 18:06:51 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-12-17 21:11:18 +00:00
var good _langs = [ ] ;
2012-10-22 10:47:50 +00:00
2010-12-17 21:11:18 +00:00
$ ( '#lang_select option' ) . each ( function ( ) {
good _langs . push ( this . value ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
// var lang = ('lang' in curPrefs) ? curPrefs.lang : null;
2010-12-17 21:11:18 +00:00
Editor . putLocale ( null , good _langs ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-24 20:13:13 +00:00
// Load extensions
2010-09-20 12:49:36 +00:00
// Bit of a hack to run extensions in local Opera/IE9
2013-02-15 21:57:35 +00:00
if ( document . location . protocol === 'file:' ) {
2010-09-20 12:49:36 +00:00
setTimeout ( extFunc , 100 ) ;
2010-03-29 18:06:51 +00:00
} else {
extFunc ( ) ;
}
2010-03-16 19:28:57 +00:00
$ . svgIcons ( curConfig . imgPath + 'svg_edit_icons.svg' , {
w : 24 , h : 24 ,
id _match : false ,
2013-02-15 13:38:55 +00:00
no _img : ! svgedit . browser . isWebkit ( ) , // Opera & Firefox 4 gives odd behavior w/images
2010-03-16 19:28:57 +00:00
fallback _path : curConfig . imgPath ,
2013-02-15 21:57:35 +00:00
fallback : {
'new_image' : 'clear.png' ,
'save' : 'save.png' ,
'open' : 'open.png' ,
'source' : 'source.png' ,
'docprops' : 'document-properties.png' ,
'wireframe' : 'wireframe.png' ,
'undo' : 'undo.png' ,
'redo' : 'redo.png' ,
'select' : 'select.png' ,
'select_node' : 'select_node.png' ,
'pencil' : 'fhpath.png' ,
'pen' : 'line.png' ,
'square' : 'square.png' ,
'rect' : 'rect.png' ,
'fh_rect' : 'freehand-square.png' ,
'circle' : 'circle.png' ,
'ellipse' : 'ellipse.png' ,
'fh_ellipse' : 'freehand-circle.png' ,
'path' : 'path.png' ,
'text' : 'text.png' ,
'image' : 'image.png' ,
'zoom' : 'zoom.png' ,
'clone' : 'clone.png' ,
'node_clone' : 'node_clone.png' ,
'delete' : 'delete.png' ,
'node_delete' : 'node_delete.png' ,
'group' : 'shape_group.png' ,
'ungroup' : 'shape_ungroup.png' ,
'move_top' : 'move_top.png' ,
'move_bottom' : 'move_bottom.png' ,
'to_path' : 'to_path.png' ,
'link_controls' : 'link_controls.png' ,
'reorient' : 'reorient.png' ,
'align_left' : 'align-left.png' ,
'align_center' : 'align-center.png' ,
'align_right' : 'align-right.png' ,
'align_top' : 'align-top.png' ,
'align_middle' : 'align-middle.png' ,
'align_bottom' : 'align-bottom.png' ,
'go_up' : 'go-up.png' ,
'go_down' : 'go-down.png' ,
'ok' : 'save.png' ,
'cancel' : 'cancel.png' ,
'arrow_right' : 'flyouth.png' ,
'arrow_down' : 'dropdown.gif'
2010-03-16 19:28:57 +00:00
} ,
placement : {
2013-02-15 21:57:35 +00:00
'#logo' : 'logo' ,
'#tool_clear div,#layer_new' : 'new_image' ,
'#tool_save div' : 'save' ,
'#tool_export div' : 'export' ,
'#tool_open div div' : 'open' ,
'#tool_import div div' : 'import' ,
'#tool_source' : 'source' ,
'#tool_docprops > div' : 'docprops' ,
'#tool_wireframe' : 'wireframe' ,
'#tool_undo' : 'undo' ,
'#tool_redo' : 'redo' ,
'#tool_select' : 'select' ,
'#tool_fhpath' : 'pencil' ,
'#tool_line' : 'pen' ,
'#tool_rect,#tools_rect_show' : 'rect' ,
'#tool_square' : 'square' ,
'#tool_fhrect' : 'fh_rect' ,
'#tool_ellipse,#tools_ellipse_show' : 'ellipse' ,
'#tool_circle' : 'circle' ,
'#tool_fhellipse' : 'fh_ellipse' ,
'#tool_path' : 'path' ,
'#tool_text,#layer_rename' : 'text' ,
'#tool_image' : 'image' ,
'#tool_zoom' : 'zoom' ,
'#tool_clone,#tool_clone_multi' : 'clone' ,
'#tool_node_clone' : 'node_clone' ,
'#layer_delete,#tool_delete,#tool_delete_multi' : 'delete' ,
'#tool_node_delete' : 'node_delete' ,
'#tool_add_subpath' : 'add_subpath' ,
'#tool_openclose_path' : 'open_path' ,
'#tool_move_top' : 'move_top' ,
'#tool_move_bottom' : 'move_bottom' ,
'#tool_topath' : 'to_path' ,
'#tool_node_link' : 'link_controls' ,
'#tool_reorient' : 'reorient' ,
'#tool_group' : 'group' ,
'#tool_ungroup' : 'ungroup' ,
'#tool_unlink_use' : 'unlink_use' ,
'#tool_alignleft, #tool_posleft' : 'align_left' ,
'#tool_aligncenter, #tool_poscenter' : 'align_center' ,
'#tool_alignright, #tool_posright' : 'align_right' ,
'#tool_aligntop, #tool_postop' : 'align_top' ,
'#tool_alignmiddle, #tool_posmiddle' : 'align_middle' ,
'#tool_alignbottom, #tool_posbottom' : 'align_bottom' ,
'#cur_position' : 'align' ,
'#linecap_butt,#cur_linecap' : 'linecap_butt' ,
'#linecap_round' : 'linecap_round' ,
'#linecap_square' : 'linecap_square' ,
'#linejoin_miter,#cur_linejoin' : 'linejoin_miter' ,
'#linejoin_round' : 'linejoin_round' ,
'#linejoin_bevel' : 'linejoin_bevel' ,
'#url_notice' : 'warning' ,
'#layer_up' : 'go_up' ,
'#layer_down' : 'go_down' ,
'#layer_moreopts' : 'context_menu' ,
'#layerlist td.layervis' : 'eye' ,
'#tool_source_save,#tool_docprops_save,#tool_prefs_save' : 'ok' ,
'#tool_source_cancel,#tool_docprops_cancel,#tool_prefs_cancel' : 'cancel' ,
'#rwidthLabel, #iwidthLabel' : 'width' ,
'#rheightLabel, #iheightLabel' : 'height' ,
'#cornerRadiusLabel span' : 'c_radius' ,
'#angleLabel' : 'angle' ,
'#linkLabel,#tool_make_link,#tool_make_link_multi' : 'globe_link' ,
'#zoomLabel' : 'zoom' ,
2010-04-20 13:21:51 +00:00
'#tool_fill label' : 'fill' ,
'#tool_stroke .icon_label' : 'stroke' ,
'#group_opacityLabel' : 'opacity' ,
2010-04-20 15:53:19 +00:00
'#blurLabel' : 'blur' ,
'#font_sizeLabel' : 'fontsize' ,
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
'.flyout_arrow_horiz' : 'arrow_right' ,
'.dropdown button, #main_button .dropdown' : 'arrow_down' ,
'#palette .palette_item:first, #fill_bg, #stroke_bg' : 'no_color'
2010-03-16 19:28:57 +00:00
} ,
resize : {
2012-11-28 01:42:57 +00:00
'#logo .svg_icon' : 28 ,
2010-03-16 19:28:57 +00:00
'.flyout_arrow_horiz .svg_icon' : 5 ,
'.layer_button .svg_icon, #layerlist td.layervis .svg_icon' : 14 ,
'.dropdown button .svg_icon' : 7 ,
'#main_button .dropdown .svg_icon' : 9 ,
2012-11-28 01:42:57 +00:00
'.palette_item:first .svg_icon' : 15 ,
'#fill_bg .svg_icon, #stroke_bg .svg_icon' : 16 ,
2013-02-15 21:57:35 +00:00
'.toolbar_button button .svg_icon' : 16 ,
2010-04-20 13:21:51 +00:00
'.stroke_tool div div .svg_icon' : 20 ,
'#tools_bottom label .svg_icon' : 18
2010-03-16 19:28:57 +00:00
} ,
callback : function ( icons ) {
$ ( '.toolbar_button button > svg, .toolbar_button button > img' ) . each ( function ( ) {
$ ( this ) . parent ( ) . prepend ( this ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-16 19:28:57 +00:00
var tleft = $ ( '#tools_left' ) ;
2013-02-15 21:57:35 +00:00
if ( tleft . length !== 0 ) {
2010-07-14 19:15:44 +00:00
var min _height = tleft . offset ( ) . top + tleft . outerHeight ( ) ;
}
2010-07-27 13:37:52 +00:00
// var size = $.pref('iconsize');
2013-02-15 21:57:35 +00:00
// if (size && size != 'm') {
2012-10-22 10:47:50 +00:00
// svgEditor.setIconSize(size);
2013-02-15 21:57:35 +00:00
// } else if ($(window).height() < min_height) {
2010-07-27 13:37:52 +00:00
// // Make smaller
// svgEditor.setIconSize('s');
// }
2012-10-22 10:47:50 +00:00
2010-03-16 19:28:57 +00:00
// Look for any missing flyout icons from plugins
$ ( '.tools_flyout' ) . each ( function ( ) {
var shower = $ ( '#' + this . id + '_show' ) ;
var sel = shower . attr ( 'data-curopt' ) ;
// Check if there's an icon here
2013-02-15 21:57:35 +00:00
if ( ! shower . children ( 'svg, img' ) . length ) {
2010-03-16 19:28:57 +00:00
var clone = $ ( sel ) . children ( ) . clone ( ) ;
2013-02-15 21:57:35 +00:00
if ( clone . length ) {
2010-07-09 19:09:37 +00:00
clone [ 0 ] . removeAttribute ( 'style' ) ; //Needed for Opera
shower . append ( clone ) ;
}
2010-03-16 19:28:57 +00:00
}
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-16 19:28:57 +00:00
svgEditor . runCallbacks ( ) ;
2012-10-22 10:47:50 +00:00
2010-10-04 16:01:58 +00:00
setTimeout ( function ( ) {
$ ( '.flyout_arrow_horiz:empty' ) . each ( function ( ) {
$ ( this ) . append ( $ . getSvgIcon ( 'arrow_right' ) . width ( 5 ) . height ( 5 ) ) ;
} ) ;
} , 1 ) ;
2010-03-16 19:28:57 +00:00
}
} ) ;
2010-03-12 20:43:33 +00:00
2013-02-15 21:57:35 +00:00
Editor . canvas = svgCanvas = new $ . SvgCanvas ( document . getElementById ( 'svgcanvas' ) , curConfig ) ;
2013-02-16 15:19:20 +00:00
Editor . showSaveWarning = false ;
2013-02-15 21:57:35 +00:00
var palette = [
'#000000' , '#3f3f3f' , '#7f7f7f' , '#bfbfbf' , '#ffffff' ,
'#ff0000' , '#ff7f00' , '#ffff00' , '#7fff00' ,
'#00ff00' , '#00ff7f' , '#00ffff' , '#007fff' ,
'#0000ff' , '#7f00ff' , '#ff00ff' , '#ff007f' ,
'#7f0000' , '#7f3f00' , '#7f7f00' , '#3f7f00' ,
'#007f00' , '#007f3f' , '#007f7f' , '#003f7f' ,
'#00007f' , '#3f007f' , '#7f007f' , '#7f003f' ,
'#ffaaaa' , '#ffd4aa' , '#ffffaa' , '#d4ffaa' ,
'#aaffaa' , '#aaffd4' , '#aaffff' , '#aad4ff' ,
'#aaaaff' , '#d4aaff' , '#ffaaff' , '#ffaad4'
] ,
2013-02-18 18:22:00 +00:00
modKey = ( svgedit . browser . isMac ( ) ? 'meta+' : 'ctrl+' ) , // ⌘
2010-03-11 21:07:13 +00:00
path = svgCanvas . pathActions ,
2010-06-21 18:06:21 +00:00
undoMgr = svgCanvas . undoMgr ,
2010-11-05 18:29:39 +00:00
Utils = svgedit . utilities ,
2013-02-20 21:40:21 +00:00
defaultImageURL = curConfig . imgPath + 'logo.png' ,
2013-02-15 21:57:35 +00:00
workarea = $ ( '#workarea' ) ,
canv _menu = $ ( '#cmenu_canvas' ) ,
layer _menu = $ ( '#cmenu_layers' ) ,
2012-10-22 10:47:50 +00:00
exportWindow = null ,
2010-09-24 16:05:46 +00:00
tool _scale = 1 ,
2010-09-27 18:58:04 +00:00
zoomInIcon = 'crosshair' ,
zoomOutIcon = 'crosshair' ,
2010-10-12 15:20:17 +00:00
ui _context = 'toolbars' ,
2013-02-15 22:07:05 +00:00
origSource = '' ,
2010-10-12 18:35:45 +00:00
paintBox = { fill : null , stroke : null } ;
2010-03-11 21:07:13 +00:00
// This sets up alternative dialog boxes. They mostly work the same way as
// their UI counterparts, expect instead of returning the result, a callback
// needs to be included that returns the result as its first parameter.
2012-10-22 10:47:50 +00:00
// In the future we may want to add additional types of dialog boxes, since
2010-03-11 21:07:13 +00:00
// they should be easy to handle this way.
( function ( ) {
2013-02-15 21:57:35 +00:00
$ ( '#dialog_container' ) . draggable ( { cancel : '#dialog_content, #dialog_buttons *' , containment : 'window' } ) ;
var box = $ ( '#dialog_box' ) ,
btn _holder = $ ( '#dialog_buttons' ) ,
dbox = function ( type , msg , callback , defText ) {
2013-02-19 15:00:56 +00:00
$ ( '#dialog_content' ) . html ( '<p>' + msg . replace ( /\n/g , '</p><p>' ) + '</p>' )
. toggleClass ( 'prompt' , ( type == 'prompt' ) ) ;
btn _holder . empty ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-19 15:00:56 +00:00
var ok = $ ( '<input type="button" value="' + uiStrings . common . ok + '">' ) . appendTo ( btn _holder ) ;
2012-10-22 10:47:50 +00:00
2013-02-19 15:00:56 +00:00
if ( type != 'alert' ) {
$ ( '<input type="button" value="' + uiStrings . common . cancel + '">' )
. appendTo ( btn _holder )
. click ( function ( ) { box . hide ( ) ; callback ( false ) ; } ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-19 15:00:56 +00:00
if ( type == 'prompt' ) {
var input = $ ( '<input type="text">' ) . prependTo ( btn _holder ) ;
input . val ( defText || '' ) ;
input . bind ( 'keydown' , 'return' , function ( ) { ok . click ( ) ; } ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-19 15:00:56 +00:00
if ( type == 'process' ) {
ok . hide ( ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-19 15:00:56 +00:00
box . show ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-19 15:00:56 +00:00
ok . click ( function ( ) {
box . hide ( ) ;
var resp = ( type == 'prompt' ) ? input . val ( ) : true ;
if ( callback ) callback ( resp ) ;
} ) . focus ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-19 15:00:56 +00:00
if ( type == 'prompt' ) input . focus ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ . alert = function ( msg , cb ) { dbox ( 'alert' , msg , cb ) ; } ;
$ . confirm = function ( msg , cb ) { dbox ( 'confirm' , msg , cb ) ; } ;
2013-02-19 15:00:56 +00:00
$ . process _cancel = function ( msg , cb ) { dbox ( 'process' , msg , cb ) ; } ;
2010-03-11 21:07:13 +00:00
$ . prompt = function ( msg , txt , cb ) { dbox ( 'prompt' , msg , cb , txt ) ; } ;
} ( ) ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var setSelectMode = function ( ) {
2010-09-09 17:38:33 +00:00
var curr = $ ( '.tool_button_current' ) ;
2013-02-15 21:57:35 +00:00
if ( curr . length && curr [ 0 ] . id !== 'tool_select' ) {
2010-09-09 17:38:33 +00:00
curr . removeClass ( 'tool_button_current' ) . addClass ( 'tool_button' ) ;
$ ( '#tool_select' ) . addClass ( 'tool_button_current' ) . removeClass ( 'tool_button' ) ;
$ ( '#styleoverrides' ) . text ( '#svgcanvas svg *{cursor:move;pointer-events:all} #svgcanvas svg{cursor:default}' ) ;
}
2010-03-11 21:07:13 +00:00
svgCanvas . setMode ( 'select' ) ;
2013-02-15 21:57:35 +00:00
workarea . css ( 'cursor' , 'auto' ) ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var togglePathEditMode = function ( editmode , elems ) {
$ ( '#path_node_panel' ) . toggle ( editmode ) ;
$ ( '#tools_bottom_2,#tools_bottom_3' ) . toggle ( ! editmode ) ;
2013-02-15 21:57:35 +00:00
if ( editmode ) {
2010-03-11 21:07:13 +00:00
// Change select icon
$ ( '.tool_button_current' ) . removeClass ( 'tool_button_current' ) . addClass ( 'tool_button' ) ;
2010-05-12 15:45:43 +00:00
$ ( '#tool_select' ) . addClass ( 'tool_button_current' ) . removeClass ( 'tool_button' ) ;
setIcon ( '#tool_select' , 'select_node' ) ;
2010-03-11 21:07:13 +00:00
multiselected = false ;
2013-02-15 21:57:35 +00:00
if ( elems . length ) {
2010-03-11 21:07:13 +00:00
selectedElement = elems [ 0 ] ;
}
} else {
2010-05-12 15:45:43 +00:00
setIcon ( '#tool_select' , 'select' ) ;
2010-01-29 18:27:26 +00:00
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// used to make the flyouts stay on the screen longer the very first time
var flyoutspeed = 1250 ;
var textBeingEntered = false ;
var selectedElement = null ;
var multiselected = false ;
var editingsource = false ;
var docprops = false ;
2010-10-05 17:13:00 +00:00
var preferences = false ;
2010-09-23 19:53:43 +00:00
var cur _context = '' ;
2013-02-15 22:07:05 +00:00
var origTitle = $ ( 'title:first' ) . text ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
var saveHandler = function ( window , svg ) {
2013-02-16 15:19:20 +00:00
Editor . showSaveWarning = false ;
2012-10-22 10:47:50 +00:00
// by default, we add the XML prolog back, systems integrating SVG-edit (wikis, CMSs)
2010-03-11 21:07:13 +00:00
// can just provide their own custom save handler and might not want the XML prolog
2010-07-27 15:38:23 +00:00
svg = '<?xml version="1.0"?>\n' + svg ;
2012-10-22 10:47:50 +00:00
2013-02-16 15:36:19 +00:00
// IE9 doesn't allow standalone Data URLs
// https://connect.microsoft.com/IE/feedback/details/542600/data-uri-images-fail-when-loaded-by-themselves
if ( svgedit . browser . isIE ( ) ) {
2013-02-15 21:57:35 +00:00
showSourceEditor ( 0 , true ) ;
2012-10-22 10:47:50 +00:00
return ;
2010-08-23 20:16:27 +00:00
}
2013-02-16 15:36:19 +00:00
// Opens the SVG in new window
2013-02-15 21:57:35 +00:00
var win = window . open ( 'data:image/svg+xml;base64,' + Utils . encode64 ( svg ) ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Alert will only appear the first time saved OR the first time the bug is encountered
var done = $ . pref ( 'save_notice_done' ) ;
2013-02-15 21:57:35 +00:00
if ( done !== 'all' ) {
2011-01-03 18:45:36 +00:00
var note = uiStrings . notification . saveFromBrowser . replace ( '%s' , 'SVG' ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Check if FF and has <defs/>
2013-02-16 15:36:19 +00:00
if ( svgedit . browser . isGecko ( ) ) {
2013-02-15 21:57:35 +00:00
if ( svg . indexOf ( '<defs' ) !== - 1 ) {
2013-02-16 15:36:19 +00:00
// warning about Mozilla bug #308590 when applicable (seems to be fixed now in Feb 2013)
2013-02-15 21:57:35 +00:00
note += '\n\n' + uiStrings . notification . defsFailOnSave ;
2010-03-11 21:07:13 +00:00
$ . pref ( 'save_notice_done' , 'all' ) ;
2013-02-15 21:57:35 +00:00
done = 'all' ;
2010-03-11 21:07:13 +00:00
} else {
$ . pref ( 'save_notice_done' , 'part' ) ;
}
} else {
2012-10-22 10:47:50 +00:00
$ . pref ( 'save_notice_done' , 'all' ) ;
2010-02-16 21:07:13 +00:00
}
2013-02-15 21:57:35 +00:00
if ( done !== 'part' ) {
2010-03-11 21:07:13 +00:00
win . alert ( note ) ;
2010-02-16 21:07:13 +00:00
}
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-04-14 17:30:25 +00:00
var exportHandler = function ( window , data ) {
var issues = data . issues ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! $ ( '#export_canvas' ) . length ) {
2010-04-14 17:30:25 +00:00
$ ( '<canvas>' , { id : 'export_canvas' } ) . hide ( ) . appendTo ( 'body' ) ;
}
var c = $ ( '#export_canvas' ) [ 0 ] ;
2012-10-22 10:47:50 +00:00
2010-04-14 17:30:25 +00:00
c . width = svgCanvas . contentW ;
c . height = svgCanvas . contentH ;
2010-08-31 13:44:05 +00:00
canvg ( c , data . svg , { renderCallback : function ( ) {
2010-08-17 20:28:04 +00:00
var datauri = c . toDataURL ( 'image/png' ) ;
exportWindow . location . href = datauri ;
var done = $ . pref ( 'export_notice_done' ) ;
2013-02-15 21:57:35 +00:00
if ( done !== 'all' ) {
2011-01-03 18:45:36 +00:00
var note = uiStrings . notification . saveFromBrowser . replace ( '%s' , 'PNG' ) ;
2012-10-22 10:47:50 +00:00
2010-08-17 20:28:04 +00:00
// Check if there's issues
2013-02-15 21:57:35 +00:00
if ( issues . length ) {
var pre = '\n \u2022 ' ;
note += ( '\n\n' + uiStrings . notification . noteTheseIssues + pre + issues . join ( pre ) ) ;
2012-10-22 10:47:50 +00:00
}
2010-08-17 20:28:04 +00:00
// Note that this will also prevent the notice even though new issues may appear later.
// May want to find a way to deal with that without annoying the user
2012-10-22 10:47:50 +00:00
$ . pref ( 'export_notice_done' , 'all' ) ;
2010-08-17 20:28:04 +00:00
exportWindow . alert ( note ) ;
}
2010-08-31 13:44:05 +00:00
} } ) ;
2010-04-14 17:30:25 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// called when we've selected a different element
2013-02-15 21:57:35 +00:00
var selectedChanged = function ( window , elems ) {
2010-03-11 21:07:13 +00:00
var mode = svgCanvas . getMode ( ) ;
2013-02-15 21:57:35 +00:00
if ( mode === 'select' ) setSelectMode ( ) ;
2010-03-11 21:07:13 +00:00
var is _node = ( mode == "pathedit" ) ;
// if elems[1] is present, then we have more than one element
2013-02-15 21:57:35 +00:00
selectedElement = ( elems . length === 1 || elems [ 1 ] == null ? elems [ 0 ] : null ) ;
2010-03-11 21:07:13 +00:00
multiselected = ( elems . length >= 2 && elems [ 1 ] != null ) ;
if ( selectedElement != null ) {
// unless we're already in always set the mode of the editor to select because
// upon creation of a text element the editor is switched into
// select mode and this event fires - we need our UI to be in sync
2010-09-29 17:36:29 +00:00
2010-10-12 18:35:45 +00:00
if ( ! is _node ) {
2010-03-11 21:07:13 +00:00
updateToolbar ( ) ;
2012-10-22 10:47:50 +00:00
}
2010-03-11 21:07:13 +00:00
} // if (elem != null)
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Deal with pathedit mode
togglePathEditMode ( is _node , elems ) ;
updateContextPanel ( ) ;
2013-02-15 21:57:35 +00:00
svgCanvas . runExtensions ( 'selectedChanged' , {
2010-03-11 21:07:13 +00:00
elems : elems ,
selectedElement : selectedElement ,
multiselected : multiselected
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2011-01-27 14:23:33 +00:00
// Call when part of element is in process of changing, generally
// on mousemove actions like rotate, move, etc.
2013-02-15 23:05:23 +00:00
var elementTransition = function ( window , elems ) {
2011-01-27 14:23:33 +00:00
var mode = svgCanvas . getMode ( ) ;
var elem = elems [ 0 ] ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! elem ) return ;
2012-10-22 10:47:50 +00:00
2011-01-27 14:23:33 +00:00
multiselected = ( elems . length >= 2 && elems [ 1 ] != null ) ;
// Only updating fields for single elements for now
2013-02-15 21:57:35 +00:00
if ( ! multiselected ) {
switch ( mode ) {
case 'rotate' :
2011-01-27 14:23:33 +00:00
var ang = svgCanvas . getRotationAngle ( elem ) ;
$ ( '#angle' ) . val ( ang ) ;
2013-02-23 18:34:23 +00:00
$ ( '#tool_reorient' ) . toggleClass ( 'disabled' , ang === 0 ) ;
2011-01-27 14:23:33 +00:00
break ;
2012-10-22 10:47:50 +00:00
2011-01-28 16:10:43 +00:00
// TODO: Update values that change on move/resize, etc
// case "select":
// case "resize":
// break;
2011-01-27 14:23:33 +00:00
}
}
2013-02-15 21:57:35 +00:00
svgCanvas . runExtensions ( 'elementTransition' , {
2011-01-27 14:23:33 +00:00
elems : elems
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// called when any element has changed
2013-02-15 21:57:35 +00:00
var elementChanged = function ( window , elems ) {
2010-10-12 20:09:37 +00:00
var mode = svgCanvas . getMode ( ) ;
2013-02-15 21:57:35 +00:00
if ( mode === 'select' ) {
2010-09-09 17:38:33 +00:00
setSelectMode ( ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
for ( var i = 0 ; i < elems . length ; ++ i ) {
var elem = elems [ i ] ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// if the element changed was the svg, then it could be a resolution change
2013-02-15 21:57:35 +00:00
if ( elem && elem . tagName === 'svg' ) {
2010-03-11 21:07:13 +00:00
populateLayers ( ) ;
updateCanvas ( ) ;
2012-10-22 10:47:50 +00:00
}
2010-03-11 21:07:13 +00:00
// Update selectedElement if element is no longer part of the image.
// This occurs for the text elements in Firefox
2013-02-15 21:57:35 +00:00
else if ( elem && selectedElement && selectedElement . parentNode == null ) {
2010-07-21 12:48:51 +00:00
// || elem && elem.tagName == "path" && !multiselected) { // This was added in r1430, but not sure why
2010-03-11 21:07:13 +00:00
selectedElement = elem ;
2010-01-29 18:27:26 +00:00
}
}
2012-10-22 10:47:50 +00:00
2013-02-16 15:19:20 +00:00
Editor . showSaveWarning = true ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// we update the contextual panel with potentially new
// positional/sizing information (we DON'T want to update the
// toolbar here as that creates an infinite loop)
// also this updates the history buttons
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// we tell it to skip focusing the text control if the
// text element was previously in focus
updateContextPanel ( ) ;
2012-10-22 10:47:50 +00:00
2010-10-12 20:09:37 +00:00
// In the event a gradient was flipped:
2013-02-15 21:57:35 +00:00
if ( selectedElement && mode === 'select' ) {
2010-10-12 20:09:37 +00:00
paintBox . fill . update ( ) ;
paintBox . stroke . update ( ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
svgCanvas . runExtensions ( 'elementChanged' , {
2010-03-11 21:07:13 +00:00
elems : elems
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2012-11-28 04:32:59 +00:00
var zoomChanged = svgCanvas . zoomChanged = function ( window , bbox , autoCenter ) {
2010-03-11 21:07:13 +00:00
var scrbar = 15 ,
res = svgCanvas . getResolution ( ) ,
w _area = workarea ,
canvas _pos = $ ( '#svgcanvas' ) . position ( ) ;
var z _info = svgCanvas . setBBoxZoom ( bbox , w _area . width ( ) - scrbar , w _area . height ( ) - scrbar ) ;
2013-02-15 21:57:35 +00:00
if ( ! z _info ) return ;
2010-03-11 21:07:13 +00:00
var zoomlevel = z _info . zoom ,
bb = z _info . bbox ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( zoomlevel < 0.001 ) {
changeZoom ( { value : 0.1 } ) ;
2010-10-21 20:30:42 +00:00
return ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 13:38:55 +00:00
$ ( '#zoom' ) . val ( ( zoomlevel * 100 ) . toFixed ( 1 ) ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( autoCenter ) {
2010-03-11 21:07:13 +00:00
updateCanvas ( ) ;
} else {
updateCanvas ( false , { x : bb . x * zoomlevel + ( bb . width * zoomlevel ) / 2 , y : bb . y * zoomlevel + ( bb . height * zoomlevel ) / 2 } ) ;
2010-01-25 21:22:03 +00:00
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( svgCanvas . getMode ( ) == 'zoom' && bb . width ) {
2010-03-11 21:07:13 +00:00
// Go to select if a zoom box was drawn
setSelectMode ( ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
zoomDone ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-09-23 19:53:43 +00:00
$ ( '#cur_context_panel' ) . delegate ( 'a' , 'click' , function ( ) {
var link = $ ( this ) ;
2013-02-15 21:57:35 +00:00
if ( link . attr ( 'data-root' ) ) {
2010-09-23 19:53:43 +00:00
svgCanvas . leaveContext ( ) ;
} else {
svgCanvas . setContext ( link . text ( ) ) ;
}
2012-11-28 05:02:46 +00:00
svgCanvas . clearSelection ( ) ;
2010-09-23 19:53:43 +00:00
return false ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-09-23 19:53:43 +00:00
var contextChanged = function ( win , context ) {
var link _str = '' ;
2013-02-15 21:57:35 +00:00
if ( context ) {
2010-09-23 19:53:43 +00:00
var str = '' ;
2011-01-30 17:11:56 +00:00
link _str = '<a href="#" data-root="y">' + svgCanvas . getCurrentDrawing ( ) . getCurrentLayerName ( ) + '</a>' ;
2012-10-22 10:47:50 +00:00
2010-09-23 19:53:43 +00:00
$ ( context ) . parentsUntil ( '#svgcontent > g' ) . andSelf ( ) . each ( function ( ) {
2013-02-15 21:57:35 +00:00
if ( this . id ) {
2010-09-23 19:53:43 +00:00
str += ' > ' + this . id ;
2013-02-15 21:57:35 +00:00
if ( this !== context ) {
2010-09-23 19:53:43 +00:00
link _str += ' > <a href="#">' + this . id + '</a>' ;
} else {
link _str += ' > ' + this . id ;
}
}
} ) ;
cur _context = str ;
} else {
cur _context = null ;
}
$ ( '#cur_context_panel' ) . toggle ( ! ! context ) . html ( link _str ) ;
updateTitle ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-10-12 15:20:17 +00:00
// Makes sure the current selected paint is available to work with
var prepPaints = function ( ) {
2010-10-12 18:35:45 +00:00
paintBox . fill . prep ( ) ;
paintBox . stroke . prep ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var flyout _funcs = { } ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var setupFlyouts = function ( holders ) {
$ . each ( holders , function ( hold _sel , btn _opts ) {
var buttons = $ ( hold _sel ) . children ( ) ;
var show _sel = hold _sel + '_show' ;
2010-07-09 19:09:37 +00:00
var shower = $ ( show _sel ) ;
2010-03-11 21:07:13 +00:00
var def = false ;
buttons . addClass ( 'tool_button' )
. unbind ( 'click mousedown mouseup' ) // may not be necessary
. each ( function ( i ) {
// Get this buttons options
var opts = btn _opts [ i ] ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Remember the function that goes with this ID
flyout _funcs [ opts . sel ] = opts . fn ;
2012-03-22 10:31:25 +00:00
2013-02-15 21:57:35 +00:00
if ( opts . isDefault ) def = i ;
2012-03-22 10:31:25 +00:00
2010-03-11 21:07:13 +00:00
// Clicking the icon in flyout should set this set's icon
2012-03-22 10:31:25 +00:00
var func = function ( event ) {
var options = opts ;
//find the currently selected tool if comes from keystroke
2013-02-15 21:57:35 +00:00
if ( event . type === 'keydown' ) {
var flyoutIsSelected = $ ( options . parent + '_show' ) . hasClass ( 'tool_button_current' ) ;
var currentOperation = $ ( options . parent + '_show' ) . attr ( 'data-curopt' ) ;
$ . each ( holders [ opts . parent ] , function ( i , tool ) {
2012-03-22 10:31:25 +00:00
if ( tool . sel == currentOperation ) {
2013-02-15 21:57:35 +00:00
if ( ! event . shiftKey || ! flyoutIsSelected ) {
2012-03-22 10:31:25 +00:00
options = tool ;
2013-02-15 21:57:35 +00:00
} else {
2012-03-22 10:31:25 +00:00
options = holders [ opts . parent ] [ i + 1 ] || holders [ opts . parent ] [ 0 ] ;
}
}
} ) ;
}
2013-02-15 21:57:35 +00:00
if ( $ ( this ) . hasClass ( 'disabled' ) ) return false ;
2010-03-11 21:07:13 +00:00
if ( toolButtonClick ( show _sel ) ) {
2012-03-22 10:31:25 +00:00
options . fn ( ) ;
2010-03-11 21:07:13 +00:00
}
2013-02-15 21:57:35 +00:00
if ( options . icon ) {
2012-03-22 10:31:25 +00:00
var icon = $ . getSvgIcon ( options . icon , true ) ;
2010-03-11 21:07:13 +00:00
} else {
2012-03-22 10:31:25 +00:00
var icon = $ ( options . sel ) . children ( ) . eq ( 0 ) . clone ( ) ;
2010-03-11 21:07:13 +00:00
}
2012-03-22 10:31:25 +00:00
2013-02-15 21:57:35 +00:00
icon [ 0 ] . setAttribute ( 'width' , shower . width ( ) ) ;
icon [ 0 ] . setAttribute ( 'height' , shower . height ( ) ) ;
2010-03-11 21:07:13 +00:00
shower . children ( ':not(.flyout_arrow_horiz)' ) . remove ( ) ;
2012-03-22 10:31:25 +00:00
shower . append ( icon ) . attr ( 'data-curopt' , options . sel ) ; // This sets the current mode
2013-02-15 13:38:55 +00:00
} ;
2012-03-22 10:31:25 +00:00
2010-03-11 21:07:13 +00:00
$ ( this ) . mouseup ( func ) ;
2012-03-22 10:31:25 +00:00
2013-02-15 21:57:35 +00:00
if ( opts . key ) {
$ ( document ) . bind ( 'keydown' , opts . key [ 0 ] + ' shift+' + opts . key [ 0 ] , func ) ;
2010-01-27 21:03:43 +00:00
}
2010-03-11 21:07:13 +00:00
} ) ;
2012-03-22 10:31:25 +00:00
2013-02-15 21:57:35 +00:00
if ( def ) {
2010-07-09 19:09:37 +00:00
shower . attr ( 'data-curopt' , btn _opts [ def ] . sel ) ;
2013-02-15 21:57:35 +00:00
} else if ( ! shower . attr ( 'data-curopt' ) ) {
2010-03-11 21:07:13 +00:00
// Set first as default
2010-07-09 19:09:37 +00:00
shower . attr ( 'data-curopt' , btn _opts [ 0 ] . sel ) ;
2010-01-27 21:03:43 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var timer ;
2011-09-24 23:38:47 +00:00
var pos = $ ( show _sel ) . position ( ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Clicking the "show" icon should set the current mode
2010-07-09 19:09:37 +00:00
shower . mousedown ( function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( shower . hasClass ( 'disabled' ) ) return false ;
2011-09-24 23:38:47 +00:00
var holder = $ ( hold _sel ) ;
2013-02-15 21:57:35 +00:00
var l = pos . left + 34 ;
var w = holder . width ( ) * - 1 ;
var time = holder . data ( 'shown_popop' ) ? 200 : 0 ;
2010-03-11 21:07:13 +00:00
timer = setTimeout ( function ( ) {
// Show corresponding menu
2013-02-15 21:57:35 +00:00
if ( ! shower . data ( 'isLibrary' ) ) {
2010-07-09 19:09:37 +00:00
holder . css ( 'left' , w ) . show ( ) . animate ( {
left : l
2013-02-15 21:57:35 +00:00
} , 150 ) ;
2010-07-09 19:09:37 +00:00
} else {
holder . css ( 'left' , l ) . show ( ) ;
}
2013-02-15 21:57:35 +00:00
holder . data ( 'shown_popop' , true ) ;
2010-03-11 21:07:13 +00:00
} , time ) ;
evt . preventDefault ( ) ;
2010-07-09 19:09:37 +00:00
} ) . mouseup ( function ( evt ) {
2010-03-11 21:07:13 +00:00
clearTimeout ( timer ) ;
var opt = $ ( this ) . attr ( 'data-curopt' ) ;
2010-07-09 19:09:37 +00:00
// Is library and popped up, so do nothing
2013-02-15 22:32:35 +00:00
if ( shower . data ( 'isLibrary' ) && $ ( show _sel . replace ( '_show' , '' ) ) . is ( ':visible' ) ) {
2010-07-09 19:09:37 +00:00
toolButtonClick ( show _sel , true ) ;
return ;
}
if ( toolButtonClick ( show _sel ) && ( opt in flyout _funcs ) ) {
2010-03-11 21:07:13 +00:00
flyout _funcs [ opt ] ( ) ;
}
} ) ;
2013-02-15 23:05:23 +00:00
// $('#tools_rect').mouseleave(function(){$('#tools_rect').fadeOut();});
2010-02-16 21:07:13 +00:00
} ) ;
2010-03-11 21:07:13 +00:00
setFlyoutTitles ( ) ;
2013-10-07 01:35:01 +00:00
setFlyoutPositions ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var makeFlyoutHolder = function ( id , child ) {
2013-02-15 21:57:35 +00:00
var div = $ ( '<div>' , {
2010-03-11 21:07:13 +00:00
'class' : 'tools_flyout' ,
id : id
} ) . appendTo ( '#svg_editor' ) . append ( child ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
return div ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var setFlyoutPositions = function ( ) {
$ ( '.tools_flyout' ) . each ( function ( ) {
var shower = $ ( '#' + this . id + '_show' ) ;
var pos = shower . offset ( ) ;
var w = shower . outerWidth ( ) ;
2010-07-27 13:37:52 +00:00
$ ( this ) . css ( { left : ( pos . left + w ) * tool _scale , top : pos . top } ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var setFlyoutTitles = function ( ) {
$ ( '.tools_flyout' ) . each ( function ( ) {
var shower = $ ( '#' + this . id + '_show' ) ;
2013-02-15 21:57:35 +00:00
if ( shower . data ( 'isLibrary' ) ) return ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var tooltips = [ ] ;
$ ( this ) . children ( ) . each ( function ( ) {
tooltips . push ( this . title ) ;
} ) ;
shower [ 0 ] . title = tooltips . join ( ' / ' ) ;
} ) ;
2013-02-15 13:38:55 +00:00
} ;
2010-07-27 13:37:52 +00:00
2012-10-22 10:47:50 +00:00
var resize _timer ;
2010-03-11 21:07:13 +00:00
var extAdded = function ( window , ext ) {
var cb _called = false ;
2010-07-27 13:37:52 +00:00
var resize _done = false ;
2010-07-26 17:48:02 +00:00
var cb _ready = true ; // Set to false to delay callback (e.g. wait for $.svgIcons)
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
function prepResize ( ) {
2013-02-15 21:57:35 +00:00
if ( resize _timer ) {
2010-07-27 13:37:52 +00:00
clearTimeout ( resize _timer ) ;
resize _timer = null ;
}
2013-02-15 21:57:35 +00:00
if ( ! resize _done ) {
2010-07-27 13:37:52 +00:00
resize _timer = setTimeout ( function ( ) {
resize _done = true ;
setIconSize ( curPrefs . iconsize ) ;
2012-10-22 10:47:50 +00:00
} , 50 ) ;
2010-07-27 13:37:52 +00:00
}
}
2010-03-11 21:07:13 +00:00
var runCallback = function ( ) {
2013-02-15 21:57:35 +00:00
if ( ext . callback && ! cb _called && cb _ready ) {
2010-03-11 21:07:13 +00:00
cb _called = true ;
ext . callback ( ) ;
2009-10-12 20:26:03 +00:00
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-05-27 13:27:04 +00:00
var btn _selects = [ ] ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ext . context _tools ) {
2010-03-11 21:07:13 +00:00
$ . each ( ext . context _tools , function ( i , tool ) {
// Add select tool
2013-02-16 10:28:17 +00:00
var cont _id = tool . container _id ? ( ' id="' + tool . container _id + '"' ) : '' ;
2010-03-11 21:07:13 +00:00
var panel = $ ( '#' + tool . panel ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// create the panel if it doesn't exist
2013-02-15 21:57:35 +00:00
if ( ! panel . length )
panel = $ ( '<div>' , { id : tool . panel } ) . appendTo ( '#tools_top' ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// TODO: Allow support for other types, or adding to existing tool
switch ( tool . type ) {
case 'tool_button' :
var html = '<div class="tool_button">' + tool . id + '</div>' ;
var div = $ ( html ) . appendTo ( panel ) ;
if ( tool . events ) {
$ . each ( tool . events , function ( evt , func ) {
$ ( div ) . bind ( evt , func ) ;
} ) ;
}
break ;
case 'select' :
var html = '<label' + cont _id + '>'
+ '<select id="' + tool . id + '">' ;
$ . each ( tool . options , function ( val , text ) {
var sel = ( val == tool . defval ) ? " selected" : "" ;
html += '<option value="' + val + '"' + sel + '>' + text + '</option>' ;
} ) ;
html += "</select></label>" ;
// Creates the tool, hides & adds it, returns the select element
var sel = $ ( html ) . appendTo ( panel ) . find ( 'select' ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ . each ( tool . events , function ( evt , func ) {
$ ( sel ) . bind ( evt , func ) ;
} ) ;
break ;
2012-10-22 10:47:50 +00:00
case 'button-select' :
2010-06-03 07:30:56 +00:00
var html = '<div id="' + tool . id + '" class="dropdown toolset" title="' + tool . title + '">'
2010-05-27 13:27:04 +00:00
+ '<div id="cur_' + tool . id + '" class="icon_label"></div><button></button></div>' ;
2012-10-22 10:47:50 +00:00
2010-05-27 13:27:04 +00:00
var list = $ ( '<ul id="' + tool . id + '_opts"></ul>' ) . appendTo ( '#option_lists' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( tool . colnum ) {
2010-06-03 13:05:48 +00:00
list . addClass ( 'optcols' + tool . colnum ) ;
}
2012-10-22 10:47:50 +00:00
2010-05-27 13:27:04 +00:00
// Creates the tool, hides & adds it, returns the select element
var dropdown = $ ( html ) . appendTo ( panel ) . children ( ) ;
2012-10-22 10:47:50 +00:00
2010-05-27 13:27:04 +00:00
btn _selects . push ( {
elem : ( '#' + tool . id ) ,
list : ( '#' + tool . id + '_opts' ) ,
2010-06-03 07:30:56 +00:00
title : tool . title ,
2010-05-27 13:27:04 +00:00
callback : tool . events . change ,
cur : ( '#cur_' + tool . id )
} ) ;
break ;
2010-03-11 21:07:13 +00:00
case 'input' :
var html = '<label' + cont _id + '>'
2012-10-22 10:47:50 +00:00
+ '<span id="' + tool . id + '_label">'
2010-03-11 21:07:13 +00:00
+ tool . label + ':</span>'
+ '<input id="' + tool . id + '" title="' + tool . title
2013-02-15 13:38:55 +00:00
+ '" size="' + ( tool . size || "4" ) + '" value="' + ( tool . defval || "" ) + '" type="text"/></label>' ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Creates the tool, hides & adds it, returns the select element
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Add to given tool.panel
var inp = $ ( html ) . appendTo ( panel ) . find ( 'input' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( tool . spindata ) {
2010-03-11 21:07:13 +00:00
inp . SpinButton ( tool . spindata ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( tool . events ) {
2010-03-11 21:07:13 +00:00
$ . each ( tool . events , function ( evt , func ) {
inp . bind ( evt , func ) ;
} ) ;
}
break ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
default :
break ;
}
} ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ext . buttons ) {
2010-03-11 21:07:13 +00:00
var fallback _obj = { } ,
placement _obj = { } ,
2013-02-15 13:38:55 +00:00
svgicons = ext . svgicons ,
holders = { } ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Add buttons given by extension
$ . each ( ext . buttons , function ( i , btn ) {
var icon ;
var id = btn . id ;
var num = i ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Give button a unique ID
while ( $ ( '#' + id ) . length ) {
id = btn . id + '_' + ( ++ num ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! svgicons ) {
2010-03-11 21:07:13 +00:00
icon = $ ( '<img src="' + btn . icon + '">' ) ;
} else {
fallback _obj [ id ] = btn . icon ;
2010-06-03 07:30:56 +00:00
var svgicon = btn . svgicon ? btn . svgicon : btn . id ;
2013-02-15 21:57:35 +00:00
if ( btn . type == 'app_menu' ) {
2010-07-14 18:05:03 +00:00
placement _obj [ '#' + id + ' > div' ] = svgicon ;
} else {
placement _obj [ '#' + id ] = svgicon ;
}
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var cls , parent ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Set button up according to its type
switch ( btn . type ) {
2010-07-09 19:09:37 +00:00
case 'mode_flyout' :
2010-03-11 21:07:13 +00:00
case 'mode' :
cls = 'tool_button' ;
2013-02-15 21:57:35 +00:00
parent = '#tools_left' ;
2010-03-11 21:07:13 +00:00
break ;
case 'context' :
cls = 'tool_button' ;
2013-02-15 21:57:35 +00:00
parent = '#' + btn . panel ;
2010-03-11 21:07:13 +00:00
// create the panel if it doesn't exist
2013-02-15 21:57:35 +00:00
if ( ! $ ( parent ) . length )
$ ( '<div>' , { id : btn . panel } ) . appendTo ( '#tools_top' ) ;
2010-03-11 21:07:13 +00:00
break ;
2010-07-14 18:05:03 +00:00
case 'app_menu' :
cls = '' ;
parent = '#main_menu ul' ;
break ;
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
var button = $ ( ( btn . list || btn . type == 'app_menu' ) ? '<li/>' : '<div/>' )
. attr ( 'id' , id )
. attr ( 'title' , btn . title )
2010-03-11 21:07:13 +00:00
. addClass ( cls ) ;
2013-02-15 21:57:35 +00:00
if ( ! btn . includeWith && ! btn . list ) {
if ( 'position' in btn ) {
2010-07-09 19:09:37 +00:00
$ ( parent ) . children ( ) . eq ( btn . position ) . before ( button ) ;
} else {
button . appendTo ( parent ) ;
}
2010-07-14 18:05:03 +00:00
2013-02-15 21:57:35 +00:00
if ( btn . type == 'mode_flyout' ) {
2010-07-09 19:09:37 +00:00
// Add to flyout menu / make flyout menu
// var opts = btn.includeWith;
// // opts.button, default, position
var ref _btn = $ ( button ) ;
2012-10-22 10:47:50 +00:00
2010-07-09 19:09:37 +00:00
var flyout _holder = ref _btn . parent ( ) ;
// Create a flyout menu if there isn't one already
2013-02-15 21:57:35 +00:00
if ( ! ref _btn . parent ( ) . hasClass ( 'tools_flyout' ) ) {
2010-07-09 19:09:37 +00:00
// Create flyout placeholder
2013-02-15 22:32:35 +00:00
var tls _id = ref _btn [ 0 ] . id . replace ( 'tool_' , 'tools_' ) ;
2010-07-09 19:09:37 +00:00
var show _btn = ref _btn . clone ( )
. attr ( 'id' , tls _id + '_show' )
2013-02-15 21:57:35 +00:00
. append ( $ ( '<div>' , { 'class' : 'flyout_arrow_horiz' } ) ) ;
2012-10-22 10:47:50 +00:00
2010-07-09 19:09:37 +00:00
ref _btn . before ( show _btn ) ;
2012-10-22 10:47:50 +00:00
2010-07-09 19:09:37 +00:00
// Create a flyout div
flyout _holder = makeFlyoutHolder ( tls _id , ref _btn ) ;
flyout _holder . data ( 'isLibrary' , true ) ;
show _btn . data ( 'isLibrary' , true ) ;
2012-10-22 10:47:50 +00:00
}
2010-07-09 19:09:37 +00:00
// var ref_data = Actions.getButtonData(opts.button);
2012-10-22 10:47:50 +00:00
2010-07-09 19:09:37 +00:00
placement _obj [ '#' + tls _id + '_show' ] = btn . id ;
// TODO: Find way to set the current icon using the iconloader if this is not default
2012-10-22 10:47:50 +00:00
2010-07-09 19:09:37 +00:00
// Include data for extension button as well as ref button
var cur _h = holders [ '#' + flyout _holder [ 0 ] . id ] = [ {
sel : '#' + id ,
fn : btn . events . click ,
icon : btn . id ,
// key: btn.key,
isDefault : true
} , ref _data ] ;
2012-10-22 10:47:50 +00:00
//
2010-07-09 19:09:37 +00:00
// // {sel:'#tool_rect', fn: clickRect, evt: 'mouseup', key: 4, parent: '#tools_rect', icon: 'rect'}
2012-10-22 10:47:50 +00:00
//
2013-10-13 23:23:08 +00:00
// var pos = ('position' in opts)?opts.position:'last';
2010-07-09 19:09:37 +00:00
// var len = flyout_holder.children().length;
2012-10-22 10:47:50 +00:00
//
2010-07-09 19:09:37 +00:00
// // Add at given position or end
2013-02-15 21:57:35 +00:00
// if (!isNaN(pos) && pos >= 0 && pos < len) {
2010-07-09 19:09:37 +00:00
// flyout_holder.children().eq(pos).before(button);
// } else {
// flyout_holder.append(button);
// cur_h.reverse();
// }
2013-02-15 21:57:35 +00:00
} else if ( btn . type == 'app_menu' ) {
2010-07-14 18:05:03 +00:00
button . append ( '<div>' ) . append ( btn . title ) ;
2010-07-09 19:09:37 +00:00
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
} else if ( btn . list ) {
2010-05-27 13:27:04 +00:00
// Add button to list
button . addClass ( 'push_button' ) ;
$ ( '#' + btn . list + '_opts' ) . append ( button ) ;
2013-02-15 21:57:35 +00:00
if ( btn . isDefault ) {
2010-05-27 13:27:04 +00:00
$ ( '#cur_' + btn . list ) . append ( button . children ( ) . clone ( ) ) ;
2010-06-03 07:30:56 +00:00
var svgicon = btn . svgicon ? btn . svgicon : btn . id ;
placement _obj [ '#cur_' + btn . list ] = svgicon ;
2010-05-27 13:27:04 +00:00
}
2013-02-15 21:57:35 +00:00
} else if ( btn . includeWith ) {
2010-03-11 21:07:13 +00:00
// Add to flyout menu / make flyout menu
var opts = btn . includeWith ;
// opts.button, default, position
var ref _btn = $ ( opts . button ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var flyout _holder = ref _btn . parent ( ) ;
// Create a flyout menu if there isn't one already
2013-02-15 21:57:35 +00:00
if ( ! ref _btn . parent ( ) . hasClass ( 'tools_flyout' ) ) {
2010-03-11 21:07:13 +00:00
// Create flyout placeholder
2013-02-15 21:57:35 +00:00
var tls _id = ref _btn [ 0 ] . id . replace ( 'tool_' , 'tools_' ) ;
2010-03-11 21:07:13 +00:00
var show _btn = ref _btn . clone ( )
. attr ( 'id' , tls _id + '_show' )
2013-02-15 21:57:35 +00:00
. append ( $ ( '<div>' , { 'class' : 'flyout_arrow_horiz' } ) ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
ref _btn . before ( show _btn ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Create a flyout div
flyout _holder = makeFlyoutHolder ( tls _id , ref _btn ) ;
2012-10-22 10:47:50 +00:00
}
2010-03-11 21:07:13 +00:00
var ref _data = Actions . getButtonData ( opts . button ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( opts . isDefault ) {
2010-03-11 21:07:13 +00:00
placement _obj [ '#' + tls _id + '_show' ] = btn . id ;
2012-10-22 10:47:50 +00:00
}
2010-03-11 21:07:13 +00:00
// TODO: Find way to set the current icon using the iconloader if this is not default
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Include data for extension button as well as ref button
var cur _h = holders [ '#' + flyout _holder [ 0 ] . id ] = [ {
sel : '#' + id ,
fn : btn . events . click ,
icon : btn . id ,
key : btn . key ,
isDefault : btn . includeWith ? btn . includeWith . isDefault : 0
} , ref _data ] ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// {sel:'#tool_rect', fn: clickRect, evt: 'mouseup', key: 4, parent: '#tools_rect', icon: 'rect'}
2012-10-22 10:47:50 +00:00
2013-10-13 23:23:08 +00:00
var pos = ( 'position' in opts ) ? opts . position : 'last' ;
2010-03-11 21:07:13 +00:00
var len = flyout _holder . children ( ) . length ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Add at given position or end
2013-02-15 21:57:35 +00:00
if ( ! isNaN ( pos ) && pos >= 0 && pos < len ) {
2010-03-11 21:07:13 +00:00
flyout _holder . children ( ) . eq ( pos ) . before ( button ) ;
} else {
flyout _holder . append ( button ) ;
cur _h . reverse ( ) ;
}
2012-10-22 10:47:50 +00:00
}
2013-02-15 21:57:35 +00:00
if ( ! svgicons ) {
2010-03-11 21:07:13 +00:00
button . append ( icon ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! btn . list ) {
2010-05-27 13:27:04 +00:00
// Add given events to button
$ . each ( btn . events , function ( name , func ) {
2013-02-20 21:45:10 +00:00
if ( name == 'click' && btn . type == 'mode' ) {
if ( btn . includeWith ) {
2010-05-27 13:27:04 +00:00
button . bind ( name , func ) ;
2013-02-20 21:45:10 +00:00
} else {
button . bind ( name , function ( ) {
if ( toolButtonClick ( button ) ) {
func ( ) ;
}
} ) ;
}
if ( btn . key ) {
$ ( document ) . bind ( 'keydown' , btn . key , func ) ;
if ( btn . title ) button . attr ( 'title' , btn . title + ' [' + btn . key + ']' ) ;
2010-03-11 21:07:13 +00:00
}
} else {
button . bind ( name , func ) ;
}
2010-05-27 13:27:04 +00:00
} ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
setupFlyouts ( holders ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-05-27 13:27:04 +00:00
$ . each ( btn _selects , function ( ) {
2012-10-22 10:47:50 +00:00
addAltDropDown ( this . elem , this . list , this . callback , { seticon : true } ) ;
2010-05-27 13:27:04 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-07-26 17:48:02 +00:00
if ( svgicons )
cb _ready = false ; // Delay callback
2010-05-27 13:27:04 +00:00
2010-03-11 21:07:13 +00:00
$ . svgIcons ( svgicons , {
w : 24 , h : 24 ,
id _match : false ,
2013-02-18 18:22:00 +00:00
no _img : ( ! svgedit . browser . isWebkit ( ) ) ,
2010-03-11 21:07:13 +00:00
fallback : fallback _obj ,
placement : placement _obj ,
callback : function ( icons ) {
2010-03-24 20:13:13 +00:00
// Non-ideal hack to make the icon match the current size
2013-02-15 21:57:35 +00:00
if ( curPrefs . iconsize && curPrefs . iconsize != 'm' ) {
2010-07-27 13:37:52 +00:00
prepResize ( ) ;
2010-03-11 21:07:13 +00:00
}
2010-07-26 17:48:02 +00:00
cb _ready = true ; // Ready for callback
2010-03-11 21:07:13 +00:00
runCallback ( ) ;
}
} ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
runCallback ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-10-12 15:20:17 +00:00
var getPaint = function ( color , opac , type ) {
2010-03-11 21:07:13 +00:00
// update the editor's fill paint
2013-02-19 11:13:03 +00:00
var opts = { alpha : opac } ;
2013-02-15 21:57:35 +00:00
if ( color . indexOf ( 'url(#' ) === 0 ) {
2010-10-12 15:20:17 +00:00
var refElem = svgCanvas . getRefElem ( color ) ;
2013-02-15 21:57:35 +00:00
if ( refElem ) {
2010-10-12 15:20:17 +00:00
refElem = refElem . cloneNode ( true ) ;
} else {
2013-02-19 11:13:03 +00:00
refElem = $ ( '#' + type + '_color defs *' ) [ 0 ] ;
2010-10-12 15:20:17 +00:00
}
opts [ refElem . tagName ] = refElem ;
2013-02-15 21:57:35 +00:00
} else if ( color . indexOf ( '#' ) === 0 ) {
2013-02-19 11:13:03 +00:00
opts . solidColor = color . substr ( 1 ) ;
2013-02-15 21:57:35 +00:00
} else {
2013-02-19 11:13:03 +00:00
opts . solidColor = 'none' ;
2010-03-11 21:07:13 +00:00
}
return new $ . jGraduate . Paint ( opts ) ;
2012-10-22 10:47:50 +00:00
} ;
2010-03-11 21:07:13 +00:00
// updates the toolbar (colors, opacity, etc) based on the selected element
// This function also updates the opacity and id elements that are in the context panel
var updateToolbar = function ( ) {
2010-12-20 21:37:04 +00:00
if ( selectedElement != null ) {
2013-02-15 23:17:56 +00:00
switch ( selectedElement . tagName ) {
2010-12-20 21:37:04 +00:00
case 'use' :
case 'image' :
case 'foreignObject' :
break ;
case 'g' :
case 'a' :
// Look for common styles
2010-12-21 14:34:54 +00:00
var gWidth = null ;
var childs = selectedElement . getElementsByTagName ( '*' ) ;
2013-02-15 21:57:35 +00:00
for ( var i = 0 , len = childs . length ; i < len ; i ++ ) {
var swidth = childs [ i ] . getAttribute ( 'stroke-width' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( i === 0 ) {
2010-12-21 14:34:54 +00:00
gWidth = swidth ;
2013-02-15 21:57:35 +00:00
} else if ( gWidth !== swidth ) {
2010-12-21 14:34:54 +00:00
gWidth = null ;
}
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#stroke_width' ) . val ( gWidth === null ? '' : gWidth ) ;
2012-10-22 10:47:50 +00:00
2010-12-20 21:37:04 +00:00
paintBox . fill . update ( true ) ;
paintBox . stroke . update ( true ) ;
2010-04-06 15:53:21 +00:00
2010-12-20 21:37:04 +00:00
break ;
default :
paintBox . fill . update ( true ) ;
paintBox . stroke . update ( true ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#stroke_width' ) . val ( selectedElement . getAttribute ( 'stroke-width' ) || 1 ) ;
$ ( '#stroke_style' ) . val ( selectedElement . getAttribute ( 'stroke-dasharray' ) || 'none' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
var attr = selectedElement . getAttribute ( 'stroke-linejoin' ) || 'miter' ;
2012-10-22 10:47:50 +00:00
2010-12-20 21:37:04 +00:00
if ( $ ( '#linejoin_' + attr ) . length != 0 )
setStrokeOpt ( $ ( '#linejoin_' + attr ) [ 0 ] ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
attr = selectedElement . getAttribute ( 'stroke-linecap' ) || 'butt' ;
2012-10-22 10:47:50 +00:00
2010-12-20 21:37:04 +00:00
if ( $ ( '#linecap_' + attr ) . length != 0 )
setStrokeOpt ( $ ( '#linecap_' + attr ) [ 0 ] ) ;
}
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// All elements including image and group have opacity
2013-02-15 21:57:35 +00:00
if ( selectedElement != null ) {
var opac _perc = ( ( selectedElement . getAttribute ( 'opacity' ) || 1.0 ) * 100 ) ;
2010-03-11 21:07:13 +00:00
$ ( '#group_opacity' ) . val ( opac _perc ) ;
$ ( '#opac_slider' ) . slider ( 'option' , 'value' , opac _perc ) ;
$ ( '#elem_id' ) . val ( selectedElement . id ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
updateToolButtonState ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-07-16 18:33:48 +00:00
var setImageURL = Editor . setImageURL = function ( url ) {
2013-02-20 21:40:21 +00:00
if ( ! url ) url = defaultImageURL ;
2010-07-16 18:33:48 +00:00
svgCanvas . setImageURL ( url ) ;
$ ( '#image_url' ) . val ( url ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( url . indexOf ( 'data:' ) === 0 ) {
2010-07-16 18:33:48 +00:00
// data URI found
$ ( '#image_url' ) . hide ( ) ;
$ ( '#change_image_url' ) . show ( ) ;
} else {
// regular URL
2013-02-20 21:40:21 +00:00
svgCanvas . embedImage ( url , function ( dataURI ) {
// Couldn't embed, so show warning
$ ( '#url_notice' ) . toggle ( ! dataURI ) ;
defaultImageURL = url ;
2010-07-16 18:33:48 +00:00
} ) ;
$ ( '#image_url' ) . show ( ) ;
$ ( '#change_image_url' ) . hide ( ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-07-29 19:43:52 +00:00
var setInputWidth = function ( elem ) {
var w = Math . min ( Math . max ( 12 + elem . value . length * 6 , 50 ) , 300 ) ;
$ ( elem ) . width ( w ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// updates the context panel tools based on the selected element
var updateContextPanel = function ( ) {
var elem = selectedElement ;
// If element has just been deleted, consider it null
2013-02-15 21:57:35 +00:00
if ( elem != null && ! elem . parentNode ) elem = null ;
2011-01-30 17:11:56 +00:00
var currentLayerName = svgCanvas . getCurrentDrawing ( ) . getCurrentLayerName ( ) ;
2010-03-11 21:07:13 +00:00
var currentMode = svgCanvas . getMode ( ) ;
2010-10-19 17:20:28 +00:00
var unit = curConfig . baseUnit !== 'px' ? curConfig . baseUnit : null ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var is _node = currentMode == 'pathedit' ; //elem ? (elem.id && elem.id.indexOf('pathpointgrip') == 0) : false;
2010-09-24 17:44:34 +00:00
var menu _items = $ ( '#cmenu_canvas li' ) ;
2010-03-11 21:07:13 +00:00
$ ( ' # selected _panel , # multiselected _panel , # g _panel , # rect _panel , # circle _panel , \
2010-12-01 17:54:11 +00:00
# ellipse _panel , # line _panel , # text _panel , # image _panel , # container _panel , # use _panel , # a _panel ' ) . hide ( ) ;
2010-03-11 21:07:13 +00:00
if ( elem != null ) {
var elname = elem . nodeName ;
2010-04-08 13:34:18 +00:00
// If this is a link with no transform and one child, pretend
// its child is selected
2013-02-15 21:57:35 +00:00
// if (elname === 'a') { // && !$(elem).attr('transform')) {
2010-04-08 13:34:18 +00:00
// elem = elem.firstChild;
// }
2010-03-11 21:07:13 +00:00
var angle = svgCanvas . getRotationAngle ( elem ) ;
$ ( '#angle' ) . val ( angle ) ;
2012-10-22 10:47:50 +00:00
2010-04-06 18:40:10 +00:00
var blurval = svgCanvas . getBlur ( elem ) ;
$ ( '#blur' ) . val ( blurval ) ;
$ ( '#blur_slider' ) . slider ( 'option' , 'value' , blurval ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( svgCanvas . addedNew ) {
if ( elname === 'image' ) {
2010-07-05 15:38:06 +00:00
// Prompt for URL if not a data URL
2013-02-15 21:57:35 +00:00
if ( svgCanvas . getHref ( elem ) . indexOf ( 'data:' ) !== 0 ) {
2010-07-05 15:38:06 +00:00
promptImgURL ( ) ;
}
2013-02-15 21:57:35 +00:00
} / * else if ( elname == 'text' ) {
2010-03-11 21:07:13 +00:00
// TODO: Do something here for new text
2011-02-09 06:14:47 +00:00
} * /
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! is _node && currentMode != 'pathedit' ) {
2010-03-11 21:07:13 +00:00
$ ( '#selected_panel' ) . show ( ) ;
// Elements in this array already have coord fields
2013-02-15 21:57:35 +00:00
if ( [ 'line' , 'circle' , 'ellipse' ] . indexOf ( elname ) >= 0 ) {
2010-03-11 21:07:13 +00:00
$ ( '#xy_panel' ) . hide ( ) ;
} else {
var x , y ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Get BBox vals for g, polyline and path
2013-02-15 21:57:35 +00:00
if ( [ 'g' , 'polyline' , 'path' ] . indexOf ( elname ) >= 0 ) {
2010-03-11 21:07:13 +00:00
var bb = svgCanvas . getStrokedBBox ( [ elem ] ) ;
2013-02-15 21:57:35 +00:00
if ( bb ) {
2010-03-11 21:07:13 +00:00
x = bb . x ;
y = bb . y ;
}
} else {
x = elem . getAttribute ( 'x' ) ;
y = elem . getAttribute ( 'y' ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( unit ) {
2011-02-02 17:28:43 +00:00
x = svgedit . units . convertUnit ( x ) ;
y = svgedit . units . convertUnit ( y ) ;
2012-10-22 10:47:50 +00:00
}
2010-03-11 21:07:13 +00:00
$ ( '#selected_x' ) . val ( x || 0 ) ;
$ ( '#selected_y' ) . val ( y || 0 ) ;
$ ( '#xy_panel' ) . show ( ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Elements in this array cannot be converted to a path
2010-09-29 18:37:51 +00:00
var no _path = [ 'image' , 'text' , 'path' , 'g' , 'use' ] . indexOf ( elname ) == - 1 ;
2010-03-11 21:07:13 +00:00
$ ( '#tool_topath' ) . toggle ( no _path ) ;
2013-02-23 18:34:23 +00:00
$ ( '#tool_reorient' ) . toggle ( elname === 'path' ) ;
$ ( '#tool_reorient' ) . toggleClass ( 'disabled' , angle === 0 ) ;
2010-03-11 21:07:13 +00:00
} else {
var point = path . getNodePoint ( ) ;
$ ( '#tool_add_subpath' ) . removeClass ( 'push_button_pressed' ) . addClass ( 'tool_button' ) ;
$ ( '#tool_node_delete' ) . toggleClass ( 'disabled' , ! path . canDeleteNodes ) ;
2012-10-22 10:47:50 +00:00
2010-03-24 20:13:13 +00:00
// Show open/close button based on selected point
2010-05-12 15:45:43 +00:00
setIcon ( '#tool_openclose_path' , path . closed _subpath ? 'open_path' : 'close_path' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( point ) {
2010-03-11 21:07:13 +00:00
var seg _type = $ ( '#seg_type' ) ;
2013-02-15 21:57:35 +00:00
if ( unit ) {
2011-02-02 17:28:43 +00:00
point . x = svgedit . units . convertUnit ( point . x ) ;
point . y = svgedit . units . convertUnit ( point . y ) ;
2010-10-19 17:20:28 +00:00
}
2010-03-11 21:07:13 +00:00
$ ( '#path_node_x' ) . val ( point . x ) ;
$ ( '#path_node_y' ) . val ( point . y ) ;
2013-02-15 21:57:35 +00:00
if ( point . type ) {
2010-03-11 21:07:13 +00:00
seg _type . val ( point . type ) . removeAttr ( 'disabled' ) ;
} else {
2013-02-15 22:32:35 +00:00
seg _type . val ( 4 ) . attr ( 'disabled' , 'disabled' ) ;
2010-03-11 21:07:13 +00:00
}
}
return ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// update contextual tools here
var panels = {
g : [ ] ,
2010-12-01 17:54:11 +00:00
a : [ ] ,
2013-02-15 22:32:35 +00:00
rect : [ 'rx' , 'width' , 'height' ] ,
image : [ 'width' , 'height' ] ,
circle : [ 'cx' , 'cy' , 'r' ] ,
ellipse : [ 'cx' , 'cy' , 'rx' , 'ry' ] ,
line : [ 'x1' , 'y1' , 'x2' , 'y2' ] ,
2010-07-22 19:10:51 +00:00
text : [ ] ,
2013-02-15 22:32:35 +00:00
use : [ ]
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var el _name = elem . tagName ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
// if ($(elem).data('gsvg')) {
2010-07-29 15:22:09 +00:00
// $('#g_panel').show();
// }
2012-10-22 10:47:50 +00:00
2010-12-01 17:54:11 +00:00
var link _href = null ;
if ( el _name === 'a' ) {
link _href = svgCanvas . getHref ( elem ) ;
$ ( '#g_panel' ) . show ( ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( elem . parentNode . tagName === 'a' ) {
if ( ! $ ( elem ) . siblings ( ) . length ) {
2010-12-01 17:54:11 +00:00
$ ( '#a_panel' ) . show ( ) ;
link _href = svgCanvas . getHref ( elem . parentNode ) ;
}
}
2012-10-22 10:47:50 +00:00
2010-12-01 17:54:11 +00:00
// Hide/show the make_link buttons
$ ( '#tool_make_link, #tool_make_link' ) . toggle ( ! link _href ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( link _href ) {
2010-12-01 17:54:11 +00:00
$ ( '#link_url' ) . val ( link _href ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( panels [ el _name ] ) {
2010-03-11 21:07:13 +00:00
var cur _panel = panels [ el _name ] ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#' + el _name + '_panel' ) . show ( ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ . each ( cur _panel , function ( i , item ) {
2010-10-19 17:20:28 +00:00
var attrVal = elem . getAttribute ( item ) ;
2013-02-15 21:57:35 +00:00
if ( curConfig . baseUnit !== 'px' && elem [ item ] ) {
2010-10-19 17:20:28 +00:00
var bv = elem [ item ] . baseVal . value ;
2011-02-02 17:28:43 +00:00
attrVal = svgedit . units . convertUnit ( bv ) ;
2010-10-19 17:20:28 +00:00
}
$ ( '#' + el _name + '_' + item ) . val ( attrVal || 0 ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( el _name == 'text' ) {
$ ( '#text_panel' ) . css ( 'display' , 'inline' ) ;
2010-03-11 21:07:13 +00:00
if ( svgCanvas . getItalic ( ) ) {
$ ( '#tool_italic' ) . addClass ( 'push_button_pressed' ) . removeClass ( 'tool_button' ) ;
2013-02-15 21:57:35 +00:00
} else {
2010-03-11 21:07:13 +00:00
$ ( '#tool_italic' ) . removeClass ( 'push_button_pressed' ) . addClass ( 'tool_button' ) ;
}
if ( svgCanvas . getBold ( ) ) {
$ ( '#tool_bold' ) . addClass ( 'push_button_pressed' ) . removeClass ( 'tool_button' ) ;
2013-02-15 21:57:35 +00:00
} else {
2010-03-11 21:07:13 +00:00
$ ( '#tool_bold' ) . removeClass ( 'push_button_pressed' ) . addClass ( 'tool_button' ) ;
}
2013-02-15 21:57:35 +00:00
$ ( '#font_family' ) . val ( elem . getAttribute ( 'font-family' ) ) ;
$ ( '#font_size' ) . val ( elem . getAttribute ( 'font-size' ) ) ;
2010-03-11 21:07:13 +00:00
$ ( '#text' ) . val ( elem . textContent ) ;
if ( svgCanvas . addedNew ) {
2010-09-16 16:20:48 +00:00
// Timeout needed for IE9
setTimeout ( function ( ) {
$ ( '#text' ) . focus ( ) . select ( ) ;
2013-02-15 22:32:35 +00:00
} , 100 ) ;
2010-03-11 21:07:13 +00:00
}
} // text
2013-02-15 21:57:35 +00:00
else if ( el _name == 'image' ) {
2010-08-16 17:53:15 +00:00
setImageURL ( svgCanvas . getHref ( elem ) ) ;
2010-03-11 21:07:13 +00:00
} // image
2013-02-15 21:57:35 +00:00
else if ( el _name === 'g' || el _name === 'use' ) {
2010-07-29 15:22:09 +00:00
$ ( '#container_panel' ) . show ( ) ;
2010-07-22 19:10:51 +00:00
var title = svgCanvas . getTitle ( ) ;
2010-07-29 19:43:52 +00:00
var label = $ ( '#g_title' ) [ 0 ] ;
label . value = title ;
setInputWidth ( label ) ;
2013-02-20 21:50:18 +00:00
label . prop ( 'disabled' , el _name == 'use' ) ;
2010-07-22 19:10:51 +00:00
}
2010-03-11 21:07:13 +00:00
}
2013-02-15 21:57:35 +00:00
menu _items [ ( el _name === 'g' ? 'en' : 'dis' ) + 'ableContextMenuItems' ] ( '#ungroup' ) ;
menu _items [ ( ( el _name === 'g' || ! multiselected ) ? 'dis' : 'en' ) + 'ableContextMenuItems' ] ( '#group' ) ;
2010-03-11 21:07:13 +00:00
} // if (elem != null)
else if ( multiselected ) {
$ ( '#multiselected_panel' ) . show ( ) ;
2010-09-24 17:44:34 +00:00
menu _items
. enableContextMenuItems ( '#group' )
. disableContextMenuItems ( '#ungroup' ) ;
2010-08-16 20:26:06 +00:00
} else {
2010-09-27 14:20:20 +00:00
menu _items . disableContextMenuItems ( '#delete,#cut,#copy,#group,#ungroup,#move_front,#move_up,#move_down,#move_back' ) ;
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// update history buttons
2013-02-16 16:07:19 +00:00
$ ( '#tool_undo' ) . toggleClass ( 'disabled' , undoMgr . getUndoStackSize ( ) === 0 ) ;
$ ( '#tool_redo' ) . toggleClass ( 'disabled' , undoMgr . getRedoStackSize ( ) === 0 ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
svgCanvas . addedNew = false ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
if ( ( elem && ! is _node ) || multiselected ) {
// update the selected elements' layer
2011-01-26 03:18:59 +00:00
$ ( '#selLayerNames' ) . removeAttr ( 'disabled' ) . val ( currentLayerName ) ;
2012-10-22 10:47:50 +00:00
2010-08-16 20:26:06 +00:00
// Enable regular menu options
2010-09-27 14:20:20 +00:00
canv _menu . enableContextMenuItems ( '#delete,#cut,#copy,#move_front,#move_up,#move_down,#move_back' ) ;
2013-02-15 21:57:35 +00:00
} else {
2010-03-11 21:07:13 +00:00
$ ( '#selLayerNames' ) . attr ( 'disabled' , 'disabled' ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#text' ) . focus ( function ( ) { textBeingEntered = true ; } ) ;
$ ( '#text' ) . blur ( function ( ) { textBeingEntered = false ; } ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// bind the selected event to our function that handles updates to the UI
2013-02-15 21:57:35 +00:00
svgCanvas . bind ( 'selected' , selectedChanged ) ;
svgCanvas . bind ( 'transition' , elementTransition ) ;
svgCanvas . bind ( 'changed' , elementChanged ) ;
svgCanvas . bind ( 'saved' , saveHandler ) ;
svgCanvas . bind ( 'exported' , exportHandler ) ;
svgCanvas . bind ( 'zoomed' , zoomChanged ) ;
svgCanvas . bind ( 'contextset' , contextChanged ) ;
svgCanvas . bind ( 'extension_added' , extAdded ) ;
svgCanvas . textActions . setInputElem ( $ ( '#text' ) [ 0 ] ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 13:38:55 +00:00
var str = '<div class="palette_item" data-rgb="none"></div>' ;
2013-02-23 19:22:00 +00:00
$ . each ( palette , function ( i , item ) {
2010-03-11 21:07:13 +00:00
str += '<div class="palette_item" style="background-color: ' + item + ';" data-rgb="' + item + '"></div>' ;
} ) ;
$ ( '#palette' ) . append ( str ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Set up editor background functionality
// TODO add checkerboard as "pattern"
2013-02-15 23:05:23 +00:00
var color _blocks = [ '#FFF' , '#888' , '#000' ] ; // ,'url(data:image/gif;base64,R0lGODlhEAAQAIAAAP%2F%2F%2F9bW1iH5BAAAAAAALAAAAAAQABAAAAIfjG%2Bgq4jM3IFLJgpswNly%2FXkcBpIiVaInlLJr9FZWAQA7)'];
2010-03-11 21:07:13 +00:00
var str = '' ;
$ . each ( color _blocks , function ( ) {
str += '<div class="color_block" style="background-color:' + this + ';"></div>' ;
} ) ;
$ ( '#bg_blocks' ) . append ( str ) ;
var blocks = $ ( '#bg_blocks div' ) ;
var cur _bg = 'cur_background' ;
blocks . each ( function ( ) {
var blk = $ ( this ) ;
blk . click ( function ( ) {
blocks . removeClass ( cur _bg ) ;
$ ( this ) . addClass ( cur _bg ) ;
} ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( $ . pref ( 'bkgd_color' ) ) {
2010-03-11 21:07:13 +00:00
setBackground ( $ . pref ( 'bkgd_color' ) , $ . pref ( 'bkgd_url' ) ) ;
2013-02-15 21:57:35 +00:00
} else if ( $ . pref ( 'bkgd_url' ) ) {
2010-03-15 14:12:45 +00:00
// No color set, only URL
setBackground ( defaultPrefs . bkgd _color , $ . pref ( 'bkgd_url' ) ) ;
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( $ . pref ( 'img_save' ) ) {
2010-03-11 21:07:13 +00:00
curPrefs . img _save = $ . pref ( 'img_save' ) ;
$ ( '#image_save_opts input' ) . val ( [ curPrefs . img _save ] ) ;
2009-11-19 15:09:21 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var changeRectRadius = function ( ctl ) {
svgCanvas . setRectRadius ( ctl . value ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var changeFontSize = function ( ctl ) {
svgCanvas . setFontSize ( ctl . value ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var changeStrokeWidth = function ( ctl ) {
var val = ctl . value ;
2013-02-15 21:57:35 +00:00
if ( val == 0 && selectedElement && [ 'line' , 'polyline' ] . indexOf ( selectedElement . nodeName ) >= 0 ) {
2010-03-11 21:07:13 +00:00
val = ctl . value = 1 ;
}
svgCanvas . setStrokeWidth ( val ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var changeRotationAngle = function ( ctl ) {
svgCanvas . setRotationAngle ( ctl . value ) ;
2013-02-23 18:34:23 +00:00
$ ( '#tool_reorient' ) . toggleClass ( 'disabled' , parseInt ( ctl . value , 10 ) === 0 ) ;
2013-02-15 13:38:55 +00:00
} ;
2010-03-11 21:07:13 +00:00
var changeZoom = function ( ctl ) {
var zoomlevel = ctl . value / 100 ;
2013-02-15 21:57:35 +00:00
if ( zoomlevel < . 001 ) {
2010-10-21 20:30:42 +00:00
ctl . value = . 1 ;
return ;
}
2010-03-11 21:07:13 +00:00
var zoom = svgCanvas . getZoom ( ) ;
var w _area = workarea ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
zoomChanged ( window , {
width : 0 ,
height : 0 ,
// center pt of scroll position
2012-10-22 10:47:50 +00:00
x : ( w _area [ 0 ] . scrollLeft + w _area . width ( ) / 2 ) / zoom ,
2010-03-11 21:07:13 +00:00
y : ( w _area [ 0 ] . scrollTop + w _area . height ( ) / 2 ) / zoom ,
zoom : zoomlevel
} , true ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var changeOpacity = function ( ctl , val ) {
2013-02-15 21:57:35 +00:00
if ( val == null ) val = ctl . value ;
2010-03-11 21:07:13 +00:00
$ ( '#group_opacity' ) . val ( val ) ;
2013-02-15 21:57:35 +00:00
if ( ! ctl || ! ctl . handle ) {
2010-03-11 21:07:13 +00:00
$ ( '#opac_slider' ) . slider ( 'option' , 'value' , val ) ;
2010-01-13 18:28:19 +00:00
}
2010-03-11 21:07:13 +00:00
svgCanvas . setOpacity ( val / 100 ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-04-06 20:16:07 +00:00
var changeBlur = function ( ctl , val , noUndo ) {
2013-02-15 21:57:35 +00:00
if ( val == null ) val = ctl . value ;
2010-04-06 18:40:10 +00:00
$ ( '#blur' ) . val ( val ) ;
2010-04-28 16:34:02 +00:00
var complete = false ;
2013-02-15 21:57:35 +00:00
if ( ! ctl || ! ctl . handle ) {
2010-04-06 18:40:10 +00:00
$ ( '#blur_slider' ) . slider ( 'option' , 'value' , val ) ;
2010-04-28 16:34:02 +00:00
complete = true ;
}
2013-02-15 21:57:35 +00:00
if ( noUndo ) {
2012-10-22 10:47:50 +00:00
svgCanvas . setBlurNoUndo ( val ) ;
2010-04-28 16:34:02 +00:00
} else {
svgCanvas . setBlur ( val , complete ) ;
2010-04-06 18:40:10 +00:00
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var operaRepaint = function ( ) {
// Repaints canvas in Opera. Needed for stroke-dasharray change as well as fill change
2013-02-15 21:57:35 +00:00
if ( ! window . opera ) return ;
2010-03-11 21:07:13 +00:00
$ ( '<p/>' ) . hide ( ) . appendTo ( 'body' ) . remove ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
$ ( '#stroke_style' ) . change ( function ( ) {
2010-04-01 19:02:36 +00:00
svgCanvas . setStrokeAttr ( 'stroke-dasharray' , $ ( this ) . val ( ) ) ;
2010-03-11 21:07:13 +00:00
operaRepaint ( ) ;
} ) ;
2010-04-01 19:02:36 +00:00
2013-02-21 20:34:11 +00:00
$ ( '#stroke_linejoin' ) . change ( function ( ) {
2010-04-01 19:02:36 +00:00
svgCanvas . setStrokeAttr ( 'stroke-linejoin' , $ ( this ) . val ( ) ) ;
operaRepaint ( ) ;
} ) ;
2010-03-11 21:07:13 +00:00
// Lose focus for select elements when changed (Allows keyboard shortcuts to work better)
$ ( 'select' ) . change ( function ( ) { $ ( this ) . blur ( ) ; } ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// fired when user wants to move elements to another layer
var promptMoveLayerOnce = false ;
2013-02-21 20:34:11 +00:00
$ ( '#selLayerNames' ) . change ( function ( ) {
2010-03-11 21:07:13 +00:00
var destLayer = this . options [ this . selectedIndex ] . value ;
2013-02-15 23:05:23 +00:00
var confirmStr = uiStrings . notification . QmoveElemsToLayer . replace ( '%s' , destLayer ) ;
2010-03-11 21:07:13 +00:00
var moveToLayer = function ( ok ) {
2013-02-15 21:57:35 +00:00
if ( ! ok ) return ;
2010-03-11 21:07:13 +00:00
promptMoveLayerOnce = true ;
svgCanvas . moveSelectedToLayer ( destLayer ) ;
svgCanvas . clearSelection ( ) ;
populateLayers ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2010-03-11 21:07:13 +00:00
if ( destLayer ) {
2013-02-15 21:57:35 +00:00
if ( promptMoveLayerOnce ) {
2010-03-11 21:07:13 +00:00
moveToLayer ( true ) ;
} else {
2013-02-15 23:05:23 +00:00
$ . confirm ( confirmStr , moveToLayer ) ;
2010-03-11 21:07:13 +00:00
}
}
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#font_family' ) . change ( function ( ) {
svgCanvas . setFontFamily ( this . value ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#seg_type' ) . change ( function ( ) {
svgCanvas . setSegType ( $ ( this ) . val ( ) ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
$ ( '#text' ) . keyup ( function ( ) {
2010-03-11 21:07:13 +00:00
svgCanvas . setTextContent ( this . value ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
$ ( '#image_url' ) . change ( function ( ) {
2012-10-22 10:47:50 +00:00
setImageURL ( this . value ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-12-01 17:54:11 +00:00
$ ( '#link_url' ) . change ( function ( ) {
2013-02-15 21:57:35 +00:00
if ( this . value . length ) {
2010-12-01 17:54:11 +00:00
svgCanvas . setLinkURL ( this . value ) ;
} else {
svgCanvas . removeHyperlink ( ) ;
}
2010-07-22 19:10:51 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2011-02-04 18:49:12 +00:00
$ ( '#g_title' ) . change ( function ( ) {
svgCanvas . setGroupTitle ( this . value ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '.attr_changer' ) . change ( function ( ) {
2013-02-15 21:57:35 +00:00
var attr = this . getAttribute ( 'data-attr' ) ;
2010-03-11 21:07:13 +00:00
var val = this . value ;
2012-05-25 04:04:22 +00:00
var valid = svgedit . units . isValidUnit ( attr , val , selectedElement ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! valid ) {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . invalidAttrValGiven ) ;
2010-03-11 21:07:13 +00:00
this . value = selectedElement . getAttribute ( attr ) ;
return false ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( attr !== 'id' ) {
2011-02-10 04:10:03 +00:00
if ( isNaN ( val ) ) {
val = svgCanvas . convertToNum ( attr , val ) ;
2013-02-15 21:57:35 +00:00
} else if ( curConfig . baseUnit !== 'px' ) {
2011-02-10 04:10:03 +00:00
// Convert unitless value to one with given unit
2012-10-22 10:47:50 +00:00
2011-02-10 04:10:03 +00:00
var unitData = svgedit . units . getTypeMap ( ) ;
2010-10-19 17:20:28 +00:00
2013-02-15 21:57:35 +00:00
if ( selectedElement [ attr ] || svgCanvas . getMode ( ) === 'pathedit' || attr === 'x' || attr === 'y' ) {
2011-02-10 04:10:03 +00:00
val *= unitData [ curConfig . baseUnit ] ;
}
2010-10-19 17:20:28 +00:00
}
2010-10-05 17:13:00 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// if the user is changing the id, then de-select the element first
// change the ID, then re-select it with the new ID
2013-02-15 21:57:35 +00:00
if ( attr === 'id' ) {
2010-03-11 21:07:13 +00:00
var elem = selectedElement ;
svgCanvas . clearSelection ( ) ;
elem . id = val ;
svgCanvas . addToSelection ( [ elem ] , true ) ;
2013-02-15 21:57:35 +00:00
} else {
2010-03-11 21:07:13 +00:00
svgCanvas . changeSelectedAttribute ( attr , val ) ;
}
2012-05-25 04:04:22 +00:00
this . blur ( ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Prevent selection of elements when shift-clicking
$ ( '#palette' ) . mouseover ( function ( ) {
var inp = $ ( '<input type="hidden">' ) ;
$ ( this ) . append ( inp ) ;
inp . focus ( ) . remove ( ) ;
2013-02-15 13:38:55 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-19 13:17:27 +00:00
$ ( '.palette_item' ) . mousedown ( function ( evt ) {
// shift key or right click for stroke
var picker = evt . shiftKey || evt . button === 2 ? 'stroke' : 'fill' ;
var color = $ ( this ) . data ( 'rgb' ) ;
var paint ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Webkit-based browsers returned 'initial' here for no stroke
2012-12-05 00:06:48 +00:00
if ( color === 'none' || color === 'transparent' || color === 'initial' ) {
2010-03-11 21:07:13 +00:00
color = 'none' ;
paint = new $ . jGraduate . Paint ( ) ;
2013-02-15 21:57:35 +00:00
} else {
2010-03-11 21:07:13 +00:00
paint = new $ . jGraduate . Paint ( { alpha : 100 , solidColor : color . substr ( 1 ) } ) ;
}
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
paintBox [ picker ] . setPaint ( paint ) ;
2013-02-19 13:17:27 +00:00
svgCanvas . setColor ( picker , color ) ;
2012-10-22 10:47:50 +00:00
2013-02-19 13:17:27 +00:00
if ( color !== 'none' && svgCanvas . getPaintOpacity ( picker ) !== 1 ) {
svgCanvas . setPaintOpacity ( picker , 1.0 ) ;
2010-03-11 21:07:13 +00:00
}
updateToolButtonState ( ) ;
2013-02-15 13:38:55 +00:00
} ) . bind ( 'contextmenu' , function ( e ) { e . preventDefault ( ) ; } ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#toggle_stroke_tools' ) . on ( 'click' , function ( ) {
$ ( '#tools_bottom' ) . toggleClass ( 'expanded' ) ;
2010-04-01 19:02:36 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// This is a common function used when a tool has been clicked (chosen)
// It does several common things:
// - removes the tool_button_current class from whatever tool currently has it
// - hides any flyouts
// - adds the tool_button_current class to the button passed in
2010-07-09 19:09:37 +00:00
var toolButtonClick = function ( button , noHiding ) {
2010-03-11 21:07:13 +00:00
if ( $ ( button ) . hasClass ( 'disabled' ) ) return false ;
2013-02-15 21:57:35 +00:00
if ( $ ( button ) . parent ( ) . hasClass ( 'tools_flyout' ) ) return true ;
2010-03-11 21:07:13 +00:00
var fadeFlyouts = fadeFlyouts || 'normal' ;
2013-02-15 21:57:35 +00:00
if ( ! noHiding ) {
2010-07-09 19:09:37 +00:00
$ ( '.tools_flyout' ) . fadeOut ( fadeFlyouts ) ;
}
2010-03-11 21:07:13 +00:00
$ ( '#styleoverrides' ) . text ( '' ) ;
2013-02-23 18:56:45 +00:00
workarea . css ( 'cursor' , 'auto' ) ;
2010-03-11 21:07:13 +00:00
$ ( '.tool_button_current' ) . removeClass ( 'tool_button_current' ) . addClass ( 'tool_button' ) ;
$ ( button ) . addClass ( 'tool_button_current' ) . removeClass ( 'tool_button' ) ;
return true ;
} ;
2012-10-22 10:47:50 +00:00
2010-04-01 16:00:50 +00:00
( function ( ) {
2012-10-22 10:47:50 +00:00
var last _x = null , last _y = null , w _area = workarea [ 0 ] ,
2010-04-01 16:00:50 +00:00
panning = false , keypan = false ;
2012-10-22 10:47:50 +00:00
2010-04-01 16:00:50 +00:00
$ ( '#svgcanvas' ) . bind ( 'mousemove mouseup' , function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( panning === false ) return ;
2010-04-01 16:00:50 +00:00
w _area . scrollLeft -= ( evt . clientX - last _x ) ;
w _area . scrollTop -= ( evt . clientY - last _y ) ;
2012-10-22 10:47:50 +00:00
2010-04-01 16:00:50 +00:00
last _x = evt . clientX ;
last _y = evt . clientY ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( evt . type === 'mouseup' ) panning = false ;
2010-04-01 16:00:50 +00:00
return false ;
} ) . mousedown ( function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( evt . button === 1 || keypan === true ) {
2010-04-01 16:00:50 +00:00
panning = true ;
last _x = evt . clientX ;
last _y = evt . clientY ;
return false ;
}
} ) ;
2012-10-22 10:47:50 +00:00
2010-04-01 16:00:50 +00:00
$ ( window ) . mouseup ( function ( ) {
panning = false ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-04-01 16:00:50 +00:00
$ ( document ) . bind ( 'keydown' , 'space' , function ( evt ) {
svgCanvas . spaceKey = keypan = true ;
evt . preventDefault ( ) ;
} ) . bind ( 'keyup' , 'space' , function ( evt ) {
evt . preventDefault ( ) ;
svgCanvas . spaceKey = keypan = false ;
2010-09-27 18:58:04 +00:00
} ) . bind ( 'keydown' , 'shift' , function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( svgCanvas . getMode ( ) === 'zoom' ) {
2010-09-27 18:58:04 +00:00
workarea . css ( 'cursor' , zoomOutIcon ) ;
}
} ) . bind ( 'keyup' , 'shift' , function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( svgCanvas . getMode ( ) === 'zoom' ) {
2010-09-27 18:58:04 +00:00
workarea . css ( 'cursor' , zoomInIcon ) ;
}
2013-02-15 13:38:55 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 13:38:55 +00:00
Editor . setPanning = function ( active ) {
svgCanvas . spaceKey = keypan = active ;
} ;
} ( ) ) ;
2012-10-22 10:47:50 +00:00
2010-05-24 19:46:32 +00:00
function setStrokeOpt ( opt , changeElem ) {
var id = opt . id ;
var bits = id . split ( '_' ) ;
var pre = bits [ 0 ] ;
var val = bits [ 1 ] ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( changeElem ) {
2010-05-24 19:46:32 +00:00
svgCanvas . setStrokeAttr ( 'stroke-' + pre , val ) ;
}
operaRepaint ( ) ;
setIcon ( '#cur_' + pre , id , 20 ) ;
$ ( opt ) . addClass ( 'current' ) . siblings ( ) . removeClass ( 'current' ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
( function ( ) {
var button = $ ( '#main_icon' ) ;
var overlay = $ ( '#main_icon span' ) ;
var list = $ ( '#main_menu' ) ;
var on _button = false ;
var height = 0 ;
var js _hover = true ;
var set _click = false ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var hideMenu = function ( ) {
list . fadeOut ( 200 ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( window ) . mouseup ( function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( ! on _button ) {
2010-03-11 21:07:13 +00:00
button . removeClass ( 'buttondown' ) ;
2012-10-22 10:47:50 +00:00
// do not hide if it was the file input as that input needs to be visible
2010-03-11 21:07:13 +00:00
// for its change event to fire
2013-02-15 21:57:35 +00:00
if ( evt . target . tagName != 'INPUT' ) {
2010-03-11 21:07:13 +00:00
list . fadeOut ( 200 ) ;
2013-02-15 21:57:35 +00:00
} else if ( ! set _click ) {
2010-03-11 21:07:13 +00:00
set _click = true ;
$ ( evt . target ) . click ( function ( ) {
2013-02-15 21:57:35 +00:00
list . css ( 'margin-left' , '-9999px' ) . show ( ) ;
2010-03-11 21:07:13 +00:00
} ) ;
}
}
on _button = false ;
2010-07-09 19:09:37 +00:00
} ) . mousedown ( function ( evt ) {
2013-02-15 21:57:35 +00:00
// $('.contextMenu').hide();
2010-08-16 20:26:06 +00:00
var islib = $ ( evt . target ) . closest ( 'div.tools_flyout, .contextMenu' ) . length ;
2013-02-15 21:57:35 +00:00
if ( ! islib ) $ ( '.tools_flyout:visible,.contextMenu' ) . fadeOut ( 250 ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
overlay . bind ( 'mousedown' , function ( ) {
if ( ! button . hasClass ( 'buttondown' ) ) {
// Margin must be reset in case it was changed before;
2013-02-16 16:07:19 +00:00
list . css ( 'margin-left' , 0 ) . show ( ) ;
2013-02-15 21:57:35 +00:00
if ( ! height ) {
2010-03-11 21:07:13 +00:00
height = list . height ( ) ;
}
2013-02-15 21:57:35 +00:00
// Using custom animation as slideDown has annoying 'bounce effect'
2010-03-11 21:07:13 +00:00
list . css ( 'height' , 0 ) . animate ( {
'height' : height
2013-02-16 16:07:19 +00:00
} , 200 ) ;
2010-03-11 21:07:13 +00:00
on _button = true ;
} else {
list . fadeOut ( 200 ) ;
}
2013-02-16 16:07:19 +00:00
button . toggleClass ( 'buttondown buttonup' ) ;
2010-03-11 21:07:13 +00:00
} ) . hover ( function ( ) {
on _button = true ;
} ) . mouseout ( function ( ) {
on _button = false ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var list _items = $ ( '#main_menu li' ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Check if JS method of hovering needs to be used (Webkit bug)
2010-01-13 18:28:19 +00:00
list _items . mouseover ( function ( ) {
2010-03-11 21:07:13 +00:00
js _hover = ( $ ( this ) . css ( 'background-color' ) == 'rgba(0, 0, 0, 0)' ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
list _items . unbind ( 'mouseover' ) ;
2013-02-15 21:57:35 +00:00
if ( js _hover ) {
2010-03-11 21:07:13 +00:00
list _items . mouseover ( function ( ) {
this . style . backgroundColor = '#FFC' ;
} ) . mouseout ( function ( ) {
this . style . backgroundColor = 'transparent' ;
return true ;
} ) ;
}
} ) ;
} ( ) ) ;
2010-07-26 17:48:02 +00:00
// Made public for UI customization.
// TODO: Group UI functions into a public svgEditor.ui interface.
Editor . addDropDown = function ( elem , callback , dropUp ) {
2011-06-24 16:28:34 +00:00
if ( $ ( elem ) . length == 0 ) return ; // Quit if called on non-existant element
2010-03-11 21:07:13 +00:00
var button = $ ( elem ) . find ( 'button' ) ;
2011-01-31 15:57:01 +00:00
var list = $ ( elem ) . find ( 'ul' ) . attr ( 'id' , $ ( elem ) [ 0 ] . id + '-list' ) ;
2010-03-11 21:07:13 +00:00
var on _button = false ;
2013-02-15 21:57:35 +00:00
if ( dropUp ) {
2010-03-11 21:07:13 +00:00
$ ( elem ) . addClass ( 'dropup' ) ;
2013-02-18 20:55:04 +00:00
} else {
// Move list to place where it can overflow container
$ ( '#option_lists' ) . append ( list ) ;
2010-03-11 21:07:13 +00:00
}
2011-01-31 15:57:01 +00:00
list . find ( 'li' ) . bind ( 'mouseup' , callback ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( window ) . mouseup ( function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( ! on _button ) {
2010-03-11 21:07:13 +00:00
button . removeClass ( 'down' ) ;
list . hide ( ) ;
}
on _button = false ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
button . bind ( 'mousedown' , function ( ) {
if ( ! button . hasClass ( 'down' ) ) {
2013-02-15 21:57:35 +00:00
if ( ! dropUp ) {
2011-01-31 15:57:01 +00:00
var pos = $ ( elem ) . position ( ) ;
list . css ( {
top : pos . top + 24 ,
left : pos . left - 10
} ) ;
}
2010-03-11 21:07:13 +00:00
list . show ( ) ;
on _button = true ;
} else {
list . hide ( ) ;
}
2013-02-16 16:07:19 +00:00
button . toggleClass ( 'down' ) ;
2010-03-11 21:07:13 +00:00
} ) . hover ( function ( ) {
on _button = true ;
2010-01-13 18:28:19 +00:00
} ) . mouseout ( function ( ) {
2010-03-11 21:07:13 +00:00
on _button = false ;
2010-01-13 18:28:19 +00:00
} ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-04-05 19:53:14 +00:00
// TODO: Combine this with addDropDown or find other way to optimize
2010-04-16 19:16:44 +00:00
var addAltDropDown = function ( elem , list , callback , opts ) {
2010-04-05 19:53:14 +00:00
var button = $ ( elem ) ;
var list = $ ( list ) ;
var on _button = false ;
2010-04-16 19:16:44 +00:00
var dropUp = opts . dropUp ;
2013-02-15 21:57:35 +00:00
if ( dropUp ) {
2010-04-05 19:53:14 +00:00
$ ( elem ) . addClass ( 'dropup' ) ;
}
2010-05-27 13:27:04 +00:00
list . find ( 'li' ) . bind ( 'mouseup' , function ( ) {
2013-02-15 21:57:35 +00:00
if ( opts . seticon ) {
2010-05-27 13:27:04 +00:00
setIcon ( '#cur_' + button [ 0 ] . id , $ ( this ) . children ( ) ) ;
$ ( this ) . addClass ( 'current' ) . siblings ( ) . removeClass ( 'current' ) ;
}
callback . apply ( this , arguments ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-04-05 19:53:14 +00:00
$ ( window ) . mouseup ( function ( evt ) {
2013-02-15 21:57:35 +00:00
if ( ! on _button ) {
2010-04-05 19:53:14 +00:00
button . removeClass ( 'down' ) ;
list . hide ( ) ;
list . css ( { top : 0 , left : 0 } ) ;
}
on _button = false ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-04-05 19:53:14 +00:00
var height = list . height ( ) ;
2013-02-18 20:55:04 +00:00
button . bind ( 'mousedown' , function ( ) {
var off = button . offset ( ) ;
2013-02-15 21:57:35 +00:00
if ( dropUp ) {
2010-04-16 19:16:44 +00:00
off . top -= list . height ( ) ;
off . left += 8 ;
} else {
2013-02-18 20:55:04 +00:00
off . top += button . height ( ) ;
2010-04-16 19:16:44 +00:00
}
2013-02-18 20:55:04 +00:00
list . offset ( off ) ;
2012-10-22 10:47:50 +00:00
2010-04-05 19:53:14 +00:00
if ( ! button . hasClass ( 'down' ) ) {
list . show ( ) ;
on _button = true ;
} else {
// CSS position must be reset for Webkit
list . hide ( ) ;
list . css ( { top : 0 , left : 0 } ) ;
}
2013-02-18 20:55:04 +00:00
button . toggleClass ( 'down' ) ;
2010-04-05 19:53:14 +00:00
} ) . hover ( function ( ) {
on _button = true ;
} ) . mouseout ( function ( ) {
on _button = false ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( opts . multiclick ) {
2010-04-16 19:16:44 +00:00
list . mousedown ( function ( ) {
on _button = true ;
} ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-07-26 17:48:02 +00:00
Editor . addDropDown ( '#font_family_dropdown' , function ( ) {
2010-03-11 21:07:13 +00:00
$ ( '#font_family' ) . val ( $ ( this ) . text ( ) ) . change ( ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-07-26 17:48:02 +00:00
Editor . addDropDown ( '#opacity_dropdown' , function ( ) {
2013-02-15 21:57:35 +00:00
if ( $ ( this ) . find ( 'div' ) . length ) return ;
2010-03-11 21:07:13 +00:00
var perc = parseInt ( $ ( this ) . text ( ) . split ( '%' ) [ 0 ] ) ;
changeOpacity ( false , perc ) ;
} , true ) ;
2012-10-22 10:47:50 +00:00
// For slider usage, see: http://jqueryui.com/demos/slider/
2013-02-15 21:57:35 +00:00
$ ( '#opac_slider' ) . slider ( {
2010-03-11 21:07:13 +00:00
start : function ( ) {
$ ( '#opacity_dropdown li:not(.special)' ) . hide ( ) ;
} ,
stop : function ( ) {
$ ( '#opacity_dropdown li' ) . show ( ) ;
$ ( window ) . mouseup ( ) ;
} ,
2013-02-23 19:22:00 +00:00
slide : function ( evt , ui ) {
2010-03-11 21:07:13 +00:00
changeOpacity ( ui ) ;
}
} ) ;
2012-10-22 10:47:50 +00:00
2011-01-31 15:57:01 +00:00
Editor . addDropDown ( '#blur_dropdown' , $ . noop ) ;
2012-10-22 10:47:50 +00:00
2010-04-28 16:34:02 +00:00
var slideStart = false ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#blur_slider' ) . slider ( {
2010-04-06 18:40:10 +00:00
max : 10 ,
2013-02-19 13:25:54 +00:00
step : 0.1 ,
2010-04-06 20:16:07 +00:00
stop : function ( evt , ui ) {
2010-04-28 16:34:02 +00:00
slideStart = false ;
2010-04-06 20:16:07 +00:00
changeBlur ( ui ) ;
2010-04-06 18:40:10 +00:00
$ ( '#blur_dropdown li' ) . show ( ) ;
$ ( window ) . mouseup ( ) ;
} ,
2010-04-28 16:34:02 +00:00
start : function ( ) {
slideStart = true ;
} ,
2013-02-23 19:22:00 +00:00
slide : function ( evt , ui ) {
2010-04-28 16:34:02 +00:00
changeBlur ( ui , null , slideStart ) ;
2010-04-06 18:40:10 +00:00
}
} ) ;
2010-07-26 17:48:02 +00:00
Editor . addDropDown ( '#zoom_dropdown' , function ( ) {
2010-03-11 21:07:13 +00:00
var item = $ ( this ) ;
2013-02-19 13:25:54 +00:00
var val = item . data ( 'val' ) ;
2013-02-15 21:57:35 +00:00
if ( val ) {
2010-03-11 21:07:13 +00:00
zoomChanged ( window , val ) ;
} else {
2013-02-19 13:25:54 +00:00
changeZoom ( { value : parseInt ( item . text ( ) , 10 ) } ) ;
2010-03-11 21:07:13 +00:00
}
} , true ) ;
2012-10-22 10:47:50 +00:00
2010-04-05 19:53:14 +00:00
addAltDropDown ( '#stroke_linecap' , '#linecap_opts' , function ( ) {
2010-05-24 19:46:32 +00:00
setStrokeOpt ( this , true ) ;
2010-04-16 19:16:44 +00:00
} , { dropUp : true } ) ;
2012-10-22 10:47:50 +00:00
2010-04-06 15:53:21 +00:00
addAltDropDown ( '#stroke_linejoin' , '#linejoin_opts' , function ( ) {
2010-05-24 19:46:32 +00:00
setStrokeOpt ( this , true ) ;
2010-04-16 19:16:44 +00:00
} , { dropUp : true } ) ;
2012-10-22 10:47:50 +00:00
2010-04-16 19:16:44 +00:00
addAltDropDown ( '#tool_position' , '#position_opts' , function ( ) {
2013-02-23 18:56:45 +00:00
var letter = this . id . replace ( 'tool_pos' , '' ) . charAt ( 0 ) ;
2010-04-16 19:16:44 +00:00
svgCanvas . alignSelectedElements ( letter , 'page' ) ;
} , { multiclick : true } ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
/ *
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
When a flyout icon is selected
( if flyout ) {
- Change the icon
- Make pressing the button run its stuff
}
- Run its stuff
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
When its shortcut key is pressed
- If not current in list , do as above
, else :
- Just run its stuff
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
* /
2012-10-22 10:47:50 +00:00
2010-04-20 13:21:51 +00:00
// Unfocus text input when workarea is mousedowned.
( function ( ) {
var inp ;
var unfocus = function ( ) {
$ ( inp ) . blur ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-09-24 16:05:46 +00:00
$ ( '#svg_editor' ) . find ( 'button, select, input:not(#text)' ) . focus ( function ( ) {
2010-04-20 13:21:51 +00:00
inp = this ;
2010-09-24 16:05:46 +00:00
ui _context = 'toolbars' ;
2010-04-20 13:21:51 +00:00
workarea . mousedown ( unfocus ) ;
} ) . blur ( function ( ) {
2010-09-24 16:05:46 +00:00
ui _context = 'canvas' ;
2010-04-20 13:21:51 +00:00
workarea . unbind ( 'mousedown' , unfocus ) ;
2010-07-20 13:35:11 +00:00
// Go back to selecting text if in textedit mode
2013-02-15 21:57:35 +00:00
if ( svgCanvas . getMode ( ) == 'textedit' ) {
2010-07-20 13:35:11 +00:00
$ ( '#text' ) . focus ( ) ;
}
2010-04-20 13:21:51 +00:00
} ) ;
} ( ) ) ;
2010-03-11 21:07:13 +00:00
var clickSelect = function ( ) {
if ( toolButtonClick ( '#tool_select' ) ) {
svgCanvas . setMode ( 'select' ) ;
$ ( '#styleoverrides' ) . text ( '#svgcanvas svg *{cursor:move;pointer-events:all}, #svgcanvas svg{cursor:default}' ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var clickFHPath = function ( ) {
if ( toolButtonClick ( '#tool_fhpath' ) ) {
svgCanvas . setMode ( 'fhpath' ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var clickLine = function ( ) {
if ( toolButtonClick ( '#tool_line' ) ) {
svgCanvas . setMode ( 'line' ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickSquare = function ( ) {
2010-07-14 19:15:44 +00:00
if ( toolButtonClick ( '#tool_square' ) ) {
svgCanvas . setMode ( 'square' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickRect = function ( ) {
2010-07-14 19:15:44 +00:00
if ( toolButtonClick ( '#tool_rect' ) ) {
svgCanvas . setMode ( 'rect' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickFHRect = function ( ) {
2010-07-14 19:15:44 +00:00
if ( toolButtonClick ( '#tool_fhrect' ) ) {
svgCanvas . setMode ( 'fhrect' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickCircle = function ( ) {
2010-07-14 19:15:44 +00:00
if ( toolButtonClick ( '#tool_circle' ) ) {
svgCanvas . setMode ( 'circle' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickEllipse = function ( ) {
2010-07-14 19:15:44 +00:00
if ( toolButtonClick ( '#tool_ellipse' ) ) {
svgCanvas . setMode ( 'ellipse' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickFHEllipse = function ( ) {
2010-07-14 19:15:44 +00:00
if ( toolButtonClick ( '#tool_fhellipse' ) ) {
svgCanvas . setMode ( 'fhellipse' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickImage = function ( ) {
2010-03-11 21:07:13 +00:00
if ( toolButtonClick ( '#tool_image' ) ) {
svgCanvas . setMode ( 'image' ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickZoom = function ( ) {
2010-03-11 21:07:13 +00:00
if ( toolButtonClick ( '#tool_zoom' ) ) {
svgCanvas . setMode ( 'zoom' ) ;
2010-09-27 18:58:04 +00:00
workarea . css ( 'cursor' , zoomInIcon ) ;
2010-03-11 21:07:13 +00:00
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var dblclickZoom = function ( ) {
2010-03-11 21:07:13 +00:00
if ( toolButtonClick ( '#tool_zoom' ) ) {
zoomImage ( ) ;
setSelectMode ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickText = function ( ) {
2010-09-24 17:24:42 +00:00
if ( toolButtonClick ( '#tool_text' ) ) {
svgCanvas . setMode ( 'text' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickPath = function ( ) {
2010-09-24 17:24:42 +00:00
if ( toolButtonClick ( '#tool_path' ) ) {
svgCanvas . setMode ( 'path' ) ;
}
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Delete is a contextual tool that only appears in the ribbon if
// an element has been selected
var deleteSelected = function ( ) {
if ( selectedElement != null || multiselected ) {
svgCanvas . deleteSelectedElements ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-08-17 18:09:50 +00:00
var cutSelected = function ( ) {
if ( selectedElement != null || multiselected ) {
svgCanvas . cutSelectedElements ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-08-17 18:09:50 +00:00
var copySelected = function ( ) {
if ( selectedElement != null || multiselected ) {
svgCanvas . copySelectedElements ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2011-01-28 20:11:18 +00:00
var pasteInCenter = function ( ) {
var zoom = svgCanvas . getZoom ( ) ;
2013-10-13 23:23:08 +00:00
var x = ( workarea [ 0 ] . scrollLeft + workarea . width ( ) / 2 ) / zoom - svgCanvas . contentW ;
var y = ( workarea [ 0 ] . scrollTop + workarea . height ( ) / 2 ) / zoom - svgCanvas . contentH ;
2012-10-22 10:47:50 +00:00
svgCanvas . pasteElements ( 'point' , x , y ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var moveToTopSelected = function ( ) {
if ( selectedElement != null ) {
svgCanvas . moveToTopSelectedElement ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var moveToBottomSelected = function ( ) {
if ( selectedElement != null ) {
svgCanvas . moveToBottomSelectedElement ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-08-20 14:52:55 +00:00
var moveUpDownSelected = function ( dir ) {
if ( selectedElement != null ) {
svgCanvas . moveUpDownSelected ( dir ) ;
}
} ;
2010-07-22 19:10:51 +00:00
2010-03-11 21:07:13 +00:00
var convertToPath = function ( ) {
if ( selectedElement != null ) {
svgCanvas . convertToPath ( ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var reorientPath = function ( ) {
if ( selectedElement != null ) {
path . reorient ( ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-12-01 17:54:11 +00:00
var makeHyperlink = function ( ) {
if ( selectedElement != null || multiselected ) {
2013-02-15 21:57:35 +00:00
$ . prompt ( uiStrings . notification . enterNewLinkURL , 'http://' , function ( url ) {
if ( url ) svgCanvas . makeHyperlink ( url ) ;
2010-12-01 17:54:11 +00:00
} ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var moveSelected = function ( dx , dy ) {
if ( selectedElement != null || multiselected ) {
2013-02-15 21:57:35 +00:00
if ( curConfig . gridSnapping ) {
2010-09-21 19:22:52 +00:00
// Use grid snap value regardless of zoom level
var multi = svgCanvas . getZoom ( ) * curConfig . snappingStep ;
dx *= multi ;
dy *= multi ;
}
2010-03-11 21:07:13 +00:00
svgCanvas . moveSelectedElements ( dx , dy ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var linkControlPoints = function ( ) {
2013-02-16 16:07:19 +00:00
$ ( '#tool_node_link' ) . toggleClass ( 'push_button_pressed tool_button' ) ;
2010-03-11 21:07:13 +00:00
path . linkControlPoints ( linked ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var clonePathNode = function ( ) {
if ( path . getNodePoint ( ) ) {
path . clonePathNode ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var deletePathNode = function ( ) {
if ( path . getNodePoint ( ) ) {
path . deletePathNode ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var addSubPath = function ( ) {
2013-02-16 16:07:19 +00:00
button . toggleClass ( 'push_button_pressed tool_button' ) ;
2010-03-11 21:07:13 +00:00
path . addSubPath ( sp ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-24 20:13:13 +00:00
var opencloseSubPath = function ( ) {
path . opencloseSubPath ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var selectNext = function ( ) {
svgCanvas . cycleElement ( 1 ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var selectPrev = function ( ) {
svgCanvas . cycleElement ( 0 ) ;
} ;
2012-10-22 10:47:50 +00:00
2013-02-23 18:34:23 +00:00
var rotateSelected = function ( cw , step ) {
2010-03-11 21:07:13 +00:00
if ( selectedElement == null || multiselected ) return ;
2013-02-15 21:57:35 +00:00
if ( ! cw ) step *= - 1 ;
2013-02-23 18:34:23 +00:00
var angle = parseFloat ( $ ( '#angle' ) . val ( ) ) + step ;
svgCanvas . setRotationAngle ( angle ) ;
2010-03-11 21:07:13 +00:00
updateContextPanel ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickClear = function ( ) {
2010-03-12 20:43:33 +00:00
var dims = curConfig . dimensions ;
2011-01-03 18:45:36 +00:00
$ . confirm ( uiStrings . notification . QwantToClear , function ( ok ) {
2013-02-15 21:57:35 +00:00
if ( ! ok ) return ;
2010-03-11 21:07:13 +00:00
setSelectMode ( ) ;
svgCanvas . clear ( ) ;
2010-03-12 20:43:33 +00:00
svgCanvas . setResolution ( dims [ 0 ] , dims [ 1 ] ) ;
2010-03-11 21:07:13 +00:00
updateCanvas ( true ) ;
zoomImage ( ) ;
populateLayers ( ) ;
updateContextPanel ( ) ;
2010-10-12 15:20:17 +00:00
prepPaints ( ) ;
2012-03-17 17:56:46 +00:00
svgCanvas . runExtensions ( 'onNewDocument' ) ;
2010-03-11 21:07:13 +00:00
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickBold = function ( ) {
2010-03-11 21:07:13 +00:00
svgCanvas . setBold ( ! svgCanvas . getBold ( ) ) ;
updateContextPanel ( ) ;
2010-07-20 13:35:11 +00:00
return false ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickItalic = function ( ) {
2010-03-11 21:07:13 +00:00
svgCanvas . setItalic ( ! svgCanvas . getItalic ( ) ) ;
updateContextPanel ( ) ;
2010-07-20 13:35:11 +00:00
return false ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickSave = function ( ) {
2010-03-11 21:07:13 +00:00
// In the future, more options can be provided here
var saveOpts = {
'images' : curPrefs . img _save ,
'round_digits' : 6
2013-02-15 13:38:55 +00:00
} ;
2010-03-11 21:07:13 +00:00
svgCanvas . save ( saveOpts ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-04-14 17:30:25 +00:00
var clickExport = function ( ) {
2010-04-22 15:22:01 +00:00
// Open placeholder window (prevents popup)
2013-10-13 23:23:08 +00:00
if ( ! customHandlers . pngsave ) {
2011-01-03 18:45:36 +00:00
var str = uiStrings . notification . loadingImage ;
2013-02-15 21:57:35 +00:00
exportWindow = window . open ( 'data:text/html;charset=utf-8,<title>' + str + '<\/title><h1>' + str + '<\/h1>' ) ;
2010-07-09 19:09:37 +00:00
}
2010-04-22 15:22:01 +00:00
2013-02-15 21:57:35 +00:00
if ( window . canvg ) {
2010-04-14 17:30:25 +00:00
svgCanvas . rasterExport ( ) ;
} else {
$ . getScript ( 'canvg/rgbcolor.js' , function ( ) {
2010-04-22 15:22:01 +00:00
$ . getScript ( 'canvg/canvg.js' , function ( ) {
svgCanvas . rasterExport ( ) ;
} ) ;
2010-04-14 17:30:25 +00:00
} ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// by default, svgCanvas.open() is a no-op.
2012-10-22 10:47:50 +00:00
// it is up to an extension mechanism (opera widget, etc)
2010-03-11 21:07:13 +00:00
// to call setCustomHandlers() which will make it do something
2013-02-21 20:34:11 +00:00
var clickOpen = function ( ) {
2010-03-11 21:07:13 +00:00
svgCanvas . open ( ) ;
} ;
2013-02-20 21:40:21 +00:00
2013-02-21 20:34:11 +00:00
var clickImport = function ( ) {
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickUndo = function ( ) {
2010-06-21 18:06:21 +00:00
if ( undoMgr . getUndoStackSize ( ) > 0 ) {
undoMgr . undo ( ) ;
2010-03-11 21:07:13 +00:00
populateLayers ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickRedo = function ( ) {
2010-06-21 18:06:21 +00:00
if ( undoMgr . getRedoStackSize ( ) > 0 ) {
undoMgr . redo ( ) ;
2010-03-11 21:07:13 +00:00
populateLayers ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickGroup = function ( ) {
2010-03-11 21:07:13 +00:00
// group
if ( multiselected ) {
svgCanvas . groupSelectedElements ( ) ;
}
// ungroup
2013-02-21 20:34:11 +00:00
else if ( selectedElement ) {
2010-03-11 21:07:13 +00:00
svgCanvas . ungroupSelectedElement ( ) ;
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var clickClone = function ( ) {
svgCanvas . cloneSelectedElements ( 20 , 20 ) ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var clickAlign = function ( ) {
2013-02-23 18:56:45 +00:00
var letter = this . id . replace ( 'tool_align' , '' ) . charAt ( 0 ) ;
2010-03-11 21:07:13 +00:00
svgCanvas . alignSelectedElements ( letter , $ ( '#align_relative_to' ) . val ( ) ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var zoomImage = function ( multiplier ) {
var res = svgCanvas . getResolution ( ) ;
2013-02-21 20:34:11 +00:00
multiplier = multiplier ? res . zoom * multiplier : 1 ;
// setResolution(res.w * multiplier, res.h * multiplier, true);
2010-03-11 21:07:13 +00:00
$ ( '#zoom' ) . val ( multiplier * 100 ) ;
svgCanvas . setZoom ( multiplier ) ;
zoomDone ( ) ;
updateCanvas ( true ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var zoomDone = function ( ) {
updateWireFrame ( ) ;
2013-02-21 20:34:11 +00:00
// updateCanvas(); // necessary?
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var clickWireframe = function ( ) {
2013-02-16 15:49:33 +00:00
$ ( '#tool_wireframe' ) . toggleClass ( 'push_button_pressed tool_button' ) ;
2010-03-11 21:07:13 +00:00
workarea . toggleClass ( 'wireframe' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( supportsNonSS ) return ;
2010-03-11 21:07:13 +00:00
var wf _rules = $ ( '#wireframe_rules' ) ;
2013-02-15 21:57:35 +00:00
if ( ! wf _rules . length ) {
2010-03-11 21:07:13 +00:00
wf _rules = $ ( '<style id="wireframe_rules"><\/style>' ) . appendTo ( 'head' ) ;
} else {
wf _rules . empty ( ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
updateWireFrame ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var updateWireFrame = function ( ) {
// Test support
2013-02-15 21:57:35 +00:00
if ( supportsNonSS ) return ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
var rule = '#workarea.wireframe #svgcontent * { stroke-width: ' + 1 / svgCanvas . getZoom ( ) + 'px; }' ;
$ ( '#wireframe_rules' ) . text ( workarea . hasClass ( 'wireframe' ) ? rule : '' ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-23 19:22:00 +00:00
var showSourceEditor = function ( e , forSaving ) {
2010-03-11 21:07:13 +00:00
if ( editingsource ) return ;
2012-10-22 10:47:50 +00:00
2013-02-23 18:56:45 +00:00
editingsource = true ;
origSource = svgCanvas . getSvgString ( ) ;
2010-08-23 20:16:27 +00:00
$ ( '#save_output_btns' ) . toggle ( ! ! forSaving ) ;
$ ( '#tool_source_back' ) . toggle ( ! forSaving ) ;
2013-02-23 18:56:45 +00:00
$ ( '#svg_source_textarea' ) . val ( origSource ) ;
2010-03-11 21:07:13 +00:00
$ ( '#svg_source_editor' ) . fadeIn ( ) ;
$ ( '#svg_source_textarea' ) . focus ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#svg_docprops_container, #svg_prefs_container' ) . draggable ( { cancel : 'button,fieldset' , containment : 'window' } ) ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var showDocProperties = function ( ) {
2010-03-11 21:07:13 +00:00
if ( docprops ) return ;
docprops = true ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// This selects the correct radio button by using the array notation
$ ( '#image_save_opts input' ) . val ( [ curPrefs . img _save ] ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// update resolution option with actual resolution
var res = svgCanvas . getResolution ( ) ;
2013-02-15 21:57:35 +00:00
if ( curConfig . baseUnit !== 'px' ) {
2011-02-02 17:28:43 +00:00
res . w = svgedit . units . convertUnit ( res . w ) + curConfig . baseUnit ;
res . h = svgedit . units . convertUnit ( res . h ) + curConfig . baseUnit ;
2010-10-27 18:15:28 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#canvas_width' ) . val ( res . w ) ;
$ ( '#canvas_height' ) . val ( res . h ) ;
2010-04-12 15:59:46 +00:00
$ ( '#canvas_title' ) . val ( svgCanvas . getDocumentTitle ( ) ) ;
2012-10-22 10:47:50 +00:00
2011-01-07 18:37:21 +00:00
$ ( '#svg_docprops' ) . show ( ) ;
2010-10-05 17:13:00 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var showPreferences = function ( ) {
2010-10-05 17:13:00 +00:00
if ( preferences ) return ;
preferences = true ;
$ ( '#main_menu' ) . hide ( ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Update background color with current one
var blocks = $ ( '#bg_blocks div' ) ;
var cur _bg = 'cur_background' ;
var canvas _bg = $ . pref ( 'bkgd_color' ) ;
var url = $ . pref ( 'bkgd_url' ) ;
2013-02-15 21:57:35 +00:00
// if (url) url = url[1];
2010-03-11 21:07:13 +00:00
blocks . each ( function ( ) {
var blk = $ ( this ) ;
var is _bg = blk . css ( 'background-color' ) == canvas _bg ;
blk . toggleClass ( cur _bg , is _bg ) ;
2013-02-15 21:57:35 +00:00
if ( is _bg ) $ ( '#canvas_bg_url' ) . removeClass ( cur _bg ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2013-02-15 21:57:35 +00:00
if ( ! canvas _bg ) blocks . eq ( 0 ) . addClass ( cur _bg ) ;
if ( url ) {
2010-03-11 21:07:13 +00:00
$ ( '#canvas_bg_url' ) . val ( url ) ;
}
2013-02-18 18:02:39 +00:00
$ ( '#grid_snapping_on' ) . prop ( 'checked' , curConfig . gridSnapping ) ;
2013-02-19 21:07:24 +00:00
$ ( '#grid_snapping_step' ) . attr ( 'value' , curConfig . snappingStep ) ;
$ ( '#grid_color' ) . attr ( 'value' , curConfig . gridColor ) ;
2012-10-22 10:47:50 +00:00
2011-01-07 18:37:21 +00:00
$ ( '#svg_prefs' ) . show ( ) ;
2012-10-22 10:47:50 +00:00
} ;
2013-02-21 20:34:11 +00:00
var saveSourceEditor = function ( ) {
2010-03-11 21:07:13 +00:00
if ( ! editingsource ) return ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var saveChanges = function ( ) {
svgCanvas . clearSelection ( ) ;
hideSourceEditor ( ) ;
zoomImage ( ) ;
populateLayers ( ) ;
2010-09-23 19:53:43 +00:00
updateTitle ( ) ;
2010-10-12 15:20:17 +00:00
prepPaints ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
if ( ! svgCanvas . setSvgString ( $ ( '#svg_source_textarea' ) . val ( ) ) ) {
2011-01-03 18:45:36 +00:00
$ . confirm ( uiStrings . notification . QerrorsRevertToSource , function ( ok ) {
2013-02-15 21:57:35 +00:00
if ( ! ok ) return false ;
2010-03-11 21:07:13 +00:00
saveChanges ( ) ;
} ) ;
} else {
saveChanges ( ) ;
}
2012-10-22 10:47:50 +00:00
setSelectMode ( ) ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-09-23 19:53:43 +00:00
var updateTitle = function ( title ) {
title = title || svgCanvas . getDocumentTitle ( ) ;
2013-02-15 22:07:05 +00:00
var newTitle = origTitle + ( title ? ': ' + title : '' ) ;
2012-10-22 10:47:50 +00:00
2010-10-07 20:20:43 +00:00
// Remove title update with current context info, isn't really necessary
2013-02-15 21:57:35 +00:00
// if (cur_context) {
2010-10-07 20:20:43 +00:00
// new_title = new_title + cur_context;
// }
2013-02-15 22:07:05 +00:00
$ ( 'title:first' ) . text ( newTitle ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var saveDocProperties = function ( ) {
2010-03-11 21:07:13 +00:00
// set title
2013-02-15 22:07:05 +00:00
var newTitle = $ ( '#canvas_title' ) . val ( ) ;
updateTitle ( newTitle ) ;
svgCanvas . setDocumentTitle ( newTitle ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// update resolution
var width = $ ( '#canvas_width' ) , w = width . val ( ) ;
var height = $ ( '#canvas_height' ) , h = height . val ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( w != 'fit' && ! svgedit . units . isValidUnit ( 'width' , w ) ) {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . invalidAttrValGiven ) ;
2010-03-11 21:07:13 +00:00
width . parent ( ) . addClass ( 'error' ) ;
return false ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
width . parent ( ) . removeClass ( 'error' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( h != 'fit' && ! svgedit . units . isValidUnit ( 'height' , h ) ) {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . invalidAttrValGiven ) ;
2010-03-11 21:07:13 +00:00
height . parent ( ) . addClass ( 'error' ) ;
return false ;
2012-10-22 10:47:50 +00:00
}
2010-03-11 21:07:13 +00:00
height . parent ( ) . removeClass ( 'error' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! svgCanvas . setResolution ( w , h ) ) {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . noContentToFitTo ) ;
2010-03-11 21:07:13 +00:00
return false ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// set image save option
curPrefs . img _save = $ ( '#image_save_opts :checked' ) . val ( ) ;
$ . pref ( 'img_save' , curPrefs . img _save ) ;
2010-10-05 17:13:00 +00:00
updateCanvas ( ) ;
hideDocProperties ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-10-05 17:13:00 +00:00
var savePreferences = function ( ) {
2010-03-11 21:07:13 +00:00
// set background
var color = $ ( '#bg_blocks div.cur_background' ) . css ( 'background-color' ) || '#FFF' ;
setBackground ( color , $ ( '#canvas_bg_url' ) . val ( ) ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// set language
var lang = $ ( '#lang_select' ) . val ( ) ;
2013-02-15 21:57:35 +00:00
if ( lang != curPrefs . lang ) {
2010-03-16 19:28:57 +00:00
Editor . putLocale ( lang ) ;
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// set icon size
setIconSize ( $ ( '#iconsize' ) . val ( ) ) ;
2012-10-22 10:47:50 +00:00
2010-09-09 12:46:34 +00:00
// set grid setting
2010-09-10 13:23:31 +00:00
curConfig . gridSnapping = $ ( '#grid_snapping_on' ) [ 0 ] . checked ;
2010-09-09 12:46:34 +00:00
curConfig . snappingStep = $ ( '#grid_snapping_step' ) . val ( ) ;
2013-02-19 21:07:24 +00:00
curConfig . gridColor = $ ( '#grid_color' ) . val ( ) ;
2010-10-01 18:59:12 +00:00
curConfig . showRulers = $ ( '#show_rulers' ) [ 0 ] . checked ;
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
$ ( '#rulers' ) . toggle ( curConfig . showRulers ) ;
2013-02-15 21:57:35 +00:00
if ( curConfig . showRulers ) updateRulers ( ) ;
2010-10-05 17:13:00 +00:00
curConfig . baseUnit = $ ( '#base_unit' ) . val ( ) ;
2012-10-22 10:47:50 +00:00
2010-09-17 19:34:23 +00:00
svgCanvas . setConfig ( curConfig ) ;
2010-09-09 12:46:34 +00:00
2010-03-11 21:07:13 +00:00
updateCanvas ( ) ;
2010-10-05 17:13:00 +00:00
hidePreferences ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
function setBackground ( color , url ) {
2013-02-23 18:56:45 +00:00
// if (color == curPrefs.bkgd_color && url == curPrefs.bkgd_url) return;
2010-03-11 21:07:13 +00:00
$ . pref ( 'bkgd_color' , color ) ;
$ . pref ( 'bkgd_url' , url ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// This should be done in svgcanvas.js for the borderRect fill
svgCanvas . setBackground ( color , url ) ;
}
2012-10-22 10:47:50 +00:00
2010-06-03 07:30:56 +00:00
var setIcon = Editor . setIcon = function ( elem , icon _id , forcedSize ) {
2010-09-29 17:36:29 +00:00
var icon = ( typeof icon _id === 'string' ) ? $ . getSvgIcon ( icon _id , true ) : icon _id . clone ( ) ;
2013-02-15 21:57:35 +00:00
if ( ! icon ) {
2010-07-21 17:12:57 +00:00
console . log ( 'NOTE: Icon image missing: ' + icon _id ) ;
return ;
}
2010-09-20 14:59:00 +00:00
$ ( elem ) . empty ( ) . append ( icon ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-15 22:32:35 +00:00
var uaPrefix = ( function ( ) {
2010-07-27 13:37:52 +00:00
var regex = /^(Moz|Webkit|Khtml|O|ms|Icab)(?=[A-Z])/ ;
var someScript = document . getElementsByTagName ( 'script' ) [ 0 ] ;
2013-02-15 21:57:35 +00:00
for ( var prop in someScript . style ) {
if ( regex . test ( prop ) ) {
2010-07-27 13:37:52 +00:00
// test is faster than match, so it's better to perform
// that on the lot and match only when necessary
return prop . match ( regex ) [ 0 ] ;
2010-05-24 19:46:32 +00:00
}
2010-05-12 15:45:43 +00:00
}
2010-07-27 13:37:52 +00:00
// Nothing found so far?
2013-02-15 21:57:35 +00:00
if ( 'WebkitOpacity' in someScript . style ) return 'Webkit' ;
if ( 'KhtmlOpacity' in someScript . style ) return 'Khtml' ;
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
return '' ;
2013-02-15 22:32:35 +00:00
} ) ( ) ;
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
var scaleElements = function ( elems , scale ) {
2013-02-15 22:32:35 +00:00
var prefix = '-' + uaPrefix . toLowerCase ( ) + '-' ;
2010-07-27 13:37:52 +00:00
var sides = [ 'top' , 'left' , 'bottom' , 'right' ] ;
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
elems . each ( function ( ) {
// Handled in CSS
2013-02-15 22:32:35 +00:00
// this.style[uaPrefix + 'Transform'] = 'scale(' + scale + ')';
2010-07-27 13:37:52 +00:00
var el = $ ( this ) ;
var w = el . outerWidth ( ) * ( scale - 1 ) ;
var h = el . outerHeight ( ) * ( scale - 1 ) ;
var margins = { } ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
for ( var i = 0 ; i < 4 ; i ++ ) {
2010-07-27 13:37:52 +00:00
var s = sides [ i ] ;
var cur = el . data ( 'orig_margin-' + s ) ;
2013-02-15 21:57:35 +00:00
if ( cur == null ) {
2013-02-23 18:56:45 +00:00
cur = parseInt ( el . css ( 'margin-' + s ) , 10 ) ;
2010-07-27 13:37:52 +00:00
// Cache the original margin
el . data ( 'orig_margin-' + s , cur ) ;
}
var val = cur * scale ;
2013-02-15 21:57:35 +00:00
if ( s === 'right' ) {
2010-07-27 13:37:52 +00:00
val += w ;
2013-02-15 21:57:35 +00:00
} else if ( s === 'bottom' ) {
2010-07-27 13:37:52 +00:00
val += h ;
}
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
el . css ( 'margin-' + s , val ) ;
2013-02-21 20:34:11 +00:00
// el.css('outline', '1px solid red');
2010-07-27 13:37:52 +00:00
}
} ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-24 20:13:13 +00:00
var setIconSize = Editor . setIconSize = function ( size , force ) {
2013-02-15 21:57:35 +00:00
if ( size == curPrefs . size && ! force ) return ;
2013-02-23 18:56:45 +00:00
2010-07-27 13:37:52 +00:00
// var elems = $('.tool_button, .push_button, .tool_button_current, .disabled, .icon_label, #url_notice, #tool_open');
var sel _toscale = ' # tools _top . toolset , # editor _panel > * , # history _panel > * , \
# main _button , # tools _left > * , # path _node _panel > * , # multiselected _panel > * , \
# g _panel > * , # tool _font _size > * , . tools _flyout ' ;
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
var elems = $ ( sel _toscale ) ;
var scale = 1 ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( typeof size == 'number' ) {
2010-07-27 13:37:52 +00:00
scale = size ;
} else {
var icon _sizes = { s : . 75 , m : 1 , l : 1.25 , xl : 1.5 } ;
scale = icon _sizes [ size ] ;
}
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
Editor . tool _scale = tool _scale = scale ;
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
setFlyoutPositions ( ) ;
// $('.tools_flyout').each(function() {
// var pos = $(this).position();
2013-10-13 23:23:08 +00:00
// console.log($(this), pos.left+(34 * scale));
2010-07-27 13:37:52 +00:00
// $(this).css({'left': pos.left+(34 * scale), 'top': pos.top+(77 * scale)});
// console.log('l', $(this).css('left'));
// });
2013-02-23 18:56:45 +00:00
// var scale = .75;
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
var hidden _ps = elems . parents ( ':hidden' ) ;
hidden _ps . css ( 'visibility' , 'hidden' ) . show ( ) ;
scaleElements ( elems , scale ) ;
hidden _ps . css ( 'visibility' , 'visible' ) . hide ( ) ;
// return;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ . pref ( 'iconsize' , size ) ;
$ ( '#iconsize' ) . val ( size ) ;
2012-10-22 10:47:50 +00:00
2010-07-27 13:37:52 +00:00
// Change icon size
// $('.tool_button, .push_button, .tool_button_current, .disabled, .icon_label, #url_notice, #tool_open')
// .find('> svg, > img').each(function() {
// this.setAttribute('width',size_num);
// this.setAttribute('height',size_num);
// });
2012-10-22 10:47:50 +00:00
//
2010-07-27 13:37:52 +00:00
// $.resizeSvgIcons({
// '.flyout_arrow_horiz > svg, .flyout_arrow_horiz > img': size_num / 5,
// '#logo > svg, #logo > img': size_num * 1.3,
// '#tools_bottom .icon_label > *': (size_num === 16 ? 18 : size_num * .75)
// });
2013-02-15 21:57:35 +00:00
// if (size != 's') {
2010-07-27 13:37:52 +00:00
// $.resizeSvgIcons({'#layerbuttons svg, #layerbuttons img': size_num * .6});
// }
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Note that all rules will be prefixed with '#svg_editor' when parsed
var cssResizeRules = {
2013-02-15 21:57:35 +00:00
// '.tool_button,\
2010-07-27 13:37:52 +00:00
// .push_button,\
// .tool_button_current,\
// .push_button_pressed,\
// .disabled,\
// .icon_label,\
2013-02-15 21:57:35 +00:00
// .tools_flyout .tool_button': {
2010-07-27 13:37:52 +00:00
// 'width': {s: '16px', l: '32px', xl: '48px'},
// 'height': {s: '16px', l: '32px', xl: '48px'},
// 'padding': {s: '1px', l: '2px', xl: '3px'}
// },
2013-02-15 21:57:35 +00:00
// '.tool_sep': {
2010-07-27 13:37:52 +00:00
// 'height': {s: '16px', l: '32px', xl: '48px'},
// 'margin': {s: '2px 2px', l: '2px 5px', xl: '2px 8px'}
// },
2013-02-15 21:57:35 +00:00
// '#main_icon': {
2010-07-27 13:37:52 +00:00
// 'width': {s: '31px', l: '53px', xl: '75px'},
// 'height': {s: '22px', l: '42px', xl: '64px'}
// },
2013-02-15 21:57:35 +00:00
'#tools_top' : {
2010-07-27 13:37:52 +00:00
'left' : 50 ,
'height' : 72
2010-03-11 21:07:13 +00:00
} ,
2013-02-15 21:57:35 +00:00
'#tools_left' : {
2010-07-27 13:37:52 +00:00
'width' : 31 ,
'top' : 74
2010-03-11 21:07:13 +00:00
} ,
2013-02-15 21:57:35 +00:00
'div#workarea' : {
2010-07-27 13:37:52 +00:00
'left' : 38 ,
'top' : 74
2010-10-28 16:47:39 +00:00
}
2013-02-15 21:57:35 +00:00
// '#tools_bottom': {
2010-07-27 13:37:52 +00:00
// 'left': {s: '27px', l: '46px', xl: '65px'},
// 'height': {s: '58px', l: '98px', xl: '145px'}
// },
2013-02-15 21:57:35 +00:00
// '#color_tools': {
2010-07-27 13:37:52 +00:00
// 'border-spacing': {s: '0 1px'},
// 'margin-top': {s: '-1px'}
// },
2013-02-15 21:57:35 +00:00
// '#color_tools .icon_label': {
2010-07-27 13:37:52 +00:00
// 'width': {l:'43px', xl: '60px'}
// },
2013-02-15 21:57:35 +00:00
// '.color_tool': {
2010-07-27 13:37:52 +00:00
// 'height': {s: '20px'}
// },
2013-02-15 21:57:35 +00:00
// '#tool_opacity': {
2010-07-27 13:37:52 +00:00
// 'top': {s: '1px'},
// 'height': {s: 'auto', l:'auto', xl:'auto'}
// },
2013-02-15 21:57:35 +00:00
// '#tools_top input, #tools_bottom input': {
2010-07-27 13:37:52 +00:00
// 'margin-top': {s: '2px', l: '4px', xl: '5px'},
// 'height': {s: 'auto', l: 'auto', xl: 'auto'},
// 'border': {s: '1px solid #555', l: 'auto', xl: 'auto'},
// 'font-size': {s: '.9em', l: '1.2em', xl: '1.4em'}
// },
2013-02-15 21:57:35 +00:00
// '#zoom_panel': {
2010-07-27 13:37:52 +00:00
// 'margin-top': {s: '3px', l: '4px', xl: '5px'}
// },
2013-02-15 21:57:35 +00:00
// '#copyright, #tools_bottom .label': {
2010-07-27 13:37:52 +00:00
// 'font-size': {l: '1.5em', xl: '2em'},
// 'line-height': {s: '15px'}
// },
2013-02-15 21:57:35 +00:00
// '#tools_bottom_2': {
2010-07-27 13:37:52 +00:00
// 'width': {l: '295px', xl: '355px'},
// 'top': {s: '4px'}
// },
2013-02-15 21:57:35 +00:00
// '#tools_top > div, #tools_top': {
2010-07-27 13:37:52 +00:00
// 'line-height': {s: '17px', l: '34px', xl: '50px'}
2012-10-22 10:47:50 +00:00
// },
2013-02-15 21:57:35 +00:00
// '.dropdown button': {
2010-07-27 13:37:52 +00:00
// 'height': {s: '18px', l: '34px', xl: '40px'},
// 'line-height': {s: '18px', l: '34px', xl: '40px'},
// 'margin-top': {s: '3px'}
// },
2013-02-15 21:57:35 +00:00
// '#tools_top label, #tools_bottom label': {
2010-07-27 13:37:52 +00:00
// 'font-size': {s: '1em', l: '1.5em', xl: '2em'},
// 'height': {s: '25px', l: '42px', xl: '64px'}
2012-10-22 10:47:50 +00:00
// },
2013-02-15 21:57:35 +00:00
// 'div.toolset': {
2010-07-27 13:37:52 +00:00
// 'height': {s: '25px', l: '42px', xl: '64px'}
// },
2013-02-15 21:57:35 +00:00
// '#tool_bold, #tool_italic': {
2010-07-27 13:37:52 +00:00
// 'font-size': {s: '1.5em', l: '3em', xl: '4.5em'}
// },
2013-02-15 21:57:35 +00:00
// '#sidepanels': {
2010-07-27 13:37:52 +00:00
// 'top': {s: '50px', l: '88px', xl: '125px'},
// 'bottom': {s: '51px', l: '68px', xl: '65px'}
// },
// '#layerbuttons': {
// 'width': {l: '130px', xl: '175px'},
// 'height': {l: '24px', xl: '30px'}
// },
// '#layerlist': {
// 'width': {l: '128px', xl: '150px'}
2012-10-22 10:47:50 +00:00
// },
2010-07-27 13:37:52 +00:00
// '.layer_button': {
// 'width': {l: '19px', xl: '28px'},
// 'height': {l: '19px', xl: '28px'}
// },
2013-02-15 21:57:35 +00:00
// 'input.spin-button': {
// 'background-image': {l: 'url('images/spinbtn_updn_big.png')', xl: 'url('images/spinbtn_updn_big.png')'},
2010-07-27 13:37:52 +00:00
// 'background-position': {l: '100% -5px', xl: '100% -2px'},
// 'padding-right': {l: '24px', xl: '24px' }
// },
2013-02-15 21:57:35 +00:00
// 'input.spin-button.up': {
2010-07-27 13:37:52 +00:00
// 'background-position': {l: '100% -45px', xl: '100% -42px'}
// },
2013-02-15 21:57:35 +00:00
// 'input.spin-button.down': {
2010-07-27 13:37:52 +00:00
// 'background-position': {l: '100% -85px', xl: '100% -82px'}
// },
2013-02-15 21:57:35 +00:00
// '#position_opts': {
2010-07-27 13:37:52 +00:00
// 'width': {all: (size_num*4) +'px'}
// }
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var rule _elem = $ ( '#tool_size_rules' ) ;
2013-02-15 21:57:35 +00:00
if ( ! rule _elem . length ) {
2010-03-11 21:07:13 +00:00
rule _elem = $ ( '<style id="tool_size_rules"><\/style>' ) . appendTo ( 'head' ) ;
} else {
rule _elem . empty ( ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( size != 'm' ) {
2013-02-15 22:07:05 +00:00
var styleStr = '' ;
2010-03-11 21:07:13 +00:00
$ . each ( cssResizeRules , function ( selector , rules ) {
selector = '#svg_editor ' + selector . replace ( /,/g , ', #svg_editor' ) ;
2013-02-15 22:07:05 +00:00
styleStr += selector + '{' ;
2010-03-11 21:07:13 +00:00
$ . each ( rules , function ( prop , values ) {
2013-02-15 21:57:35 +00:00
if ( typeof values === 'number' ) {
2010-07-27 13:37:52 +00:00
var val = ( values * scale ) + 'px' ;
2013-02-15 21:57:35 +00:00
} else if ( values [ size ] || values . all ) {
2010-07-27 13:37:52 +00:00
var val = ( values [ size ] || values . all ) ;
2010-03-11 21:07:13 +00:00
}
2013-02-15 22:07:05 +00:00
styleStr += ( prop + ':' + val + ';' ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2013-02-15 22:07:05 +00:00
styleStr += '}' ;
2010-03-11 21:07:13 +00:00
} ) ;
2013-02-15 22:32:35 +00:00
//this.style[uaPrefix + 'Transform'] = 'scale(' + scale + ')';
var prefix = '-' + uaPrefix . toLowerCase ( ) + '-' ;
2013-02-15 22:07:05 +00:00
styleStr += ( sel _toscale + '{' + prefix + 'transform: scale(' + scale + ');}'
2010-07-27 13:37:52 +00:00
+ ' #svg_editor div.toolset .toolset {' + prefix + 'transform: scale(1); margin: 1px !important;}' // Hack for markers
+ ' #svg_editor .ui-slider {' + prefix + 'transform: scale(' + ( 1 / scale ) + ');}' // Hack for sliders
) ;
2013-02-15 22:07:05 +00:00
rule _elem . text ( styleStr ) ;
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
setFlyoutPositions ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var cancelOverlays = function ( ) {
$ ( '#dialog_box' ) . hide ( ) ;
2010-10-05 17:13:00 +00:00
if ( ! editingsource && ! docprops && ! preferences ) {
2013-02-15 21:57:35 +00:00
if ( cur _context ) {
2010-09-23 19:53:43 +00:00
svgCanvas . leaveContext ( ) ;
}
return ;
2013-02-15 13:38:55 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
if ( editingsource ) {
2013-02-15 22:07:05 +00:00
if ( origSource !== $ ( '#svg_source_textarea' ) . val ( ) ) {
2011-01-03 18:45:36 +00:00
$ . confirm ( uiStrings . notification . QignoreSourceChanges , function ( ok ) {
2013-02-15 21:57:35 +00:00
if ( ok ) hideSourceEditor ( ) ;
2010-03-11 21:07:13 +00:00
} ) ;
} else {
hideSourceEditor ( ) ;
}
2013-02-15 13:38:55 +00:00
} else if ( docprops ) {
2010-03-11 21:07:13 +00:00
hideDocProperties ( ) ;
2010-10-05 17:13:00 +00:00
} else if ( preferences ) {
hidePreferences ( ) ;
2010-03-11 21:07:13 +00:00
}
2011-02-23 20:06:33 +00:00
resetScrollPos ( ) ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var hideSourceEditor = function ( ) {
2010-03-11 21:07:13 +00:00
$ ( '#svg_source_editor' ) . hide ( ) ;
editingsource = false ;
$ ( '#svg_source_textarea' ) . blur ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var hideDocProperties = function ( ) {
2010-03-11 21:07:13 +00:00
$ ( '#svg_docprops' ) . hide ( ) ;
$ ( '#canvas_width,#canvas_height' ) . removeAttr ( 'disabled' ) ;
$ ( '#resolution' ) [ 0 ] . selectedIndex = 0 ;
$ ( '#image_save_opts input' ) . val ( [ curPrefs . img _save ] ) ;
docprops = false ;
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var hidePreferences = function ( ) {
2010-10-05 17:13:00 +00:00
$ ( '#svg_prefs' ) . hide ( ) ;
preferences = false ;
} ;
2010-04-23 19:40:09 +00:00
var win _wh = { width : $ ( window ) . width ( ) , height : $ ( window ) . height ( ) } ;
2011-02-23 20:06:33 +00:00
var resetScrollPos = $ . noop , curScrollPos ;
2012-10-22 10:47:50 +00:00
2011-02-23 17:10:21 +00:00
// Fix for Issue 781: Drawing area jumps to top-left corner on window resize (IE9)
2013-02-15 21:57:35 +00:00
if ( svgedit . browser . isIE ( ) ) {
2011-02-23 17:10:21 +00:00
( function ( ) {
2011-02-23 20:06:33 +00:00
resetScrollPos = function ( ) {
2013-02-23 18:56:45 +00:00
if ( workarea [ 0 ] . scrollLeft === 0 && workarea [ 0 ] . scrollTop === 0 ) {
2011-02-23 20:06:33 +00:00
workarea [ 0 ] . scrollLeft = curScrollPos . left ;
workarea [ 0 ] . scrollTop = curScrollPos . top ;
2011-02-23 17:10:21 +00:00
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2011-02-23 20:06:33 +00:00
curScrollPos = {
2011-02-23 17:10:21 +00:00
left : workarea [ 0 ] . scrollLeft ,
top : workarea [ 0 ] . scrollTop
} ;
2012-10-22 10:47:50 +00:00
2011-02-23 20:06:33 +00:00
$ ( window ) . resize ( resetScrollPos ) ;
2011-02-23 17:10:21 +00:00
svgEditor . ready ( function ( ) {
// TODO: Find better way to detect when to do this to minimize
// flickering effect
setTimeout ( function ( ) {
2011-02-23 20:06:33 +00:00
resetScrollPos ( ) ;
2011-02-23 17:10:21 +00:00
} , 500 ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2011-02-23 17:10:21 +00:00
workarea . scroll ( function ( ) {
2011-02-23 20:06:33 +00:00
curScrollPos = {
2011-02-23 17:10:21 +00:00
left : workarea [ 0 ] . scrollLeft ,
top : workarea [ 0 ] . scrollTop
} ;
} ) ;
} ( ) ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( window ) . resize ( function ( evt ) {
2010-04-23 19:40:09 +00:00
$ . each ( win _wh , function ( type , val ) {
var curval = $ ( window ) [ type ] ( ) ;
2013-02-15 21:57:35 +00:00
workarea [ 0 ] [ 'scroll' + ( type === 'width' ? 'Left' : 'Top' ) ] -= ( curval - val ) / 2 ;
2010-04-23 19:40:09 +00:00
win _wh [ type ] = curval ;
} ) ;
2013-10-07 01:35:01 +00:00
setFlyoutPositions ( ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
( function ( ) {
workarea . scroll ( function ( ) {
2013-10-13 23:23:08 +00:00
// TODO: jQuery's scrollLeft/Top() wouldn't require a null check
2011-07-07 17:36:58 +00:00
if ( $ ( '#ruler_x' ) . length != 0 ) {
$ ( '#ruler_x' ) [ 0 ] . scrollLeft = workarea [ 0 ] . scrollLeft ;
}
if ( $ ( '#ruler_y' ) . length != 0 ) {
2012-10-22 10:47:50 +00:00
$ ( '#ruler_y' ) [ 0 ] . scrollTop = workarea [ 0 ] . scrollTop ;
2011-07-07 17:36:58 +00:00
}
2010-10-01 18:59:12 +00:00
} ) ;
} ( ) ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#url_notice' ) . click ( function ( ) {
$ . alert ( this . title ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#change_image_url' ) . click ( promptImgURL ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
function promptImgURL ( ) {
2010-08-16 17:53:15 +00:00
var curhref = svgCanvas . getHref ( selectedElement ) ;
2013-02-15 21:57:35 +00:00
curhref = curhref . indexOf ( 'data:' ) === 0 ? '' : curhref ;
2011-01-03 18:45:36 +00:00
$ . prompt ( uiStrings . notification . enterNewImgURL , curhref , function ( url ) {
2013-02-15 21:57:35 +00:00
if ( url ) setImageURL ( url ) ;
2010-03-11 21:07:13 +00:00
} ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// added these event handlers for all the push buttons so they
// behave more like buttons being pressed-in and not images
( function ( ) {
2013-02-15 21:57:35 +00:00
var toolnames = [ 'clear' , 'open' , 'save' , 'source' , 'delete' , 'delete_multi' , 'paste' , 'clone' , 'clone_multi' , 'move_top' , 'move_bottom' ] ;
2010-03-11 21:07:13 +00:00
var all _tools = '' ;
var cur _class = 'tool_button_current' ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ . each ( toolnames , function ( i , item ) {
2013-02-15 21:57:35 +00:00
all _tools += '#tool_' + item + ( i == toolnames . length - 1 ? ',' : '' ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( all _tools ) . mousedown ( function ( ) {
$ ( this ) . addClass ( cur _class ) ;
} ) . bind ( 'mousedown mouseout' , function ( ) {
$ ( this ) . removeClass ( cur _class ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#tool_undo, #tool_redo' ) . mousedown ( function ( ) {
2010-03-11 21:07:13 +00:00
if ( ! $ ( this ) . hasClass ( 'disabled' ) ) $ ( this ) . addClass ( cur _class ) ;
2013-02-15 21:57:35 +00:00
} ) . bind ( 'mousedown mouseout' , function ( ) {
2010-03-11 21:07:13 +00:00
$ ( this ) . removeClass ( cur _class ) ; }
) ;
} ( ) ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// switch modifier key in tooltips if mac
// NOTE: This code is not used yet until I can figure out how to successfully bind ctrl/meta
// in Opera and Chrome
2013-02-18 18:22:00 +00:00
if ( svgedit . browser . isMac ( ) && ! window . opera ) {
2013-02-15 21:57:35 +00:00
var shortcutButtons = [ 'tool_clear' , 'tool_save' , 'tool_source' , 'tool_undo' , 'tool_redo' , 'tool_clone' ] ;
2010-03-11 21:07:13 +00:00
var i = shortcutButtons . length ;
while ( i -- ) {
var button = document . getElementById ( shortcutButtons [ i ] ) ;
2013-02-23 18:56:45 +00:00
if ( button ) {
2010-08-21 01:08:24 +00:00
var title = button . title ;
2013-02-15 21:57:35 +00:00
var index = title . indexOf ( 'Ctrl+' ) ;
button . title = [ title . substr ( 0 , index ) , 'Cmd+' , title . substr ( index + 5 ) ] . join ( '' ) ;
2010-08-21 01:08:24 +00:00
}
2010-03-11 21:07:13 +00:00
}
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// TODO: go back to the color boxes having white background-color and then setting
2013-10-13 23:23:08 +00:00
// background-image to none.png (otherwise partially transparent gradients look weird)
2010-03-11 21:07:13 +00:00
var colorPicker = function ( elem ) {
var picker = elem . attr ( 'id' ) == 'stroke_color' ? 'stroke' : 'fill' ;
2010-03-12 20:43:33 +00:00
// var opacity = (picker == 'stroke' ? $('#stroke_opacity') : $('#fill_opacity'));
2010-10-12 18:35:45 +00:00
var paint = paintBox [ picker ] . paint ;
2010-03-11 21:07:13 +00:00
var title = ( picker == 'stroke' ? 'Pick a Stroke Paint and Opacity' : 'Pick a Fill Paint and Opacity' ) ;
var was _none = false ;
2012-11-28 02:08:13 +00:00
var pos = elem . offset ( ) ;
2013-02-15 21:57:35 +00:00
$ ( '#color_picker' )
. draggable ( { cancel : '.jGraduate_tabs, .jGraduate_colPick, .jGraduate_gradPick, .jPicker' , containment : 'window' } )
2012-11-28 02:08:13 +00:00
. css ( curConfig . colorPickerCSS || { 'left' : pos . left - 140 , 'bottom' : 40 } )
2010-03-11 21:07:13 +00:00
. jGraduate (
2012-10-22 10:47:50 +00:00
{
2010-03-11 21:07:13 +00:00
paint : paint ,
window : { pickerTitle : title } ,
2011-01-07 17:35:05 +00:00
images : { clientPath : curConfig . jGraduatePath } ,
newstop : 'inverse'
2010-03-11 21:07:13 +00:00
} ,
function ( p ) {
paint = new $ . jGraduate . Paint ( p ) ;
2010-10-12 18:35:45 +00:00
paintBox [ picker ] . setPaint ( paint ) ;
svgCanvas . setPaint ( picker , paint ) ;
2010-03-11 21:07:13 +00:00
$ ( '#color_picker' ) . hide ( ) ;
} ,
2013-02-19 15:00:56 +00:00
function ( ) {
2010-03-11 21:07:13 +00:00
$ ( '#color_picker' ) . hide ( ) ;
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var updateToolButtonState = function ( ) {
2010-06-28 20:09:34 +00:00
var bNoFill = ( svgCanvas . getColor ( 'fill' ) == 'none' ) ;
var bNoStroke = ( svgCanvas . getColor ( 'stroke' ) == 'none' ) ;
2010-03-11 21:07:13 +00:00
var buttonsNeedingStroke = [ '#tool_fhpath' , '#tool_line' ] ;
var buttonsNeedingFillAndStroke = [ '#tools_rect .tool_button' , '#tools_ellipse .tool_button' , '#tool_text' , '#tool_path' ] ;
if ( bNoStroke ) {
2010-10-28 16:04:35 +00:00
for ( var index in buttonsNeedingStroke ) {
2010-03-11 21:07:13 +00:00
var button = buttonsNeedingStroke [ index ] ;
if ( $ ( button ) . hasClass ( 'tool_button_current' ) ) {
clickSelect ( ) ;
}
$ ( button ) . addClass ( 'disabled' ) ;
}
2013-02-15 21:57:35 +00:00
} else {
2010-10-28 16:04:35 +00:00
for ( var index in buttonsNeedingStroke ) {
2010-03-11 21:07:13 +00:00
var button = buttonsNeedingStroke [ index ] ;
$ ( button ) . removeClass ( 'disabled' ) ;
}
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
if ( bNoStroke && bNoFill ) {
2010-10-28 16:04:35 +00:00
for ( var index in buttonsNeedingFillAndStroke ) {
2010-03-11 21:07:13 +00:00
var button = buttonsNeedingFillAndStroke [ index ] ;
if ( $ ( button ) . hasClass ( 'tool_button_current' ) ) {
clickSelect ( ) ;
}
$ ( button ) . addClass ( 'disabled' ) ;
}
2013-02-15 21:57:35 +00:00
} else {
2010-10-28 16:04:35 +00:00
for ( var index in buttonsNeedingFillAndStroke ) {
2010-03-11 21:07:13 +00:00
var button = buttonsNeedingFillAndStroke [ index ] ;
$ ( button ) . removeClass ( 'disabled' ) ;
}
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
svgCanvas . runExtensions ( 'toolButtonStateUpdate' , {
2010-03-11 21:07:13 +00:00
nofill : bNoFill ,
nostroke : bNoStroke
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Disable flyouts if all inside are disabled
$ ( '.tools_flyout' ) . each ( function ( ) {
var shower = $ ( '#' + this . id + '_show' ) ;
var has _enabled = false ;
$ ( this ) . children ( ) . each ( function ( ) {
2013-02-15 21:57:35 +00:00
if ( ! $ ( this ) . hasClass ( 'disabled' ) ) {
2010-03-11 21:07:13 +00:00
has _enabled = true ;
}
} ) ;
shower . toggleClass ( 'disabled' , ! has _enabled ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
operaRepaint ( ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
var PaintBox = function ( container , type ) {
var cur = curConfig [ type === 'fill' ? 'initFill' : 'initStroke' ] ;
// set up gradients to be used for the buttons
var svgdocbox = new DOMParser ( ) . parseFromString (
2012-11-28 04:32:59 +00:00
' < svg xmlns = "http://www.w3.org/2000/svg" > < rect width = "16.5" height = "16.5" \
2010-10-12 18:35:45 +00:00
fill = "#' + cur.color + '" opacity = "' + cur.opacity + '" / > \
< defs > < linearGradient id = "gradbox_" / > < / d e f s > < / s v g > ' , ' t e x t / x m l ' ) ;
var docElem = svgdocbox . documentElement ;
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
docElem = $ ( container ) [ 0 ] . appendChild ( document . importNode ( docElem , true ) ) ;
docElem . setAttribute ( 'width' , 16.5 ) ;
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
this . rect = docElem . firstChild ;
this . defs = docElem . getElementsByTagName ( 'defs' ) [ 0 ] ;
this . grad = this . defs . firstChild ;
this . paint = new $ . jGraduate . Paint ( { solidColor : cur . color } ) ;
this . type = type ;
this . setPaint = function ( paint , apply ) {
this . paint = paint ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
var fillAttr = 'none' ;
2010-10-12 18:35:45 +00:00
var ptype = paint . type ;
var opac = paint . alpha / 100 ;
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
switch ( ptype ) {
case 'solidColor' :
2013-02-15 21:57:35 +00:00
fillAttr = ( paint [ ptype ] != 'none' ) ? '#' + paint [ ptype ] : paint [ ptype ] ;
2010-10-12 18:35:45 +00:00
break ;
case 'linearGradient' :
case 'radialGradient' :
this . defs . removeChild ( this . grad ) ;
this . grad = this . defs . appendChild ( paint [ ptype ] ) ;
var id = this . grad . id = 'gradbox_' + this . type ;
2013-02-15 21:57:35 +00:00
fillAttr = 'url(#' + id + ')' ;
2010-10-12 18:35:45 +00:00
}
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
this . rect . setAttribute ( 'fill' , fillAttr ) ;
this . rect . setAttribute ( 'opacity' , opac ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( apply ) {
2010-10-12 18:35:45 +00:00
svgCanvas . setColor ( this . type , paintColor , true ) ;
svgCanvas . setPaintOpacity ( this . type , paintOpacity , true ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-10-12 20:09:37 +00:00
this . update = function ( apply ) {
2013-02-15 21:57:35 +00:00
if ( ! selectedElement ) return ;
2010-10-12 20:09:37 +00:00
var type = this . type ;
2012-10-22 10:47:50 +00:00
2011-03-09 20:00:49 +00:00
switch ( selectedElement . tagName ) {
case 'use' :
case 'image' :
case 'foreignObject' :
2012-10-22 10:47:50 +00:00
// These elements don't have fill or stroke, so don't change
2011-03-09 20:00:49 +00:00
// the current value
return ;
case 'g' :
case 'a' :
2010-12-20 21:37:04 +00:00
var gPaint = null ;
2012-10-22 10:47:50 +00:00
2010-12-20 21:37:04 +00:00
var childs = selectedElement . getElementsByTagName ( '*' ) ;
2013-02-15 21:57:35 +00:00
for ( var i = 0 , len = childs . length ; i < len ; i ++ ) {
2010-12-20 21:37:04 +00:00
var elem = childs [ i ] ;
var p = elem . getAttribute ( type ) ;
2013-02-15 21:57:35 +00:00
if ( i === 0 ) {
2010-12-20 21:37:04 +00:00
gPaint = p ;
2013-02-15 21:57:35 +00:00
} else if ( gPaint !== p ) {
2010-12-20 21:37:04 +00:00
gPaint = null ;
2010-12-21 14:34:54 +00:00
break ;
2010-12-20 21:37:04 +00:00
}
}
2013-02-15 21:57:35 +00:00
if ( gPaint === null ) {
2010-12-20 21:37:04 +00:00
// No common color, don't update anything
2010-12-21 14:34:54 +00:00
var paintColor = null ;
2010-12-20 21:37:04 +00:00
return ;
}
var paintColor = gPaint ;
var paintOpacity = 1 ;
2011-03-09 20:00:49 +00:00
break ;
default :
2013-02-15 21:57:35 +00:00
var paintOpacity = parseFloat ( selectedElement . getAttribute ( type + '-opacity' ) ) ;
2010-12-20 21:37:04 +00:00
if ( isNaN ( paintOpacity ) ) {
paintOpacity = 1.0 ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
var defColor = type === 'fill' ? 'black' : 'none' ;
2010-12-20 21:37:04 +00:00
var paintColor = selectedElement . getAttribute ( type ) || defColor ;
}
2013-02-15 21:57:35 +00:00
if ( apply ) {
2010-10-12 20:09:37 +00:00
svgCanvas . setColor ( type , paintColor , true ) ;
svgCanvas . setPaintOpacity ( type , paintOpacity , true ) ;
}
2010-12-20 21:37:04 +00:00
2012-10-22 10:47:50 +00:00
paintOpacity *= 100 ;
2010-10-12 20:09:37 +00:00
var paint = getPaint ( paintColor , paintOpacity , type ) ;
// update the rect inside #fill_color/#stroke_color
this . setPaint ( paint ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
this . prep = function ( ) {
var ptype = this . paint . type ;
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
switch ( ptype ) {
case 'linearGradient' :
case 'radialGradient' :
var paint = new $ . jGraduate . Paint ( { copy : this . paint } ) ;
svgCanvas . setPaint ( type , paint ) ;
}
2013-02-15 13:38:55 +00:00
} ;
2010-10-12 18:35:45 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-10-12 18:35:45 +00:00
paintBox . fill = new PaintBox ( '#fill_color' , 'fill' ) ;
paintBox . stroke = new PaintBox ( '#stroke_color' , 'stroke' ) ;
2010-09-15 20:10:43 +00:00
$ ( '#stroke_width' ) . val ( curConfig . initStroke . width ) ;
$ ( '#group_opacity' ) . val ( curConfig . initOpacity * 100 ) ;
2012-10-22 10:47:50 +00:00
2010-09-15 20:10:43 +00:00
// Use this SVG elem to test vectorEffect support
2013-02-15 22:32:35 +00:00
var testEl = paintBox . fill . rect . cloneNode ( false ) ;
2013-02-23 18:56:45 +00:00
testEl . setAttribute ( 'style' , 'vector-effect:non-scaling-stroke' ) ;
2013-02-15 22:32:35 +00:00
var supportsNonSS = ( testEl . style . vectorEffect === 'non-scaling-stroke' ) ;
testEl . removeAttribute ( 'style' ) ;
2010-10-12 18:35:45 +00:00
var svgdocbox = paintBox . fill . rect . ownerDocument ;
2010-09-15 20:10:43 +00:00
// Use this to test support for blur element. Seems to work to test support in Webkit
2013-02-16 15:02:26 +00:00
var blurTest = svgdocbox . createElementNS ( svgedit . NS . SVG , 'feGaussianBlur' ) ;
2013-02-15 22:32:35 +00:00
if ( typeof blurTest . stdDeviationX === 'undefined' ) {
2010-09-15 20:10:43 +00:00
$ ( '#tool_blur' ) . hide ( ) ;
2010-04-06 18:40:10 +00:00
}
2013-02-15 22:32:35 +00:00
$ ( blurTest ) . remove ( ) ;
2012-10-22 10:47:50 +00:00
2010-09-27 18:58:04 +00:00
// Test for zoom icon support
( function ( ) {
2013-02-15 22:32:35 +00:00
var pre = '-' + uaPrefix . toLowerCase ( ) + '-zoom-' ;
2010-09-27 18:58:04 +00:00
var zoom = pre + 'in' ;
workarea . css ( 'cursor' , zoom ) ;
2013-02-15 21:57:35 +00:00
if ( workarea . css ( 'cursor' ) === zoom ) {
2010-09-27 18:58:04 +00:00
zoomInIcon = zoom ;
zoomOutIcon = pre + 'out' ;
}
workarea . css ( 'cursor' , 'auto' ) ;
} ( ) ) ;
2010-03-11 21:07:13 +00:00
// Test for embedImage support (use timeout to not interfere with page load)
setTimeout ( function ( ) {
2010-08-20 14:52:55 +00:00
svgCanvas . embedImage ( 'images/logo.png' , function ( datauri ) {
2013-02-15 21:57:35 +00:00
if ( ! datauri ) {
2010-03-11 21:07:13 +00:00
// Disable option
2013-02-15 22:32:35 +00:00
$ ( '#image_save_opts [value=embed]' ) . attr ( 'disabled' , 'disabled' ) ;
2010-03-11 21:07:13 +00:00
$ ( '#image_save_opts input' ) . val ( [ 'ref' ] ) ;
curPrefs . img _save = 'ref' ;
2013-02-15 22:32:35 +00:00
$ ( '#image_opt_embed' ) . css ( 'color' , '#666' ) . attr ( 'title' , uiStrings . notification . featNotSupported ) ;
2009-11-09 18:59:53 +00:00
}
} ) ;
2013-02-15 13:38:55 +00:00
} , 1000 ) ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
$ ( '#fill_color, #tool_fill .icon_label' ) . click ( function ( ) {
2010-04-20 13:21:51 +00:00
colorPicker ( $ ( '#fill_color' ) ) ;
2010-03-11 21:07:13 +00:00
updateToolButtonState ( ) ;
2009-11-09 18:59:53 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
$ ( '#stroke_color, #tool_stroke .icon_label' ) . click ( function ( ) {
2010-04-20 13:21:51 +00:00
colorPicker ( $ ( '#stroke_color' ) ) ;
2010-03-11 21:07:13 +00:00
updateToolButtonState ( ) ;
2009-11-18 16:50:25 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-04-20 13:21:51 +00:00
$ ( '#group_opacityLabel' ) . click ( function ( ) {
$ ( '#opacity_dropdown button' ) . mousedown ( ) ;
$ ( window ) . mouseup ( ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-04-20 13:21:51 +00:00
$ ( '#zoomLabel' ) . click ( function ( ) {
$ ( '#zoom_dropdown button' ) . mousedown ( ) ;
$ ( window ) . mouseup ( ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-23 19:22:00 +00:00
$ ( '#tool_move_top' ) . mousedown ( function ( evt ) {
2010-03-11 21:07:13 +00:00
$ ( '#tools_stacking' ) . show ( ) ;
evt . preventDefault ( ) ;
} ) ;
2012-10-22 10:47:50 +00:00
$ ( '.layer_button' ) . mousedown ( function ( ) {
2010-03-11 21:07:13 +00:00
$ ( this ) . addClass ( 'layer_buttonpressed' ) ;
} ) . mouseout ( function ( ) {
$ ( this ) . removeClass ( 'layer_buttonpressed' ) ;
} ) . mouseup ( function ( ) {
$ ( this ) . removeClass ( 'layer_buttonpressed' ) ;
} ) ;
2012-10-22 10:47:50 +00:00
$ ( '.push_button' ) . mousedown ( function ( ) {
2010-03-11 21:07:13 +00:00
if ( ! $ ( this ) . hasClass ( 'disabled' ) ) {
$ ( this ) . addClass ( 'push_button_pressed' ) . removeClass ( 'push_button' ) ;
2009-07-30 22:45:59 +00:00
}
2010-03-11 21:07:13 +00:00
} ) . mouseout ( function ( ) {
$ ( this ) . removeClass ( 'push_button_pressed' ) . addClass ( 'push_button' ) ;
} ) . mouseup ( function ( ) {
$ ( this ) . removeClass ( 'push_button_pressed' ) . addClass ( 'push_button' ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-18 18:02:39 +00:00
// ask for a layer name
2010-03-11 21:07:13 +00:00
$ ( '#layer_new' ) . click ( function ( ) {
2011-01-16 23:07:35 +00:00
var i = svgCanvas . getCurrentDrawing ( ) . getNumLayers ( ) ;
2010-09-16 14:04:47 +00:00
do {
2013-02-15 21:57:35 +00:00
var uniqName = uiStrings . layers . layer + ' ' + ++ i ;
2011-01-16 23:18:51 +00:00
} while ( svgCanvas . getCurrentDrawing ( ) . hasLayer ( uniqName ) ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 23:17:56 +00:00
$ . prompt ( uiStrings . notification . enterUniqueLayerName , uniqName , function ( newName ) {
2010-03-11 21:07:13 +00:00
if ( ! newName ) return ;
2011-01-16 23:18:51 +00:00
if ( svgCanvas . getCurrentDrawing ( ) . hasLayer ( newName ) ) {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . dupeLayerName ) ;
2010-03-11 21:07:13 +00:00
return ;
}
svgCanvas . createLayer ( newName ) ;
updateContextPanel ( ) ;
populateLayers ( ) ;
} ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-09-16 14:04:47 +00:00
function deleteLayer ( ) {
2010-03-11 21:07:13 +00:00
if ( svgCanvas . deleteCurrentLayer ( ) ) {
updateContextPanel ( ) ;
populateLayers ( ) ;
// This matches what SvgCanvas does
// TODO: make this behavior less brittle (svg-editor should get which
// layer is selected from the canvas and then select that one in the UI)
2013-02-15 21:57:35 +00:00
$ ( '#layerlist tr.layer' ) . removeClass ( 'layersel' ) ;
$ ( '#layerlist tr.layer:first' ) . addClass ( 'layersel' ) ;
2009-07-30 22:45:59 +00:00
}
2010-09-16 14:04:47 +00:00
}
2012-10-22 10:47:50 +00:00
2010-09-16 14:04:47 +00:00
function cloneLayer ( ) {
2011-01-30 17:11:56 +00:00
var name = svgCanvas . getCurrentDrawing ( ) . getCurrentLayerName ( ) + ' copy' ;
2012-10-22 10:47:50 +00:00
2011-01-03 18:45:36 +00:00
$ . prompt ( uiStrings . notification . enterUniqueLayerName , name , function ( newName ) {
2010-09-16 14:04:47 +00:00
if ( ! newName ) return ;
2011-01-16 23:18:51 +00:00
if ( svgCanvas . getCurrentDrawing ( ) . hasLayer ( newName ) ) {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . dupeLayerName ) ;
2010-09-16 14:04:47 +00:00
return ;
}
svgCanvas . cloneLayer ( newName ) ;
updateContextPanel ( ) ;
populateLayers ( ) ;
} ) ;
}
2012-10-22 10:47:50 +00:00
2010-09-16 14:04:47 +00:00
function mergeLayer ( ) {
2013-02-15 21:57:35 +00:00
if ( $ ( '#layerlist tr.layersel' ) . index ( ) == svgCanvas . getCurrentDrawing ( ) . getNumLayers ( ) - 1 ) return ;
2010-09-16 14:04:47 +00:00
svgCanvas . mergeLayer ( ) ;
updateContextPanel ( ) ;
populateLayers ( ) ;
}
2012-10-22 10:47:50 +00:00
2010-09-16 14:04:47 +00:00
function moveLayer ( pos ) {
var curIndex = $ ( '#layerlist tr.layersel' ) . index ( ) ;
2011-01-16 23:07:35 +00:00
var total = svgCanvas . getCurrentDrawing ( ) . getNumLayers ( ) ;
2013-02-15 21:57:35 +00:00
if ( curIndex > 0 || curIndex < total - 1 ) {
2010-09-16 14:04:47 +00:00
curIndex += pos ;
2010-03-11 21:07:13 +00:00
svgCanvas . setCurrentLayerPosition ( total - curIndex - 1 ) ;
populateLayers ( ) ;
2009-06-17 17:36:00 +00:00
}
2010-09-16 14:04:47 +00:00
}
2012-10-22 10:47:50 +00:00
2010-09-16 14:04:47 +00:00
$ ( '#layer_delete' ) . click ( deleteLayer ) ;
2012-10-22 10:47:50 +00:00
2010-09-16 14:04:47 +00:00
$ ( '#layer_up' ) . click ( function ( ) {
moveLayer ( - 1 ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#layer_down' ) . click ( function ( ) {
2010-09-16 14:04:47 +00:00
moveLayer ( 1 ) ;
2010-02-04 19:06:49 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#layer_rename' ) . click ( function ( ) {
var curIndex = $ ( '#layerlist tr.layersel' ) . prevAll ( ) . length ;
var oldName = $ ( '#layerlist tr.layersel td.layername' ) . text ( ) ;
2013-02-15 21:57:35 +00:00
$ . prompt ( uiStrings . notification . enterNewLayerName , '' , function ( newName ) {
2010-03-11 21:07:13 +00:00
if ( ! newName ) return ;
2011-01-16 23:18:51 +00:00
if ( oldName == newName || svgCanvas . getCurrentDrawing ( ) . hasLayer ( newName ) ) {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . layerHasThatName ) ;
2010-03-11 21:07:13 +00:00
return ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
svgCanvas . renameCurrentLayer ( newName ) ;
populateLayers ( ) ;
} ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var SIDEPANEL _MAXWIDTH = 300 ;
2012-03-31 18:02:08 +00:00
var SIDEPANEL _OPENWIDTH = 150 ;
2010-06-09 18:17:58 +00:00
var sidedrag = - 1 , sidedragging = false , allowmove = false ;
2012-10-22 10:47:50 +00:00
2013-02-20 22:28:45 +00:00
var changeSidePanelWidth = function ( delta ) {
var rulerX = $ ( '#ruler_x' ) ;
$ ( '#sidepanels' ) . width ( '+=' + delta ) ;
$ ( '#layerpanel' ) . width ( '+=' + delta ) ;
rulerX . css ( 'right' , parseInt ( rulerX . css ( 'right' ) , 10 ) + delta ) ;
workarea . css ( 'right' , parseInt ( workarea . css ( 'right' ) , 10 ) + delta ) ;
2013-09-23 18:55:52 +00:00
svgCanvas . runExtensions ( "workareaResized" ) ;
2013-02-20 22:28:45 +00:00
} ;
var resizeSidePanel = function ( evt ) {
2010-06-09 18:17:58 +00:00
if ( ! allowmove ) return ;
2010-04-20 15:53:19 +00:00
if ( sidedrag == - 1 ) return ;
sidedragging = true ;
2013-02-20 22:28:45 +00:00
var deltaX = sidedrag - evt . pageX ;
var sideWidth = $ ( '#sidepanels' ) . width ( ) ;
if ( sideWidth + deltaX > SIDEPANEL _MAXWIDTH ) {
deltaX = SIDEPANEL _MAXWIDTH - sideWidth ;
sideWidth = SIDEPANEL _MAXWIDTH ;
} else if ( sideWidth + deltaX < 2 ) {
deltaX = 2 - sideWidth ;
sideWidth = 2 ;
2010-04-20 15:53:19 +00:00
}
2013-02-20 22:28:45 +00:00
if ( deltaX == 0 ) return ;
sidedrag -= deltaX ;
changeSidePanelWidth ( deltaX ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#sidepanel_handle' )
2010-04-20 15:53:19 +00:00
. mousedown ( function ( evt ) {
sidedrag = evt . pageX ;
2013-02-20 22:28:45 +00:00
$ ( window ) . mousemove ( resizeSidePanel ) ;
2010-06-09 18:17:58 +00:00
allowmove = false ;
// Silly hack for Chrome, which always runs mousemove right after mousedown
setTimeout ( function ( ) {
allowmove = true ;
} , 20 ) ;
2010-04-20 15:53:19 +00:00
} )
2010-03-11 21:07:13 +00:00
. mouseup ( function ( evt ) {
if ( ! sidedragging ) toggleSidePanel ( ) ;
sidedrag = - 1 ;
sidedragging = false ;
} ) ;
2010-04-20 15:53:19 +00:00
$ ( window ) . mouseup ( function ( ) {
sidedrag = - 1 ;
sidedragging = false ;
2013-02-20 22:28:45 +00:00
$ ( '#svg_editor' ) . unbind ( 'mousemove' , resizeSidePanel ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// if width is non-zero, then fully close it, otherwise fully open it
// the optional close argument forces the side panel closed
2013-02-15 22:32:35 +00:00
var toggleSidePanel = function ( close ) {
2013-02-20 22:28:45 +00:00
var w = $ ( '#sidepanels' ) . width ( ) ;
var deltaX = ( w > 2 || close ? 2 : SIDEPANEL _OPENWIDTH ) - w ;
changeSidePanelWidth ( deltaX ) ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// this function highlights the layer passed in (by fading out the other layers)
// if no layer is passed in, this function restores the other layers
var toggleHighlightLayer = function ( layerNameToHighlight ) {
2011-01-16 23:07:35 +00:00
var curNames = new Array ( svgCanvas . getCurrentDrawing ( ) . getNumLayers ( ) ) ;
2011-01-26 03:18:59 +00:00
for ( var i = 0 ; i < curNames . length ; ++ i ) { curNames [ i ] = svgCanvas . getCurrentDrawing ( ) . getLayerName ( i ) ; }
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
if ( layerNameToHighlight ) {
for ( var i = 0 ; i < curNames . length ; ++ i ) {
if ( curNames [ i ] != layerNameToHighlight ) {
2011-01-28 07:39:30 +00:00
svgCanvas . getCurrentDrawing ( ) . setLayerOpacity ( curNames [ i ] , 0.5 ) ;
2010-03-11 21:07:13 +00:00
}
}
2013-02-15 21:57:35 +00:00
} else {
2010-03-11 21:07:13 +00:00
for ( var i = 0 ; i < curNames . length ; ++ i ) {
2011-01-28 07:39:30 +00:00
svgCanvas . getCurrentDrawing ( ) . setLayerOpacity ( curNames [ i ] , 1.0 ) ;
2010-03-11 21:07:13 +00:00
}
}
} ;
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
var populateLayers = function ( ) {
2013-10-12 03:08:26 +00:00
svgCanvas . clearSelection ( ) ;
2013-02-23 19:22:00 +00:00
var layerlist = $ ( '#layerlist tbody' ) . empty ( ) ;
var selLayerNames = $ ( '#selLayerNames' ) . empty ( ) ;
var drawing = svgCanvas . getCurrentDrawing ( ) ;
var currentLayerName = drawing . getCurrentLayerName ( ) ;
2011-01-16 23:07:35 +00:00
var layer = svgCanvas . getCurrentDrawing ( ) . getNumLayers ( ) ;
2010-03-11 21:07:13 +00:00
var icon = $ . getSvgIcon ( 'eye' ) ;
// we get the layers in the reverse z-order (the layer rendered on top is listed first)
while ( layer -- ) {
2013-02-23 19:22:00 +00:00
var name = drawing . getLayerName ( layer ) ;
var layerTr = $ ( '<tr class="layer">' ) . toggleClass ( 'layersel' , name === currentLayerName ) ;
var layerVis = $ ( '<td class="layervis">' ) . toggleClass ( 'layerinvis' , ! drawing . getLayerVisibility ( name ) ) ;
var layerName = $ ( '<td class="layername">' + name + '</td>' ) ;
layerlist . append ( layerTr . append ( layerVis , layerName ) ) ;
selLayerNames . append ( '<option value="' + name + '">' + name + '</option>' ) ;
2010-03-11 21:07:13 +00:00
}
2013-02-15 21:57:35 +00:00
if ( icon !== undefined ) {
2010-03-11 21:07:13 +00:00
var copy = icon . clone ( ) ;
2013-02-23 19:22:00 +00:00
$ ( 'td.layervis' , layerlist ) . append ( icon . clone ( ) ) ;
$ . resizeSvgIcons ( { 'td.layervis .svg_icon' : 14 } ) ;
2010-03-11 21:07:13 +00:00
}
// handle selection of layer
$ ( '#layerlist td.layername' )
2013-02-23 19:22:00 +00:00
. mouseup ( function ( evt ) {
2013-02-15 21:57:35 +00:00
$ ( '#layerlist tr.layer' ) . removeClass ( 'layersel' ) ;
2013-02-23 19:22:00 +00:00
$ ( this . parentNode ) . addClass ( 'layersel' ) ;
2010-03-11 21:07:13 +00:00
svgCanvas . setCurrentLayer ( this . textContent ) ;
evt . preventDefault ( ) ;
} )
2013-02-23 19:22:00 +00:00
. mouseover ( function ( ) {
2010-03-11 21:07:13 +00:00
toggleHighlightLayer ( this . textContent ) ;
} )
2013-02-23 19:22:00 +00:00
. mouseout ( function ( ) {
2010-03-11 21:07:13 +00:00
toggleHighlightLayer ( ) ;
} ) ;
2013-02-23 19:22:00 +00:00
$ ( '#layerlist td.layervis' ) . click ( function ( ) {
2010-03-11 21:07:13 +00:00
var row = $ ( this . parentNode ) . prevAll ( ) . length ;
var name = $ ( '#layerlist tr.layer:eq(' + row + ') td.layername' ) . text ( ) ;
var vis = $ ( this ) . hasClass ( 'layerinvis' ) ;
svgCanvas . setLayerVisibility ( name , vis ) ;
2013-02-16 16:07:19 +00:00
$ ( this ) . toggleClass ( 'layerinvis' ) ;
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// if there were too few rows, let's add a few to make it not so lonely
2010-08-20 14:52:55 +00:00
var num = 5 - $ ( '#layerlist tr.layer' ) . size ( ) ;
2010-03-11 21:07:13 +00:00
while ( num -- > 0 ) {
// FIXME: there must a better way to do this
2013-02-15 21:57:35 +00:00
layerlist . append ( '<tr><td style="color:white">_</td><td/></tr>' ) ;
2010-03-11 21:07:13 +00:00
}
} ;
2009-11-19 15:09:21 +00:00
populateLayers ( ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// function changeResolution(x,y) {
// var zoom = svgCanvas.getResolution().zoom;
// setResolution(x * zoom, y * zoom);
// }
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var centerCanvas = function ( ) {
// this centers the canvas vertically in the workarea (horizontal handled in CSS)
workarea . css ( 'line-height' , workarea . height ( ) + 'px' ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( window ) . bind ( 'load resize' , centerCanvas ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
function stepFontSize ( elem , step ) {
var orig _val = elem . value - 0 ;
var sug _val = orig _val + step ;
var increasing = sug _val >= orig _val ;
2013-02-15 21:57:35 +00:00
if ( step === 0 ) return orig _val ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( orig _val >= 24 ) {
if ( increasing ) {
2010-03-11 21:07:13 +00:00
return Math . round ( orig _val * 1.1 ) ;
}
2013-02-15 22:32:35 +00:00
return Math . round ( orig _val / 1.1 ) ;
2013-02-15 21:57:35 +00:00
} else if ( orig _val <= 1 ) {
if ( increasing ) {
2012-10-22 10:47:50 +00:00
return orig _val * 2 ;
2010-03-11 21:07:13 +00:00
}
2013-02-15 22:32:35 +00:00
return orig _val / 2 ;
2009-10-14 05:11:51 +00:00
}
2013-02-15 22:32:35 +00:00
return sug _val ;
2009-10-14 05:11:51 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
function stepZoom ( elem , step ) {
var orig _val = elem . value - 0 ;
2013-02-15 21:57:35 +00:00
if ( orig _val === 0 ) return 100 ;
2010-03-11 21:07:13 +00:00
var sug _val = orig _val + step ;
2013-02-15 21:57:35 +00:00
if ( step === 0 ) return orig _val ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( orig _val >= 100 ) {
2010-03-11 21:07:13 +00:00
return sug _val ;
}
2013-02-15 22:32:35 +00:00
if ( sug _val >= orig _val ) {
return orig _val * 2 ;
}
return orig _val / 2 ;
2009-09-24 18:59:39 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// function setResolution(w, h, center) {
// updateCanvas();
// // w-=0; h-=0;
// // $('#svgcanvas').css( { 'width': w, 'height': h } );
// // $('#canvas_width').val(w);
// // $('#canvas_height').val(h);
2012-10-22 10:47:50 +00:00
// //
2013-02-15 21:57:35 +00:00
// // if (center) {
2010-03-11 21:07:13 +00:00
// // var w_area = workarea;
// // var scroll_y = h/2 - w_area.height()/2;
// // var scroll_x = w/2 - w_area.width()/2;
// // w_area[0].scrollTop = scroll_y;
// // w_area[0].scrollLeft = scroll_x;
// // }
// }
2012-10-22 10:47:50 +00:00
2013-02-21 20:34:11 +00:00
$ ( '#resolution' ) . change ( function ( ) {
2010-03-11 21:07:13 +00:00
var wh = $ ( '#canvas_width,#canvas_height' ) ;
2013-02-15 21:57:35 +00:00
if ( ! this . selectedIndex ) {
if ( $ ( '#canvas_width' ) . val ( ) == 'fit' ) {
wh . removeAttr ( 'disabled' ) . val ( 100 ) ;
2010-03-11 21:07:13 +00:00
}
2013-02-15 21:57:35 +00:00
} else if ( this . value == 'content' ) {
wh . val ( 'fit' ) . attr ( 'disabled' , 'disabled' ) ;
2010-03-11 21:07:13 +00:00
} else {
var dims = this . value . split ( 'x' ) ;
$ ( '#canvas_width' ) . val ( dims [ 0 ] ) ;
$ ( '#canvas_height' ) . val ( dims [ 1 ] ) ;
2013-02-15 21:57:35 +00:00
wh . removeAttr ( 'disabled' ) ;
2010-03-11 21:07:13 +00:00
}
2009-10-14 05:11:51 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
//Prevent browser from erroneously repopulating fields
2013-02-15 21:57:35 +00:00
$ ( 'input,select' ) . attr ( 'autocomplete' , 'off' ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Associate all button actions as well as non-button keyboard shortcuts
var Actions = function ( ) {
// sel:'selector', fn:function, evt:'event', key:[key, preventDefault, NoDisableInInput]
var tool _buttons = [
2013-02-15 21:57:35 +00:00
{ sel : '#tool_select' , fn : clickSelect , evt : 'click' , key : [ 'V' , true ] } ,
{ sel : '#tool_fhpath' , fn : clickFHPath , evt : 'click' , key : [ 'Q' , true ] } ,
{ sel : '#tool_line' , fn : clickLine , evt : 'click' , key : [ 'L' , true ] } ,
{ sel : '#tool_rect' , fn : clickRect , evt : 'mouseup' , key : [ 'R' , true ] , parent : '#tools_rect' , icon : 'rect' } ,
{ sel : '#tool_square' , fn : clickSquare , evt : 'mouseup' , parent : '#tools_rect' , icon : 'square' } ,
{ sel : '#tool_fhrect' , fn : clickFHRect , evt : 'mouseup' , parent : '#tools_rect' , icon : 'fh_rect' } ,
{ sel : '#tool_ellipse' , fn : clickEllipse , evt : 'mouseup' , key : [ 'E' , true ] , parent : '#tools_ellipse' , icon : 'ellipse' } ,
{ sel : '#tool_circle' , fn : clickCircle , evt : 'mouseup' , parent : '#tools_ellipse' , icon : 'circle' } ,
{ sel : '#tool_fhellipse' , fn : clickFHEllipse , evt : 'mouseup' , parent : '#tools_ellipse' , icon : 'fh_ellipse' } ,
{ sel : '#tool_path' , fn : clickPath , evt : 'click' , key : [ 'P' , true ] } ,
{ sel : '#tool_text' , fn : clickText , evt : 'click' , key : [ 'T' , true ] } ,
{ sel : '#tool_image' , fn : clickImage , evt : 'mouseup' } ,
{ sel : '#tool_zoom' , fn : clickZoom , evt : 'mouseup' , key : [ 'Z' , true ] } ,
{ sel : '#tool_clear' , fn : clickClear , evt : 'mouseup' , key : [ 'N' , true ] } ,
{ sel : '#tool_save' , fn : function ( ) { editingsource ? saveSourceEditor ( ) : clickSave ( ) } , evt : 'mouseup' , key : [ 'S' , true ] } ,
{ sel : '#tool_export' , fn : clickExport , evt : 'mouseup' } ,
{ sel : '#tool_open' , fn : clickOpen , evt : 'mouseup' , key : [ 'O' , true ] } ,
{ sel : '#tool_import' , fn : clickImport , evt : 'mouseup' } ,
{ sel : '#tool_source' , fn : showSourceEditor , evt : 'click' , key : [ 'U' , true ] } ,
{ sel : '#tool_wireframe' , fn : clickWireframe , evt : 'click' , key : [ 'F' , true ] } ,
2013-02-18 18:52:07 +00:00
{ sel : '#tool_source_cancel,.overlay,#tool_docprops_cancel,#tool_prefs_cancel' , fn : cancelOverlays , evt : 'click' , key : [ 'esc' , false , false ] , hidekey : true } ,
2013-02-15 21:57:35 +00:00
{ sel : '#tool_source_save' , fn : saveSourceEditor , evt : 'click' } ,
{ sel : '#tool_docprops_save' , fn : saveDocProperties , evt : 'click' } ,
{ sel : '#tool_docprops' , fn : showDocProperties , evt : 'mouseup' } ,
{ sel : '#tool_prefs_save' , fn : savePreferences , evt : 'click' } ,
2013-02-15 22:32:35 +00:00
{ sel : '#tool_prefs_option' , fn : function ( ) { showPreferences ( ) ; return false ; } , evt : 'mouseup' } ,
2013-02-15 21:57:35 +00:00
{ sel : '#tool_delete,#tool_delete_multi' , fn : deleteSelected , evt : 'click' , key : [ 'del/backspace' , true ] } ,
{ sel : '#tool_reorient' , fn : reorientPath , evt : 'click' } ,
{ sel : '#tool_node_link' , fn : linkControlPoints , evt : 'click' } ,
{ sel : '#tool_node_clone' , fn : clonePathNode , evt : 'click' } ,
{ sel : '#tool_node_delete' , fn : deletePathNode , evt : 'click' } ,
{ sel : '#tool_openclose_path' , fn : opencloseSubPath , evt : 'click' } ,
{ sel : '#tool_add_subpath' , fn : addSubPath , evt : 'click' } ,
{ sel : '#tool_move_top' , fn : moveToTopSelected , evt : 'click' , key : 'ctrl+shift+]' } ,
{ sel : '#tool_move_bottom' , fn : moveToBottomSelected , evt : 'click' , key : 'ctrl+shift+[' } ,
{ sel : '#tool_topath' , fn : convertToPath , evt : 'click' } ,
{ sel : '#tool_make_link,#tool_make_link_multi' , fn : makeHyperlink , evt : 'click' } ,
{ sel : '#tool_undo' , fn : clickUndo , evt : 'click' , key : [ 'Z' , true ] } ,
{ sel : '#tool_redo' , fn : clickRedo , evt : 'click' , key : [ 'Y' , true ] } ,
{ sel : '#tool_clone,#tool_clone_multi' , fn : clickClone , evt : 'click' , key : [ 'D' , true ] } ,
{ sel : '#tool_group' , fn : clickGroup , evt : 'click' , key : [ 'G' , true ] } ,
{ sel : '#tool_ungroup' , fn : clickGroup , evt : 'click' } ,
{ sel : '#tool_unlink_use' , fn : clickGroup , evt : 'click' } ,
{ sel : '[id^=tool_align]' , fn : clickAlign , evt : 'click' } ,
2010-03-11 21:07:13 +00:00
// these two lines are required to make Opera work properly with the flyout mechanism
2013-02-15 21:57:35 +00:00
// {sel: '#tools_rect_show', fn: clickRect, evt: 'click'},
// {sel: '#tools_ellipse_show', fn: clickEllipse, evt: 'click'},
{ sel : '#tool_bold' , fn : clickBold , evt : 'mousedown' } ,
{ sel : '#tool_italic' , fn : clickItalic , evt : 'mousedown' } ,
{ sel : '#sidepanel_handle' , fn : toggleSidePanel , key : [ 'X' ] } ,
{ sel : '#copy_save_done' , fn : cancelOverlays , evt : 'click' } ,
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Shortcuts not associated with buttons
2012-10-22 10:47:50 +00:00
2010-09-25 04:28:57 +00:00
{ key : 'ctrl+left' , fn : function ( ) { rotateSelected ( 0 , 1 ) } } ,
{ key : 'ctrl+right' , fn : function ( ) { rotateSelected ( 1 , 1 ) } } ,
2012-10-22 10:47:50 +00:00
{ key : 'ctrl+shift+left' , fn : function ( ) { rotateSelected ( 0 , 5 ) } } ,
2010-09-25 04:28:57 +00:00
{ key : 'ctrl+shift+right' , fn : function ( ) { rotateSelected ( 1 , 5 ) } } ,
2010-03-11 21:07:13 +00:00
{ key : 'shift+O' , fn : selectPrev } ,
{ key : 'shift+P' , fn : selectNext } ,
2010-08-20 19:27:57 +00:00
{ key : [ modKey + 'up' , true ] , fn : function ( ) { zoomImage ( 2 ) ; } } ,
{ key : [ modKey + 'down' , true ] , fn : function ( ) { zoomImage ( . 5 ) ; } } ,
{ key : [ modKey + ']' , true ] , fn : function ( ) { moveUpDownSelected ( 'Up' ) ; } } ,
2013-02-15 13:38:55 +00:00
{ key : [ modKey + '[' , true ] , fn : function ( ) { moveUpDownSelected ( 'Down' ) ; } } ,
2010-03-11 21:07:13 +00:00
{ key : [ 'up' , true ] , fn : function ( ) { moveSelected ( 0 , - 1 ) ; } } ,
{ key : [ 'down' , true ] , fn : function ( ) { moveSelected ( 0 , 1 ) ; } } ,
{ key : [ 'left' , true ] , fn : function ( ) { moveSelected ( - 1 , 0 ) ; } } ,
{ key : [ 'right' , true ] , fn : function ( ) { moveSelected ( 1 , 0 ) ; } } ,
2010-09-21 19:22:52 +00:00
{ key : 'shift+up' , fn : function ( ) { moveSelected ( 0 , - 10 ) } } ,
{ key : 'shift+down' , fn : function ( ) { moveSelected ( 0 , 10 ) } } ,
{ key : 'shift+left' , fn : function ( ) { moveSelected ( - 10 , 0 ) } } ,
{ key : 'shift+right' , fn : function ( ) { moveSelected ( 10 , 0 ) } } ,
2011-01-30 05:55:33 +00:00
{ key : [ 'alt+up' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( 0 , - 1 ) } } ,
{ key : [ 'alt+down' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( 0 , 1 ) } } ,
{ key : [ 'alt+left' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( - 1 , 0 ) } } ,
{ key : [ 'alt+right' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( 1 , 0 ) } } ,
{ key : [ 'alt+shift+up' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( 0 , - 10 ) } } ,
{ key : [ 'alt+shift+down' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( 0 , 10 ) } } ,
{ key : [ 'alt+shift+left' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( - 10 , 0 ) } } ,
2012-10-22 10:47:50 +00:00
{ key : [ 'alt+shift+right' , true ] , fn : function ( ) { svgCanvas . cloneSelectedElements ( 10 , 0 ) } } ,
2011-01-28 20:11:18 +00:00
{ key : 'A' , fn : function ( ) { svgCanvas . selectAllInCurrentLayer ( ) ; } } ,
// Standard shortcuts
{ key : modKey + 'z' , fn : clickUndo } ,
{ key : modKey + 'shift+z' , fn : clickRedo } ,
{ key : modKey + 'y' , fn : clickRedo } ,
2012-10-22 10:47:50 +00:00
2011-01-28 20:11:18 +00:00
{ key : modKey + 'x' , fn : cutSelected } ,
{ key : modKey + 'c' , fn : copySelected } ,
{ key : modKey + 'v' , fn : pasteInCenter }
2010-03-11 21:07:13 +00:00
] ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Tooltips not directly associated with a single function
var key _assocs = {
'4/Shift+4' : '#tools_rect_show' ,
'5/Shift+5' : '#tools_ellipse_show'
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
return {
setAll : function ( ) {
var flyouts = { } ;
2012-10-22 10:47:50 +00:00
2013-10-13 23:23:08 +00:00
$ . each ( tool _buttons , function ( i , opts ) {
2010-03-11 21:07:13 +00:00
// Bind function to button
2013-02-15 21:57:35 +00:00
if ( opts . sel ) {
2010-03-11 21:07:13 +00:00
var btn = $ ( opts . sel ) ;
2010-07-14 19:15:44 +00:00
if ( btn . length == 0 ) return true ; // Skip if markup does not exist
2013-02-15 13:38:55 +00:00
if ( opts . evt ) {
2013-02-15 22:32:35 +00:00
if ( svgedit . browser . isTouch ( ) && opts . evt === 'click' ) {
opts . evt = 'mousedown' ;
}
2010-03-11 21:07:13 +00:00
btn [ opts . evt ] ( opts . fn ) ;
}
2012-10-22 10:47:50 +00:00
2010-07-14 19:15:44 +00:00
// Add to parent flyout menu, if able to be displayed
2013-02-15 21:57:35 +00:00
if ( opts . parent && $ ( opts . parent + '_show' ) . length != 0 ) {
2010-03-11 21:07:13 +00:00
var f _h = $ ( opts . parent ) ;
2013-02-15 21:57:35 +00:00
if ( ! f _h . length ) {
2010-03-11 21:07:13 +00:00
f _h = makeFlyoutHolder ( opts . parent . substr ( 1 ) ) ;
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
f _h . append ( btn ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! $ . isArray ( flyouts [ opts . parent ] ) ) {
2010-03-11 21:07:13 +00:00
flyouts [ opts . parent ] = [ ] ;
}
flyouts [ opts . parent ] . push ( opts ) ;
}
2010-01-29 18:27:26 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Bind function to shortcut key
2013-02-15 21:57:35 +00:00
if ( opts . key ) {
2010-03-11 21:07:13 +00:00
// Set shortcut based on options
var keyval , shortcut = '' , disInInp = true , fn = opts . fn , pd = false ;
2013-02-15 21:57:35 +00:00
if ( $ . isArray ( opts . key ) ) {
2010-03-11 21:07:13 +00:00
keyval = opts . key [ 0 ] ;
2013-02-15 21:57:35 +00:00
if ( opts . key . length > 1 ) pd = opts . key [ 1 ] ;
if ( opts . key . length > 2 ) disInInp = opts . key [ 2 ] ;
2010-03-11 21:07:13 +00:00
} else {
keyval = opts . key ;
}
keyval += '' ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ . each ( keyval . split ( '/' ) , function ( i , key ) {
2010-04-01 17:38:23 +00:00
$ ( document ) . bind ( 'keydown' , key , function ( e ) {
2010-03-11 21:07:13 +00:00
fn ( ) ;
2013-02-15 21:57:35 +00:00
if ( pd ) {
2010-03-11 21:07:13 +00:00
e . preventDefault ( ) ;
}
// Prevent default on ALL keys?
return false ;
} ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Put shortcut in title
2013-02-15 21:57:35 +00:00
if ( opts . sel && ! opts . hidekey && btn . attr ( 'title' ) ) {
2013-02-15 22:07:05 +00:00
var newTitle = btn . attr ( 'title' ) . split ( '[' ) [ 0 ] + ' (' + keyval + ')' ;
2010-03-11 21:07:13 +00:00
key _assocs [ keyval ] = opts . sel ;
// Disregard for menu items
2013-02-15 21:57:35 +00:00
if ( ! btn . parents ( '#main_menu' ) . length ) {
2013-02-15 22:07:05 +00:00
btn . attr ( 'title' , newTitle ) ;
2010-03-11 21:07:13 +00:00
}
}
2010-01-29 18:27:26 +00:00
}
2010-03-11 21:07:13 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Setup flyouts
setupFlyouts ( flyouts ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Misc additional actions
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
// Make 'return' keypress trigger the change event
2012-10-22 10:47:50 +00:00
$ ( '.attr_changer, #image_url' ) . bind ( 'keydown' , 'return' ,
2010-03-11 21:07:13 +00:00
function ( evt ) { $ ( this ) . change ( ) ; evt . preventDefault ( ) ; }
) ;
2012-10-22 10:47:50 +00:00
2010-09-24 16:05:46 +00:00
$ ( window ) . bind ( 'keydown' , 'tab' , function ( e ) {
2013-02-15 21:57:35 +00:00
if ( ui _context === 'canvas' ) {
2010-09-24 16:05:46 +00:00
e . preventDefault ( ) ;
selectNext ( ) ;
}
} ) . bind ( 'keydown' , 'shift+tab' , function ( e ) {
2013-02-15 21:57:35 +00:00
if ( ui _context === 'canvas' ) {
2010-09-24 16:05:46 +00:00
e . preventDefault ( ) ;
selectPrev ( ) ;
}
} ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( '#tool_zoom' ) . dblclick ( dblclickZoom ) ;
} ,
setTitles : function ( ) {
2013-10-13 23:23:08 +00:00
$ . each ( key _assocs , function ( keyval , sel ) {
2010-03-11 21:07:13 +00:00
var menu = ( $ ( sel ) . parents ( '#main_menu' ) . length ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
$ ( sel ) . each ( function ( ) {
2013-02-15 21:57:35 +00:00
if ( menu ) {
2010-03-11 21:07:13 +00:00
var t = $ ( this ) . text ( ) . split ( ' [' ) [ 0 ] ;
} else {
2012-10-22 10:47:50 +00:00
var t = this . title . split ( ' [' ) [ 0 ] ;
2010-03-11 21:07:13 +00:00
}
var key _str = '' ;
// Shift+Up
$ . each ( keyval . split ( '/' ) , function ( i , key ) {
var mod _bits = key . split ( '+' ) , mod = '' ;
2013-02-15 21:57:35 +00:00
if ( mod _bits . length > 1 ) {
2010-03-11 21:07:13 +00:00
mod = mod _bits [ 0 ] + '+' ;
key = mod _bits [ 1 ] ;
}
key _str += ( i ? '/' : '' ) + mod + ( uiStrings [ 'key_' + key ] || key ) ;
} ) ;
2013-02-15 21:57:35 +00:00
if ( menu ) {
2010-03-11 21:07:13 +00:00
this . lastChild . textContent = t + ' [' + key _str + ']' ;
} else {
this . title = t + ' [' + key _str + ']' ;
2009-12-29 21:20:45 +00:00
}
2009-12-11 20:35:51 +00:00
} ) ;
} ) ;
2010-03-11 21:07:13 +00:00
} ,
getButtonData : function ( sel ) {
var b ;
$ . each ( tool _buttons , function ( i , btn ) {
2013-02-15 21:57:35 +00:00
if ( btn . sel === sel ) b = btn ;
2010-03-11 21:07:13 +00:00
} ) ;
return b ;
}
} ;
} ( ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
Actions . setAll ( ) ;
2012-10-22 10:47:50 +00:00
2010-03-16 19:28:57 +00:00
// Select given tool
Editor . ready ( function ( ) {
2010-10-29 13:13:34 +00:00
var tool ,
itool = curConfig . initTool ,
2013-02-15 21:57:35 +00:00
container = $ ( '#tools_left, #svg_editor .tools_flyout' ) ,
pre _tool = container . find ( '#tool_' + itool ) ,
reg _tool = container . find ( '#' + itool ) ;
if ( pre _tool . length ) {
2010-03-16 19:28:57 +00:00
tool = pre _tool ;
2013-02-23 19:22:00 +00:00
} else if ( reg _tool . length ) {
2010-03-16 19:28:57 +00:00
tool = reg _tool ;
} else {
2013-02-15 21:57:35 +00:00
tool = $ ( '#tool_select' ) ;
2010-03-16 19:28:57 +00:00
}
tool . click ( ) . mouseup ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( curConfig . wireframe ) {
2010-03-16 19:28:57 +00:00
$ ( '#tool_wireframe' ) . click ( ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( curConfig . showlayers ) {
2010-03-16 19:28:57 +00:00
toggleSidePanel ( ) ;
}
2012-10-22 10:47:50 +00:00
2011-01-12 15:10:01 +00:00
$ ( '#rulers' ) . toggle ( ! ! curConfig . showRulers ) ;
2012-10-22 10:47:50 +00:00
2011-06-24 16:28:34 +00:00
if ( curConfig . showRulers ) {
2012-10-22 10:47:50 +00:00
$ ( '#show_rulers' ) [ 0 ] . checked = true ;
2011-06-24 16:28:34 +00:00
}
2013-02-15 21:57:35 +00:00
if ( curConfig . baseUnit ) {
2010-10-05 17:13:00 +00:00
$ ( '#base_unit' ) . val ( curConfig . baseUnit ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-19 21:07:24 +00:00
if ( curConfig . gridSnapping ) {
$ ( '#grid_snapping_on' ) [ 0 ] . checked = true ;
}
2013-02-15 21:57:35 +00:00
if ( curConfig . snappingStep ) {
2010-09-10 13:23:31 +00:00
$ ( '#grid_snapping_step' ) . val ( curConfig . snappingStep ) ;
}
2013-02-19 21:07:24 +00:00
if ( curConfig . gridColor ) {
$ ( '#grid_color' ) . val ( curConfig . gridColor ) ;
}
2010-03-16 19:28:57 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-16 10:28:17 +00:00
// init SpinButtons
$ ( '#rect_rx' ) . SpinButton ( { min : 0 , max : 1000 , callback : changeRectRadius } ) ;
$ ( '#stroke_width' ) . SpinButton ( { min : 0 , max : 99 , smallStep : 0.1 , callback : changeStrokeWidth } ) ;
2010-03-11 21:07:13 +00:00
$ ( '#angle' ) . SpinButton ( { min : - 180 , max : 180 , step : 5 , callback : changeRotationAngle } ) ;
2013-02-16 10:28:17 +00:00
$ ( '#font_size' ) . SpinButton ( { min : 0.001 , stepfunc : stepFontSize , callback : changeFontSize } ) ;
$ ( '#group_opacity' ) . SpinButton ( { min : 0 , max : 100 , step : 5 , callback : changeOpacity } ) ;
$ ( '#blur' ) . SpinButton ( { min : 0 , max : 10 , step : . 1 , callback : changeBlur } ) ;
2011-02-23 20:06:33 +00:00
$ ( '#zoom' ) . SpinButton ( { min : 0.001 , max : 10000 , step : 50 , stepfunc : stepZoom , callback : changeZoom } )
2012-10-22 10:47:50 +00:00
// Set default zoom
2011-02-23 20:06:33 +00:00
. val ( svgCanvas . getZoom ( ) * 100 ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#workarea' ) . contextMenu ( {
2010-08-16 20:26:06 +00:00
menu : 'cmenu_canvas' ,
inSpeed : 0
} ,
function ( action , el , pos ) {
2013-02-15 22:32:35 +00:00
switch ( action ) {
2010-08-16 20:26:06 +00:00
case 'delete' :
deleteSelected ( ) ;
break ;
2010-08-17 18:09:50 +00:00
case 'cut' :
cutSelected ( ) ;
break ;
case 'copy' :
copySelected ( ) ;
break ;
case 'paste' :
svgCanvas . pasteElements ( ) ;
break ;
case 'paste_in_place' :
svgCanvas . pasteElements ( 'in_place' ) ;
break ;
2010-09-24 17:44:34 +00:00
case 'group' :
svgCanvas . groupSelectedElements ( ) ;
break ;
2012-10-22 10:47:50 +00:00
case 'ungroup' :
svgCanvas . ungroupSelectedElement ( ) ;
2010-09-24 17:44:34 +00:00
break ;
2011-01-28 20:11:18 +00:00
case 'move_front' :
moveToTopSelected ( ) ;
break ;
case 'move_up' :
moveUpDownSelected ( 'Up' ) ;
break ;
2010-08-20 14:52:55 +00:00
case 'move_down' :
2010-08-20 19:27:57 +00:00
moveUpDownSelected ( 'Down' ) ;
2010-08-20 14:52:55 +00:00
break ;
2010-09-27 14:20:20 +00:00
case 'move_back' :
moveToBottomSelected ( ) ;
2010-08-20 14:52:55 +00:00
break ;
2013-02-15 13:38:55 +00:00
default :
2013-02-23 19:22:00 +00:00
if ( svgedit . contextmenu && svgedit . contextmenu . hasCustomHandler ( action ) ) {
2013-02-15 13:38:55 +00:00
svgedit . contextmenu . getCustomHandler ( action ) . call ( ) ;
}
break ;
2010-08-17 18:09:50 +00:00
}
2013-02-15 21:57:35 +00:00
if ( svgCanvas . clipBoard . length ) {
2010-08-17 18:09:50 +00:00
canv _menu . enableContextMenuItems ( '#paste,#paste_in_place' ) ;
2010-08-16 20:26:06 +00:00
}
2013-02-15 22:32:35 +00:00
}
) ;
2012-10-22 10:47:50 +00:00
2010-09-16 14:04:47 +00:00
var lmenu _func = function ( action , el , pos ) {
switch ( action ) {
case 'dupe' :
cloneLayer ( ) ;
break ;
case 'delete' :
deleteLayer ( ) ;
break ;
case 'merge_down' :
mergeLayer ( ) ;
break ;
case 'merge_all' :
svgCanvas . mergeAllLayers ( ) ;
updateContextPanel ( ) ;
populateLayers ( ) ;
break ;
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#layerlist' ) . contextMenu ( {
2010-09-16 14:04:47 +00:00
menu : 'cmenu_layers' ,
inSpeed : 0
} ,
lmenu _func
) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ ( '#layer_moreopts' ) . contextMenu ( {
2010-09-16 14:04:47 +00:00
menu : 'cmenu_layers' ,
inSpeed : 0 ,
allowLeft : true
} ,
lmenu _func
) ;
2012-10-22 10:47:50 +00:00
2010-08-17 18:09:50 +00:00
$ ( '.contextMenu li' ) . mousedown ( function ( ev ) {
ev . preventDefault ( ) ;
2013-02-15 13:38:55 +00:00
} ) ;
2012-10-22 10:47:50 +00:00
2010-08-16 20:26:06 +00:00
$ ( '#cmenu_canvas li' ) . disableContextMenu ( ) ;
2010-08-17 18:09:50 +00:00
canv _menu . enableContextMenuItems ( '#delete,#cut,#copy' ) ;
2012-10-22 10:47:50 +00:00
window . onbeforeunload = function ( ) {
2013-01-09 23:14:35 +00:00
if ( 'localStorage' in window ) {
var name = 'svgedit-' + Editor . curConfig . canvasName ;
window . localStorage . setItem ( name , svgCanvas . getSvgString ( ) ) ;
2013-02-16 15:19:20 +00:00
Editor . showSaveWarning = false ;
2013-01-09 23:14:35 +00:00
}
2012-10-22 10:47:50 +00:00
// Suppress warning if page is empty
2013-02-15 21:57:35 +00:00
if ( undoMgr . getUndoStackSize ( ) === 0 ) {
2013-02-16 15:19:20 +00:00
Editor . showSaveWarning = false ;
2010-03-26 17:51:41 +00:00
}
2013-02-16 15:19:20 +00:00
// showSaveWarning is set to 'false' when the page is saved.
if ( ! curConfig . no _save _warning && Editor . showSaveWarning ) {
2010-03-26 17:51:41 +00:00
// Browser already asks question about closing the page
2012-10-22 10:47:50 +00:00
return uiStrings . notification . unsavedChanges ;
2010-03-26 17:51:41 +00:00
}
} ;
2012-10-22 10:47:50 +00:00
2010-07-01 20:14:12 +00:00
Editor . openPrep = function ( func ) {
$ ( '#main_menu' ) . hide ( ) ;
2013-02-15 21:57:35 +00:00
if ( undoMgr . getUndoStackSize ( ) === 0 ) {
2010-07-01 20:14:12 +00:00
func ( true ) ;
} else {
2011-01-03 18:45:36 +00:00
$ . confirm ( uiStrings . notification . QwantToOpen , func ) ;
2010-07-01 20:14:12 +00:00
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// use HTML5 File API: http://www.w3.org/TR/FileAPI/
// if browser has HTML5 File API support, then we will show the open menu item
2013-10-13 23:23:08 +00:00
// and provide a file input to click. When that change event fires, it will
2010-03-11 21:07:13 +00:00
// get the text contents of the file and send it to the canvas
if ( window . FileReader ) {
2013-02-15 21:57:35 +00:00
var importImage = function ( e ) {
2013-10-12 03:08:26 +00:00
$ . process _cancel ( uiStrings . notification . loadingImage ) ;
2012-10-22 10:47:50 +00:00
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
2013-02-15 21:57:35 +00:00
$ ( '#workarea' ) . removeAttr ( 'style' ) ;
2012-10-22 10:47:50 +00:00
$ ( '#main_menu' ) . hide ( ) ;
2013-02-15 21:57:35 +00:00
var file = ( e . type == 'drop' ) ? e . dataTransfer . files [ 0 ] : this . files [ 0 ] ;
if ( ! file ) {
return ;
}
if ( file . type . indexOf ( 'image' ) != - 1 ) {
//detected an image
//svg handling
if ( file . type . indexOf ( 'svg' ) != - 1 ) {
var reader = new FileReader ( ) ;
reader . onloadend = function ( e ) {
svgCanvas . importSvgString ( e . target . result , true ) ;
svgCanvas . ungroupSelectedElement ( ) ;
svgCanvas . ungroupSelectedElement ( ) ;
svgCanvas . groupSelectedElements ( ) ;
svgCanvas . alignSelectedElements ( 'm' , 'page' ) ;
svgCanvas . alignSelectedElements ( 'c' , 'page' ) ;
} ;
reader . readAsText ( file ) ;
} else {
//bitmap handling
var reader = new FileReader ( ) ;
reader . onloadend = function ( e ) {
// let's insert the new image until we know its dimensions
insertNewImage = function ( width , height ) {
var newImage = svgCanvas . addSvgElementFromJson ( {
element : 'image' ,
attr : {
x : 0 ,
y : 0 ,
width : width ,
height : height ,
id : svgCanvas . getNextId ( ) ,
style : 'pointer-events:inherit'
}
} ) ;
svgCanvas . setHref ( newImage , e . target . result ) ;
svgCanvas . selectOnly ( [ newImage ] ) ;
svgCanvas . alignSelectedElements ( 'm' , 'page' ) ;
svgCanvas . alignSelectedElements ( 'c' , 'page' ) ;
updateContextPanel ( ) ;
2012-10-22 10:47:50 +00:00
} ;
2013-02-15 21:57:35 +00:00
// create dummy img so we know the default dimensions
var imgWidth = 100 ;
var imgHeight = 100 ;
var img = new Image ( ) ;
img . src = e . target . result ;
img . style . opacity = 0 ;
img . onload = function ( ) {
imgWidth = img . offsetWidth ;
imgHeight = img . offsetHeight ;
insertNewImage ( imgWidth , imgHeight ) ;
2012-10-22 10:47:50 +00:00
} ;
2013-02-15 21:57:35 +00:00
} ;
reader . readAsDataURL ( file ) ;
2012-10-22 10:47:50 +00:00
}
}
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
function onDragEnter ( e ) {
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
// and indicator should be displayed here, such as "drop files here"
}
function onDragOver ( e ) {
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
}
function onDragLeave ( e ) {
e . stopPropagation ( ) ;
e . preventDefault ( ) ;
// hypothetical indicator should be removed here
}
workarea [ 0 ] . addEventListener ( 'dragenter' , onDragEnter , false ) ;
workarea [ 0 ] . addEventListener ( 'dragover' , onDragOver , false ) ;
workarea [ 0 ] . addEventListener ( 'dragleave' , onDragLeave , false ) ;
2013-02-15 21:57:35 +00:00
workarea [ 0 ] . addEventListener ( 'drop' , importImage , false ) ;
2012-10-22 10:47:50 +00:00
var open = $ ( '<input type="file">' ) . change ( function ( ) {
2010-03-26 18:55:49 +00:00
var f = this ;
2010-07-01 20:14:12 +00:00
Editor . openPrep ( function ( ok ) {
2013-02-15 21:57:35 +00:00
if ( ! ok ) return ;
2010-03-26 18:55:49 +00:00
svgCanvas . clear ( ) ;
2013-02-15 21:57:35 +00:00
if ( f . files . length == 1 ) {
2013-10-12 03:08:26 +00:00
$ . process _cancel ( uiStrings . notification . loadingImage ) ;
2010-03-26 18:55:49 +00:00
var reader = new FileReader ( ) ;
reader . onloadend = function ( e ) {
2010-10-08 17:54:45 +00:00
loadSvgString ( e . target . result ) ;
2010-03-26 18:55:49 +00:00
updateCanvas ( ) ;
} ;
reader . readAsText ( f . files [ 0 ] ) ;
}
2010-07-01 20:14:12 +00:00
} ) ;
2009-12-11 20:35:51 +00:00
} ) ;
2013-02-15 21:57:35 +00:00
$ ( '#tool_open' ) . show ( ) . prepend ( open ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
var imgImport = $ ( '<input type="file">' ) . change ( importImage ) ;
$ ( '#tool_import' ) . show ( ) . prepend ( imgImport ) ;
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2010-07-01 20:14:12 +00:00
var updateCanvas = Editor . updateCanvas = function ( center , new _ctr ) {
2010-03-11 21:07:13 +00:00
var w = workarea . width ( ) , h = workarea . height ( ) ;
var w _orig = w , h _orig = h ;
var zoom = svgCanvas . getZoom ( ) ;
var w _area = workarea ;
2013-02-15 21:57:35 +00:00
var cnvs = $ ( '#svgcanvas' ) ;
2010-03-11 21:07:13 +00:00
var old _ctr = {
x : w _area [ 0 ] . scrollLeft + w _orig / 2 ,
y : w _area [ 0 ] . scrollTop + h _orig / 2
} ;
var multi = curConfig . canvas _expansion ;
2010-04-09 14:18:46 +00:00
w = Math . max ( w _orig , svgCanvas . contentW * zoom * multi ) ;
h = Math . max ( h _orig , svgCanvas . contentH * zoom * multi ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( w == w _orig && h == h _orig ) {
2013-02-15 22:32:35 +00:00
workarea . css ( 'overflow' , 'hidden' ) ;
2010-03-11 21:07:13 +00:00
} else {
2013-02-15 22:32:35 +00:00
workarea . css ( 'overflow' , 'scroll' ) ;
2010-03-11 21:07:13 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var old _can _y = cnvs . height ( ) / 2 ;
var old _can _x = cnvs . width ( ) / 2 ;
cnvs . width ( w ) . height ( h ) ;
var new _can _y = h / 2 ;
var new _can _x = w / 2 ;
var offset = svgCanvas . updateCanvas ( w , h ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var ratio = new _can _x / old _can _x ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var scroll _x = w / 2 - w _orig / 2 ;
var scroll _y = h / 2 - h _orig / 2 ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( ! new _ctr ) {
2010-03-11 21:07:13 +00:00
var old _dist _x = old _ctr . x - old _can _x ;
var new _x = new _can _x + old _dist _x * ratio ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var old _dist _y = old _ctr . y - old _can _y ;
var new _y = new _can _y + old _dist _y * ratio ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
new _ctr = {
2010-04-09 14:18:46 +00:00
x : new _x ,
y : new _y
2010-03-11 21:07:13 +00:00
} ;
} else {
new _ctr . x += offset . x ,
new _ctr . y += offset . y ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( center ) {
2010-07-12 20:19:43 +00:00
// Go to top-left for larger documents
2013-02-15 21:57:35 +00:00
if ( svgCanvas . contentW > w _area . width ( ) ) {
2010-07-12 20:19:43 +00:00
// Top-left
workarea [ 0 ] . scrollLeft = offset . x - 10 ;
workarea [ 0 ] . scrollTop = offset . y - 10 ;
} else {
// Center
w _area [ 0 ] . scrollLeft = scroll _x ;
w _area [ 0 ] . scrollTop = scroll _y ;
}
2010-03-11 21:07:13 +00:00
} else {
w _area [ 0 ] . scrollLeft = new _ctr . x - w _orig / 2 ;
w _area [ 0 ] . scrollTop = new _ctr . y - h _orig / 2 ;
}
2013-02-15 21:57:35 +00:00
if ( curConfig . showRulers ) {
2010-10-01 18:59:12 +00:00
updateRulers ( cnvs , zoom ) ;
workarea . scroll ( ) ;
}
2013-10-12 03:08:26 +00:00
$ ( '#dialog_box' ) . hide ( ) ;
2013-02-15 13:38:55 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
// Make [1,2,5] array
var r _intervals = [ ] ;
2013-02-15 21:57:35 +00:00
for ( var i = . 1 ; i < 1E5 ; i *= 10 ) {
2010-10-01 18:59:12 +00:00
r _intervals . push ( 1 * i ) ;
r _intervals . push ( 2 * i ) ;
r _intervals . push ( 5 * i ) ;
}
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
function updateRulers ( scanvas , zoom ) {
2013-02-15 21:57:35 +00:00
if ( ! zoom ) zoom = svgCanvas . getZoom ( ) ;
if ( ! scanvas ) scanvas = $ ( '#svgcanvas' ) ;
2012-10-22 10:47:50 +00:00
2010-10-21 20:30:42 +00:00
var limit = 30000 ;
2013-02-20 21:04:54 +00:00
var contentElem = svgCanvas . getContentElem ( ) ;
2010-11-06 00:45:21 +00:00
var units = svgedit . units . getTypeMap ( ) ;
2010-10-05 17:13:00 +00:00
var unit = units [ curConfig . baseUnit ] ; // 1 = 1px
2012-10-22 10:47:50 +00:00
2013-02-20 21:04:54 +00:00
// draw x ruler then y ruler
2013-02-15 21:57:35 +00:00
for ( var d = 0 ; d < 2 ; d ++ ) {
2013-02-20 21:04:54 +00:00
var isX = ( d === 0 ) ;
var dim = isX ? 'x' : 'y' ;
var lentype = isX ? 'width' : 'height' ;
var contentDim = contentElem . getAttribute ( dim ) - 0 ;
2012-10-22 10:47:50 +00:00
2010-12-16 17:06:25 +00:00
var $hcanv _orig = $ ( '#ruler_' + dim + ' canvas:first' ) ;
2012-10-22 10:47:50 +00:00
2010-12-16 17:06:25 +00:00
// Bit of a hack to fully clear the canvas in Safari & IE9
2013-02-15 23:17:56 +00:00
var $hcanv = $hcanv _orig . clone ( ) ;
2010-12-16 17:06:25 +00:00
$hcanv _orig . replaceWith ( $hcanv ) ;
2012-10-22 10:47:50 +00:00
2010-10-21 20:30:42 +00:00
var hcanv = $hcanv [ 0 ] ;
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
// Set the canvas size to the width of the container
2010-10-21 20:30:42 +00:00
var ruler _len = scanvas [ lentype ] ( ) ;
var total _len = ruler _len ;
hcanv . parentNode . style [ lentype ] = total _len + 'px' ;
var ctx _num = 0 ;
var ctx _arr ;
2013-02-15 21:57:35 +00:00
var ctx = hcanv . getContext ( '2d' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
ctx . fillStyle = 'rgb(200,0,0)' ;
ctx . fillRect ( 0 , 0 , hcanv . width , hcanv . height ) ;
2012-10-22 10:47:50 +00:00
2010-10-21 20:30:42 +00:00
// Remove any existing canvasses
$hcanv . siblings ( ) . remove ( ) ;
2012-10-22 10:47:50 +00:00
2010-10-21 20:30:42 +00:00
// Create multiple canvases when necessary (due to browser limits)
2013-02-15 21:57:35 +00:00
if ( ruler _len >= limit ) {
2013-02-15 13:38:55 +00:00
var num = parseInt ( ruler _len / limit , 10 ) + 1 ;
2010-10-21 20:30:42 +00:00
ctx _arr = Array ( num ) ;
ctx _arr [ 0 ] = ctx ;
2013-02-15 21:57:35 +00:00
for ( var i = 1 ; i < num ; i ++ ) {
2010-10-21 20:30:42 +00:00
hcanv [ lentype ] = limit ;
var copy = hcanv . cloneNode ( true ) ;
hcanv . parentNode . appendChild ( copy ) ;
ctx _arr [ i ] = copy . getContext ( '2d' ) ;
}
2012-10-22 10:47:50 +00:00
2010-10-21 20:30:42 +00:00
copy [ lentype ] = ruler _len % limit ;
2012-10-22 10:47:50 +00:00
2010-10-21 20:30:42 +00:00
// set copy width to last
ruler _len = limit ;
}
2012-10-22 10:47:50 +00:00
2010-10-21 20:30:42 +00:00
hcanv [ lentype ] = ruler _len ;
2012-10-22 10:47:50 +00:00
2010-10-05 17:13:00 +00:00
var u _multi = unit * zoom ;
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
// Calculate the main number interval
2010-10-05 17:13:00 +00:00
var raw _m = 50 / u _multi ;
2010-10-01 18:59:12 +00:00
var multi = 1 ;
2013-02-15 21:57:35 +00:00
for ( var i = 0 ; i < r _intervals . length ; i ++ ) {
2010-10-01 18:59:12 +00:00
var num = r _intervals [ i ] ;
multi = num ;
2013-02-15 21:57:35 +00:00
if ( raw _m <= num ) {
2010-10-01 18:59:12 +00:00
break ;
}
}
2012-10-22 10:47:50 +00:00
2010-10-05 17:13:00 +00:00
var big _int = multi * u _multi ;
2013-02-15 21:57:35 +00:00
ctx . font = '9px sans-serif' ;
2010-10-21 20:30:42 +00:00
2013-02-20 21:04:54 +00:00
var ruler _d = ( ( contentDim / u _multi ) % multi ) * u _multi ;
2010-10-21 20:30:42 +00:00
var label _pos = ruler _d - big _int ;
2013-02-20 21:04:54 +00:00
// draw big intervals
2010-10-21 20:30:42 +00:00
for ( ; ruler _d < total _len ; ruler _d += big _int ) {
label _pos += big _int ;
2013-02-20 21:04:54 +00:00
var real _d = ruler _d - contentDim ;
2010-10-21 20:30:42 +00:00
2013-02-20 21:04:54 +00:00
var cur _d = Math . round ( ruler _d ) + 0.5 ;
if ( isX ) {
2010-10-01 18:59:12 +00:00
ctx . moveTo ( cur _d , 15 ) ;
ctx . lineTo ( cur _d , 0 ) ;
} else {
ctx . moveTo ( 15 , cur _d ) ;
ctx . lineTo ( 0 , cur _d ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-20 21:04:54 +00:00
var num = ( label _pos - contentDim ) / u _multi ;
2010-10-29 13:13:34 +00:00
var label ;
2013-02-15 21:57:35 +00:00
if ( multi >= 1 ) {
2010-10-05 17:13:00 +00:00
label = Math . round ( num ) ;
} else {
var decs = ( multi + '' ) . split ( '.' ) [ 1 ] . length ;
label = num . toFixed ( decs ) - 0 ;
}
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
// Change 1000s to Ks
2013-02-15 21:57:35 +00:00
if ( label !== 0 && label !== 1000 && label % 1000 === 0 ) {
2010-10-01 18:59:12 +00:00
label = ( label / 1000 ) + 'K' ;
}
2012-10-22 10:47:50 +00:00
2013-02-20 21:04:54 +00:00
if ( isX ) {
2010-10-01 18:59:12 +00:00
ctx . fillText ( label , ruler _d + 2 , 8 ) ;
} else {
2013-02-20 21:04:54 +00:00
// draw label vertically
2010-10-01 18:59:12 +00:00
var str = ( label + '' ) . split ( '' ) ;
2013-02-15 21:57:35 +00:00
for ( var i = 0 ; i < str . length ; i ++ ) {
2010-10-01 18:59:12 +00:00
ctx . fillText ( str [ i ] , 1 , ( ruler _d + 9 ) + i * 9 ) ;
}
}
2012-10-22 10:47:50 +00:00
2010-10-01 18:59:12 +00:00
var part = big _int / 10 ;
2013-02-20 21:04:54 +00:00
// draw the small intervals
2013-02-15 21:57:35 +00:00
for ( var i = 1 ; i < 10 ; i ++ ) {
2013-02-20 21:04:54 +00:00
var sub _d = Math . round ( ruler _d + part * i ) + 0.5 ;
2013-02-15 21:57:35 +00:00
if ( ctx _arr && sub _d > ruler _len ) {
2010-10-21 20:30:42 +00:00
ctx _num ++ ;
ctx . stroke ( ) ;
2013-02-15 21:57:35 +00:00
if ( ctx _num >= ctx _arr . length ) {
2010-10-21 20:30:42 +00:00
i = 10 ;
ruler _d = total _len ;
continue ;
}
ctx = ctx _arr [ ctx _num ] ;
ruler _d -= limit ;
2013-02-20 21:04:54 +00:00
sub _d = Math . round ( ruler _d + part * i ) + 0.5 ;
2010-10-21 20:30:42 +00:00
}
2012-10-22 10:47:50 +00:00
2013-02-20 21:04:54 +00:00
// odd lines are slighly longer
var line _num = ( i % 2 ) ? 12 : 10 ;
if ( isX ) {
2010-10-01 18:59:12 +00:00
ctx . moveTo ( sub _d , 15 ) ;
ctx . lineTo ( sub _d , line _num ) ;
} else {
ctx . moveTo ( 15 , sub _d ) ;
2013-02-20 21:04:54 +00:00
ctx . lineTo ( line _num , sub _d ) ;
2010-10-01 18:59:12 +00:00
}
}
}
2013-02-15 21:57:35 +00:00
ctx . strokeStyle = '#000' ;
2010-10-01 18:59:12 +00:00
ctx . stroke ( ) ;
}
2009-12-11 20:35:51 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// $(function() {
updateCanvas ( true ) ;
// });
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// var revnums = "svg-editor.js ($Rev$) ";
// revnums += svgCanvas.getVersion();
2013-02-15 21:57:35 +00:00
// $('#copyright')[0].setAttribute('title', revnums);
2012-10-22 10:47:50 +00:00
2010-09-20 18:55:10 +00:00
// Callback handler for embedapi.js
2013-02-15 13:38:55 +00:00
try {
2013-02-15 21:57:35 +00:00
window . addEventListener ( 'message' , function ( e ) {
2013-10-13 23:59:32 +00:00
if ( ! e . data || typeof e . data !== 'object' || e . data . namespace !== 'svgCanvas' ) {
return ;
}
var cbid = e . data . id ,
name = e . data . name ,
args = e . data . args ;
2013-02-15 13:38:55 +00:00
try {
2013-10-13 23:59:32 +00:00
e . source . postMessage ( { namespace : 'svg-edit' , id : cbid , result : svgCanvas [ name ] ( args ) } , '*' ) ;
2013-02-15 13:38:55 +00:00
} catch ( err ) {
2013-10-13 23:59:32 +00:00
e . source . postMessage ( { namespace : 'svg-edit' , id : cbid , error : err . message } , '*' ) ;
2013-02-15 13:38:55 +00:00
}
} , false ) ;
} catch ( err ) {
window . embed _error = err ;
2010-09-20 18:55:10 +00:00
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// For Compatibility with older extensions
$ ( function ( ) {
window . svgCanvas = svgCanvas ;
svgCanvas . ready = svgEditor . ready ;
} ) ;
2012-10-22 10:47:50 +00:00
2010-12-31 20:02:35 +00:00
Editor . setLang = function ( lang , allStrings ) {
2010-03-11 21:07:13 +00:00
$ . pref ( 'lang' , lang ) ;
$ ( '#lang_select' ) . val ( lang ) ;
2013-02-15 21:57:35 +00:00
if ( ! allStrings ) {
return ;
}
var notif = allStrings . notification ;
// $.extend will only replace the given strings
var oldLayerName = $ ( '#layerlist tr.layersel td.layername' ) . text ( ) ;
var rename _layer = ( oldLayerName == uiStrings . common . layer + ' 1' ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ . extend ( uiStrings , allStrings ) ;
svgCanvas . setUiStrings ( allStrings ) ;
Actions . setTitles ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
if ( rename _layer ) {
svgCanvas . renameCurrentLayer ( uiStrings . common . layer + ' 1' ) ;
populateLayers ( ) ;
}
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
svgCanvas . runExtensions ( 'langChanged' , lang ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
// Update flyout tooltips
setFlyoutTitles ( ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
// Copy title for certain tool elements
var elems = {
'#stroke_color' : '#tool_stroke .icon_label, #tool_stroke .color_block' ,
'#fill_color' : '#tool_fill label, #tool_fill .color_block' ,
'#linejoin_miter' : '#cur_linejoin' ,
'#linecap_butt' : '#cur_linecap'
} ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
$ . each ( elems , function ( source , dest ) {
$ ( dest ) . attr ( 'title' , $ ( source ) [ 0 ] . title ) ;
} ) ;
2012-10-22 10:47:50 +00:00
2013-02-15 21:57:35 +00:00
// Copy alignment titles
$ ( '#multiselected_panel div[id^=tool_align]' ) . each ( function ( ) {
$ ( '#tool_pos' + this . id . substr ( 10 ) ) [ 0 ] . title = this . title ;
} ) ;
2010-01-15 21:33:55 +00:00
} ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
var callbacks = [ ] ;
2012-10-22 10:47:50 +00:00
2010-10-08 18:21:46 +00:00
function loadSvgString ( str , callback ) {
var success = svgCanvas . setSvgString ( str ) !== false ;
callback = callback || $ . noop ;
2013-02-15 21:57:35 +00:00
if ( success ) {
2010-10-08 18:21:46 +00:00
callback ( true ) ;
} else {
2011-01-03 18:45:36 +00:00
$ . alert ( uiStrings . notification . errorLoadingSVG , function ( ) {
2010-10-08 18:21:46 +00:00
callback ( false ) ;
} ) ;
2010-10-08 17:54:45 +00:00
}
}
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
Editor . ready = function ( cb ) {
2013-02-15 21:57:35 +00:00
if ( ! isReady ) {
2010-03-11 21:07:13 +00:00
callbacks . push ( cb ) ;
} else {
cb ( ) ;
}
} ;
2010-01-15 21:33:55 +00:00
2010-03-11 21:07:13 +00:00
Editor . runCallbacks = function ( ) {
$ . each ( callbacks , function ( ) {
this ( ) ;
} ) ;
2013-02-15 21:57:35 +00:00
isReady = true ;
2010-03-11 21:07:13 +00:00
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
Editor . loadFromString = function ( str ) {
Editor . ready ( function ( ) {
2010-10-08 17:54:45 +00:00
loadSvgString ( str ) ;
2010-03-11 21:07:13 +00:00
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2011-02-03 14:58:47 +00:00
Editor . disableUI = function ( featList ) {
// $(function() {
// $('#tool_wireframe, #tool_image, #main_button, #tool_source, #sidepanels').remove();
// $('#tools_top').css('left', 5);
// });
} ;
2012-10-22 10:47:50 +00:00
2010-10-08 18:21:46 +00:00
Editor . loadFromURL = function ( url , opts ) {
2013-02-15 21:57:35 +00:00
if ( ! opts ) opts = { } ;
2010-10-11 12:21:37 +00:00
2010-10-08 18:21:46 +00:00
var cache = opts . cache ;
var cb = opts . callback ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
Editor . ready ( function ( ) {
$ . ajax ( {
'url' : url ,
'dataType' : 'text' ,
2010-07-19 15:38:23 +00:00
cache : ! ! cache ,
2013-05-08 17:29:18 +00:00
beforeSend : function ( ) {
2013-10-12 03:08:26 +00:00
$ . process _cancel ( uiStrings . notification . loadingImage ) ;
} ,
2010-10-08 18:21:46 +00:00
success : function ( str ) {
loadSvgString ( str , cb ) ;
} ,
2010-04-23 19:40:09 +00:00
error : function ( xhr , stat , err ) {
2013-02-15 21:57:35 +00:00
if ( xhr . status != 404 && xhr . responseText ) {
2010-10-08 18:21:46 +00:00
loadSvgString ( xhr . responseText , cb ) ;
2010-04-23 19:40:09 +00:00
} else {
2013-02-15 21:57:35 +00:00
$ . alert ( uiStrings . notification . URLloadFail + ': \n' + err + '' , cb ) ;
2010-03-11 21:07:13 +00:00
}
2013-05-08 17:29:18 +00:00
} ,
2013-10-12 03:08:26 +00:00
complete : function ( ) {
$ ( '#dialog_box' ) . hide ( ) ;
}
2010-03-11 21:07:13 +00:00
} ) ;
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
Editor . loadFromDataURI = function ( str ) {
Editor . ready ( function ( ) {
var pre = 'data:image/svg+xml;base64,' ;
var src = str . substring ( pre . length ) ;
2010-11-30 20:51:07 +00:00
loadSvgString ( svgedit . utilities . decode64 ( src ) ) ;
2010-03-11 21:07:13 +00:00
} ) ;
} ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
Editor . addExtension = function ( ) {
var args = arguments ;
2012-10-22 10:47:50 +00:00
2010-09-20 12:49:36 +00:00
// Note that we don't want this on Editor.ready since some extensions
// may want to run before then (like server_opensave).
$ ( function ( ) {
2013-02-15 21:57:35 +00:00
if ( svgCanvas ) svgCanvas . addExtension . apply ( this , args ) ;
2010-03-11 21:07:13 +00:00
} ) ;
} ;
2009-11-09 18:59:53 +00:00
2010-03-16 19:28:57 +00:00
return Editor ;
} ( jQuery ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
// Run init once DOM is loaded
$ ( svgEditor . init ) ;
2012-10-22 10:47:50 +00:00
2010-03-11 21:07:13 +00:00
} ) ( ) ;