mirror of
https://github.com/xevidos/codiad.git
synced 2025-01-08 22:11:55 +01:00
Updated download scripts for next release
This commit is contained in:
parent
a830b33606
commit
f80c926f85
6 changed files with 112 additions and 1607 deletions
|
@ -58,6 +58,7 @@ Task List:
|
||||||
* Add move files
|
* Add move files
|
||||||
* Add permissions module ( more in depth permissions such as read/write, delete, etc )
|
* Add permissions module ( more in depth permissions such as read/write, delete, etc )
|
||||||
* Add print code
|
* Add print code
|
||||||
|
* Add support for more archive types ( Add commands add more accepted PHP extension types )
|
||||||
* Add support for more database systems ( MSSQL, Oracle, SQLite, Filesystem storage, etc )
|
* Add support for more database systems ( MSSQL, Oracle, SQLite, Filesystem storage, etc )
|
||||||
* Add terminal support ( optional per permission level )
|
* Add terminal support ( optional per permission level )
|
||||||
* Add in auto save timer that saves after the user stops typing instead of after every change
|
* Add in auto save timer that saves after the user stops typing instead of after every change
|
||||||
|
|
|
@ -27,6 +27,11 @@ class Archive {
|
||||||
".DS_Store"
|
".DS_Store"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const MIME_TYPE_EXTENSIONS = array(
|
||||||
|
|
||||||
|
"application/zip" => "zip",
|
||||||
|
);
|
||||||
|
|
||||||
const SUPPORTED_TYPES = array(
|
const SUPPORTED_TYPES = array(
|
||||||
|
|
||||||
//"gz",
|
//"gz",
|
||||||
|
@ -113,7 +118,7 @@ class Archive {
|
||||||
|
|
||||||
if( extension_loaded( self::EXTENSIONS["{$type}"] ) ) {
|
if( extension_loaded( self::EXTENSIONS["{$type}"] ) ) {
|
||||||
|
|
||||||
$response = call_user_func( array( $archive, "{$type}_c" ), $path, $output );
|
$response = call_user_func( array( $archive, "{$type}_c" ), $path, $output );
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
//$response = $archive->execute( $type, "compress", $path, dirname( $path ) );
|
//$response = $archive->execute( $type, "compress", $path, dirname( $path ) );
|
||||||
|
@ -227,9 +232,17 @@ class Archive {
|
||||||
$path = rtrim( $path, '/' );
|
$path = rtrim( $path, '/' );
|
||||||
//$output = rtrim( $output, '/' ) . '/';
|
//$output = rtrim( $output, '/' ) . '/';
|
||||||
$archive = new ZipArchive();
|
$archive = new ZipArchive();
|
||||||
if( $archive->open( $output, ZIPARCHIVE::CREATE ) !== true ) {
|
|
||||||
|
if( file_exists( $output ) ) {
|
||||||
|
|
||||||
|
$result = $archive->open( $output, ZIPARCHIVE::OVERWRITE );
|
||||||
|
} else {
|
||||||
|
|
||||||
|
$result = $archive->open( $output, ZIPARCHIVE::CREATE );
|
||||||
|
}
|
||||||
|
|
||||||
|
if( $result !== true ) {
|
||||||
|
|
||||||
echo var_dump( $path, $output );
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,84 +1,109 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Copyright (c) Codiad & Kent Safranski (codiad.com), distributed
|
* Copyright (c) Codiad & Kent Safranski (codiad.com), distributed
|
||||||
* as-is and without warranty under the MIT License. See
|
* as-is and without warranty under the MIT License. See
|
||||||
* [root]/license.txt for more. This information must remain intact.
|
* [root]/license.txt for more. This information must remain intact.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
require_once('../../common.php');
|
require_once( '../../common.php' );
|
||||||
|
require_once( './class.archive.php' );
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////
|
||||||
// Verify Session or Key
|
// Verify Session or Key
|
||||||
//////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
checkSession();
|
checkSession();
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////
|
$response = array(
|
||||||
// Check $_GET for invalid path
|
"status" => "none",
|
||||||
//////////////////////////////////////////////////////////////////
|
"message" => null,
|
||||||
//TODO check if the User is allowed to access the project
|
);
|
||||||
if (!isset($_GET['path'])
|
|
||||||
|| preg_match('#^[\\\/]?$#i', trim($_GET['path'])) // download all Projects
|
if( ! isset( $_GET["path"] ) && ! isset( $_POST["path"] ) ) {
|
||||||
|| preg_match('#[\:*?\"<>\|]#i', $_GET['path']) //illegal chars in filenames
|
|
||||||
|| substr_count($_GET['path'], './') > 0) { // change directory up to escape Workspace
|
$response["status"] = "error";
|
||||||
exit('<script>parent.codiad.message.error("Wrong data send")</script>');
|
$response["message"] = "Missing path.";
|
||||||
|
exit( json_encode( $response ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////
|
$path = ( isset( $_GET["path"] ) ) ? $_GET["path"] : $_POST["path"];
|
||||||
// Run Download
|
$full_path = "";
|
||||||
//////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
if (isset($_GET['type']) && ($_GET['type']=='directory' || $_GET['type']=='root')) {
|
if( Common::isAbsPath( $path ) ) {
|
||||||
// Create tarball
|
|
||||||
$filename = explode("/", $_GET['path']);
|
$full_path = realpath( $path );
|
||||||
//$filename = array_pop($filename) . "-" . date('Y.m.d') . ".tar.gz";
|
|
||||||
$filename = array_pop($filename) . "-" . date('Y.m.d');
|
|
||||||
$targetPath = DATA . '/';
|
|
||||||
$dir = WORKSPACE . '/' . $_GET['path'];
|
|
||||||
if (!is_dir($dir)) {
|
|
||||||
exit('<script>parent.codiad.message.error("Directory not found.")</script>');
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////
|
|
||||||
// Check system() command and a non windows OS
|
|
||||||
//////////////////////////////////////////////////////////////////
|
|
||||||
if (extension_loaded('zip')) { //Check if zip-Extension is availiable
|
|
||||||
//build zipfile
|
|
||||||
require_once 'class.dirzip.php';
|
|
||||||
|
|
||||||
$filename .= '.zip';
|
|
||||||
$download_file = $targetPath.$filename;
|
|
||||||
Archive::compress( $dir, $targetPath . $filename );
|
|
||||||
} elseif (isAvailable('system') && stripos(PHP_OS, 'win') === false) {
|
|
||||||
# Execute the tar command and save file
|
|
||||||
$filename .= '.tar.gz';
|
|
||||||
|
|
||||||
system("tar -pczf ".escapeshellarg($targetPath.$filename)." -C ".escapeshellarg(WORKSPACE)." ".escapeshellarg($_GET['path']));
|
|
||||||
$download_file = $targetPath.$filename;
|
|
||||||
} else {
|
|
||||||
exit('<script>parent.codiad.message.error("Could not pack the folder, zip-extension missing")</script>');
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
$filename = explode("/", $_GET['path']);
|
|
||||||
$filename = array_pop($filename);
|
$full_path = WORKSPACE . "/$path";
|
||||||
$download_file = WORKSPACE . '/' . $_GET['path'];
|
$full_path = realpath( $full_path );
|
||||||
}
|
}
|
||||||
|
|
||||||
header('Content-Description: File Transfer');
|
if( $full_path === false ) {
|
||||||
header('Content-Type: application/octet-stream');
|
|
||||||
header('Content-Disposition: attachment; filename="'.basename($filename).'"');
|
$response["status"] = "error";
|
||||||
header('Content-Transfer-Encoding: binary');
|
$response["message"] = "Invalid path.";
|
||||||
header('Expires: 0');
|
exit( json_encode( $response ) );
|
||||||
header('Cache-Control: must-revalidate');
|
|
||||||
header('Pragma: public');
|
|
||||||
header('Content-Length: ' . filesize($download_file));
|
|
||||||
if (ob_get_contents()) {
|
|
||||||
ob_end_clean();
|
|
||||||
}
|
}
|
||||||
flush();
|
|
||||||
readfile($download_file);
|
if( ! Permissions::has_read( $path ) ) {
|
||||||
// Remove temp tarball
|
|
||||||
if ($_GET['type']=='directory' || $_GET['type']=='root') {
|
$response["status"] = "error";
|
||||||
unlink($download_file);
|
$response["message"] = "You do not have access to this path.";
|
||||||
|
exit( json_encode( $response ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
if( is_dir( $full_path ) ) {
|
||||||
|
|
||||||
|
$temp_path = tempnam( sys_get_temp_dir(), 'codiad_download_' . date( "U" ) );
|
||||||
|
$result = Archive::compress( $full_path, $temp_path, 'default' );
|
||||||
|
$mime_type = mime_content_type( $temp_path );
|
||||||
|
|
||||||
|
if( in_array( $mime_type, array_keys( Archive::MIME_TYPE_EXTENSIONS ) ) ) {
|
||||||
|
|
||||||
|
$extension = Archive::MIME_TYPE_EXTENSIONS["$mime_type"];
|
||||||
|
}
|
||||||
|
|
||||||
|
if( $result ) {
|
||||||
|
|
||||||
|
header( 'Content-Description: File Transfer' );
|
||||||
|
header( 'Content-Type: application/octet-stream' );
|
||||||
|
header( 'Content-Disposition: attachment; filename="' . basename( $full_path ) . ".$extension" . '"' );
|
||||||
|
header( 'Content-Transfer-Encoding: binary' );
|
||||||
|
header( 'Expires: 0' );
|
||||||
|
header( 'Cache-Control: must-revalidate' );
|
||||||
|
header( 'Pragma: public' );
|
||||||
|
header( 'Content-Length: ' . filesize( $temp_path ) );
|
||||||
|
|
||||||
|
if( ob_get_contents() ) {
|
||||||
|
|
||||||
|
ob_end_clean();
|
||||||
|
}
|
||||||
|
|
||||||
|
flush();
|
||||||
|
file_put_contents( "php://output", file_get_contents( $temp_path ) );
|
||||||
|
} else {
|
||||||
|
|
||||||
|
$response["status"] = "error";
|
||||||
|
$response["message"] = "An archive could not be created.";
|
||||||
|
}
|
||||||
|
unlink( $temp_path );
|
||||||
|
} else {
|
||||||
|
|
||||||
|
header( 'Content-Description: File Transfer' );
|
||||||
|
header( 'Content-Type: application/octet-stream' );
|
||||||
|
header( 'Content-Disposition: attachment; filename="' . basename( $full_path ) . '"' );
|
||||||
|
header( 'Content-Transfer-Encoding: binary' );
|
||||||
|
header( 'Expires: 0' );
|
||||||
|
header( 'Cache-Control: must-revalidate' );
|
||||||
|
header( 'Pragma: public' );
|
||||||
|
header( 'Content-Length: ' . filesize( $full_path ) );
|
||||||
|
|
||||||
|
if( ob_get_contents() ) {
|
||||||
|
|
||||||
|
ob_end_clean();
|
||||||
|
}
|
||||||
|
|
||||||
|
flush();
|
||||||
|
readfile( $full_path );
|
||||||
}
|
}
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,172 +0,0 @@
|
||||||
/*
|
|
||||||
* jQuery Iframe Transport Plugin 1.5
|
|
||||||
* https://github.com/blueimp/jQuery-File-Upload
|
|
||||||
*
|
|
||||||
* Copyright 2011, Sebastian Tschan
|
|
||||||
* https://blueimp.net
|
|
||||||
*
|
|
||||||
* Licensed under the MIT license:
|
|
||||||
* http://www.opensource.org/licenses/MIT
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*jslint unparam: true, nomen: true */
|
|
||||||
/*global define, window, document */
|
|
||||||
|
|
||||||
(function (factory) {
|
|
||||||
'use strict';
|
|
||||||
if (typeof define === 'function' && define.amd) {
|
|
||||||
// Register as an anonymous AMD module:
|
|
||||||
define(['jquery'], factory);
|
|
||||||
} else {
|
|
||||||
// Browser globals:
|
|
||||||
factory(window.jQuery);
|
|
||||||
}
|
|
||||||
}(function ($) {
|
|
||||||
'use strict';
|
|
||||||
|
|
||||||
// Helper variable to create unique names for the transport iframes:
|
|
||||||
var counter = 0;
|
|
||||||
|
|
||||||
// The iframe transport accepts three additional options:
|
|
||||||
// options.fileInput: a jQuery collection of file input fields
|
|
||||||
// options.paramName: the parameter name for the file form data,
|
|
||||||
// overrides the name property of the file input field(s),
|
|
||||||
// can be a string or an array of strings.
|
|
||||||
// options.formData: an array of objects with name and value properties,
|
|
||||||
// equivalent to the return data of .serializeArray(), e.g.:
|
|
||||||
// [{name: 'a', value: 1}, {name: 'b', value: 2}]
|
|
||||||
$.ajaxTransport('iframe', function (options) {
|
|
||||||
if (options.async && (options.type === 'POST' || options.type === 'GET')) {
|
|
||||||
var form,
|
|
||||||
iframe;
|
|
||||||
return {
|
|
||||||
send: function (_, completeCallback) {
|
|
||||||
form = $('<form style="display:none;"></form>');
|
|
||||||
form.attr('accept-charset', options.formAcceptCharset);
|
|
||||||
// javascript:false as initial iframe src
|
|
||||||
// prevents warning popups on HTTPS in IE6.
|
|
||||||
// IE versions below IE8 cannot set the name property of
|
|
||||||
// elements that have already been added to the DOM,
|
|
||||||
// so we set the name along with the iframe HTML markup:
|
|
||||||
iframe = $(
|
|
||||||
'<iframe src="javascript:false;" name="iframe-transport-' +
|
|
||||||
(counter += 1) + '"></iframe>'
|
|
||||||
).bind('load', function () {
|
|
||||||
var fileInputClones,
|
|
||||||
paramNames = $.isArray(options.paramName) ?
|
|
||||||
options.paramName : [options.paramName];
|
|
||||||
iframe
|
|
||||||
.unbind('load')
|
|
||||||
.bind('load', function () {
|
|
||||||
var response;
|
|
||||||
// Wrap in a try/catch block to catch exceptions thrown
|
|
||||||
// when trying to access cross-domain iframe contents:
|
|
||||||
try {
|
|
||||||
response = iframe.contents();
|
|
||||||
// Google Chrome and Firefox do not throw an
|
|
||||||
// exception when calling iframe.contents() on
|
|
||||||
// cross-domain requests, so we unify the response:
|
|
||||||
if (!response.length || !response[0].firstChild) {
|
|
||||||
throw new Error();
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
response = undefined;
|
|
||||||
}
|
|
||||||
// The complete callback returns the
|
|
||||||
// iframe content document as response object:
|
|
||||||
completeCallback(
|
|
||||||
200,
|
|
||||||
'success',
|
|
||||||
{'iframe': response}
|
|
||||||
);
|
|
||||||
// Fix for IE endless progress bar activity bug
|
|
||||||
// (happens on form submits to iframe targets):
|
|
||||||
$('<iframe src="javascript:false;"></iframe>')
|
|
||||||
.appendTo(form);
|
|
||||||
form.remove();
|
|
||||||
});
|
|
||||||
form
|
|
||||||
.prop('target', iframe.prop('name'))
|
|
||||||
.prop('action', options.url)
|
|
||||||
.prop('method', options.type);
|
|
||||||
if (options.formData) {
|
|
||||||
$.each(options.formData, function (index, field) {
|
|
||||||
$('<input type="hidden"/>')
|
|
||||||
.prop('name', field.name)
|
|
||||||
.val(field.value)
|
|
||||||
.appendTo(form);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (options.fileInput && options.fileInput.length &&
|
|
||||||
options.type === 'POST') {
|
|
||||||
fileInputClones = options.fileInput.clone();
|
|
||||||
// Insert a clone for each file input field:
|
|
||||||
options.fileInput.after(function (index) {
|
|
||||||
return fileInputClones[index];
|
|
||||||
});
|
|
||||||
if (options.paramName) {
|
|
||||||
options.fileInput.each(function (index) {
|
|
||||||
$(this).prop(
|
|
||||||
'name',
|
|
||||||
paramNames[index] || options.paramName
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// Appending the file input fields to the hidden form
|
|
||||||
// removes them from their original location:
|
|
||||||
form
|
|
||||||
.append(options.fileInput)
|
|
||||||
.prop('enctype', 'multipart/form-data')
|
|
||||||
// enctype must be set as encoding for IE:
|
|
||||||
.prop('encoding', 'multipart/form-data');
|
|
||||||
}
|
|
||||||
form.submit();
|
|
||||||
// Insert the file input fields at their original location
|
|
||||||
// by replacing the clones with the originals:
|
|
||||||
if (fileInputClones && fileInputClones.length) {
|
|
||||||
options.fileInput.each(function (index, input) {
|
|
||||||
var clone = $(fileInputClones[index]);
|
|
||||||
$(input).prop('name', clone.prop('name'));
|
|
||||||
clone.replaceWith(input);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
form.append(iframe).appendTo(document.body);
|
|
||||||
},
|
|
||||||
abort: function () {
|
|
||||||
if (iframe) {
|
|
||||||
// javascript:false as iframe src aborts the request
|
|
||||||
// and prevents warning popups on HTTPS in IE6.
|
|
||||||
// concat is used to avoid the "Script URL" JSLint error:
|
|
||||||
iframe
|
|
||||||
.unbind('load')
|
|
||||||
.prop('src', 'javascript'.concat(':false;'));
|
|
||||||
}
|
|
||||||
if (form) {
|
|
||||||
form.remove();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// The iframe transport returns the iframe content document as response.
|
|
||||||
// The following adds converters from iframe to text, json, html, and script:
|
|
||||||
$.ajaxSetup({
|
|
||||||
converters: {
|
|
||||||
'iframe text': function (iframe) {
|
|
||||||
return $(iframe[0].body).text();
|
|
||||||
},
|
|
||||||
'iframe json': function (iframe) {
|
|
||||||
return $.parseJSON($(iframe[0].body).text());
|
|
||||||
},
|
|
||||||
'iframe html': function (iframe) {
|
|
||||||
return $(iframe[0].body).html();
|
|
||||||
},
|
|
||||||
'iframe script': function (iframe) {
|
|
||||||
return $.globalEval($(iframe[0].body).text());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
}));
|
|
|
@ -1,282 +0,0 @@
|
||||||
/*
|
|
||||||
* jQuery UI Widget 1.8.23+amd
|
|
||||||
* https://github.com/blueimp/jQuery-File-Upload
|
|
||||||
*
|
|
||||||
* Copyright 2012, AUTHORS.txt (http://jqueryui.com/about)
|
|
||||||
* Dual licensed under the MIT or GPL Version 2 licenses.
|
|
||||||
* http://jquery.org/license
|
|
||||||
*
|
|
||||||
* http://docs.jquery.com/UI/Widget
|
|
||||||
*/
|
|
||||||
|
|
||||||
(function (factory) {
|
|
||||||
if (typeof define === "function" && define.amd) {
|
|
||||||
// Register as an anonymous AMD module:
|
|
||||||
define(["jquery"], factory);
|
|
||||||
} else {
|
|
||||||
// Browser globals:
|
|
||||||
factory(jQuery);
|
|
||||||
}
|
|
||||||
}(function( $, undefined ) {
|
|
||||||
|
|
||||||
// jQuery 1.4+
|
|
||||||
if ( $.cleanData ) {
|
|
||||||
var _cleanData = $.cleanData;
|
|
||||||
$.cleanData = function( elems ) {
|
|
||||||
for ( var i = 0, elem; (elem = elems[i]) != null; i++ ) {
|
|
||||||
try {
|
|
||||||
$( elem ).triggerHandler( "remove" );
|
|
||||||
// http://bugs.jquery.com/ticket/8235
|
|
||||||
} catch( e ) {}
|
|
||||||
}
|
|
||||||
_cleanData( elems );
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
var _remove = $.fn.remove;
|
|
||||||
$.fn.remove = function( selector, keepData ) {
|
|
||||||
return this.each(function() {
|
|
||||||
if ( !keepData ) {
|
|
||||||
if ( !selector || $.filter( selector, [ this ] ).length ) {
|
|
||||||
$( "*", this ).add( [ this ] ).each(function() {
|
|
||||||
try {
|
|
||||||
$( this ).triggerHandler( "remove" );
|
|
||||||
// http://bugs.jquery.com/ticket/8235
|
|
||||||
} catch( e ) {}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return _remove.call( $(this), selector, keepData );
|
|
||||||
});
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
$.widget = function( name, base, prototype ) {
|
|
||||||
var namespace = name.split( "." )[ 0 ],
|
|
||||||
fullName;
|
|
||||||
name = name.split( "." )[ 1 ];
|
|
||||||
fullName = namespace + "-" + name;
|
|
||||||
|
|
||||||
if ( !prototype ) {
|
|
||||||
prototype = base;
|
|
||||||
base = $.Widget;
|
|
||||||
}
|
|
||||||
|
|
||||||
// create selector for plugin
|
|
||||||
$.expr[ ":" ][ fullName ] = function( elem ) {
|
|
||||||
return !!$.data( elem, name );
|
|
||||||
};
|
|
||||||
|
|
||||||
$[ namespace ] = $[ namespace ] || {};
|
|
||||||
$[ namespace ][ name ] = function( options, element ) {
|
|
||||||
// allow instantiation without initializing for simple inheritance
|
|
||||||
if ( arguments.length ) {
|
|
||||||
this._createWidget( options, element );
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var basePrototype = new base();
|
|
||||||
// we need to make the options hash a property directly on the new instance
|
|
||||||
// otherwise we'll modify the options hash on the prototype that we're
|
|
||||||
// inheriting from
|
|
||||||
// $.each( basePrototype, function( key, val ) {
|
|
||||||
// if ( $.isPlainObject(val) ) {
|
|
||||||
// basePrototype[ key ] = $.extend( {}, val );
|
|
||||||
// }
|
|
||||||
// });
|
|
||||||
basePrototype.options = $.extend( true, {}, basePrototype.options );
|
|
||||||
$[ namespace ][ name ].prototype = $.extend( true, basePrototype, {
|
|
||||||
namespace: namespace,
|
|
||||||
widgetName: name,
|
|
||||||
widgetEventPrefix: $[ namespace ][ name ].prototype.widgetEventPrefix || name,
|
|
||||||
widgetBaseClass: fullName
|
|
||||||
}, prototype );
|
|
||||||
|
|
||||||
$.widget.bridge( name, $[ namespace ][ name ] );
|
|
||||||
};
|
|
||||||
|
|
||||||
$.widget.bridge = function( name, object ) {
|
|
||||||
$.fn[ name ] = function( options ) {
|
|
||||||
var isMethodCall = typeof options === "string",
|
|
||||||
args = Array.prototype.slice.call( arguments, 1 ),
|
|
||||||
returnValue = this;
|
|
||||||
|
|
||||||
// allow multiple hashes to be passed on init
|
|
||||||
options = !isMethodCall && args.length ?
|
|
||||||
$.extend.apply( null, [ true, options ].concat(args) ) :
|
|
||||||
options;
|
|
||||||
|
|
||||||
// prevent calls to internal methods
|
|
||||||
if ( isMethodCall && options.charAt( 0 ) === "_" ) {
|
|
||||||
return returnValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( isMethodCall ) {
|
|
||||||
this.each(function() {
|
|
||||||
var instance = $.data( this, name ),
|
|
||||||
methodValue = instance && $.isFunction( instance[options] ) ?
|
|
||||||
instance[ options ].apply( instance, args ) :
|
|
||||||
instance;
|
|
||||||
// TODO: add this back in 1.9 and use $.error() (see #5972)
|
|
||||||
// if ( !instance ) {
|
|
||||||
// throw "cannot call methods on " + name + " prior to initialization; " +
|
|
||||||
// "attempted to call method '" + options + "'";
|
|
||||||
// }
|
|
||||||
// if ( !$.isFunction( instance[options] ) ) {
|
|
||||||
// throw "no such method '" + options + "' for " + name + " widget instance";
|
|
||||||
// }
|
|
||||||
// var methodValue = instance[ options ].apply( instance, args );
|
|
||||||
if ( methodValue !== instance && methodValue !== undefined ) {
|
|
||||||
returnValue = methodValue;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
this.each(function() {
|
|
||||||
var instance = $.data( this, name );
|
|
||||||
if ( instance ) {
|
|
||||||
instance.option( options || {} )._init();
|
|
||||||
} else {
|
|
||||||
$.data( this, name, new object( options, this ) );
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return returnValue;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
$.Widget = function( options, element ) {
|
|
||||||
// allow instantiation without initializing for simple inheritance
|
|
||||||
if ( arguments.length ) {
|
|
||||||
this._createWidget( options, element );
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
$.Widget.prototype = {
|
|
||||||
widgetName: "widget",
|
|
||||||
widgetEventPrefix: "",
|
|
||||||
options: {
|
|
||||||
disabled: false
|
|
||||||
},
|
|
||||||
_createWidget: function( options, element ) {
|
|
||||||
// $.widget.bridge stores the plugin instance, but we do it anyway
|
|
||||||
// so that it's stored even before the _create function runs
|
|
||||||
$.data( element, this.widgetName, this );
|
|
||||||
this.element = $( element );
|
|
||||||
this.options = $.extend( true, {},
|
|
||||||
this.options,
|
|
||||||
this._getCreateOptions(),
|
|
||||||
options );
|
|
||||||
|
|
||||||
var self = this;
|
|
||||||
this.element.bind( "remove." + this.widgetName, function() {
|
|
||||||
self.destroy();
|
|
||||||
});
|
|
||||||
|
|
||||||
this._create();
|
|
||||||
this._trigger( "create" );
|
|
||||||
this._init();
|
|
||||||
},
|
|
||||||
_getCreateOptions: function() {
|
|
||||||
return $.metadata && $.metadata.get( this.element[0] )[ this.widgetName ];
|
|
||||||
},
|
|
||||||
_create: function() {},
|
|
||||||
_init: function() {},
|
|
||||||
|
|
||||||
destroy: function() {
|
|
||||||
this.element
|
|
||||||
.unbind( "." + this.widgetName )
|
|
||||||
.removeData( this.widgetName );
|
|
||||||
this.widget()
|
|
||||||
.unbind( "." + this.widgetName )
|
|
||||||
.removeAttr( "aria-disabled" )
|
|
||||||
.removeClass(
|
|
||||||
this.widgetBaseClass + "-disabled " +
|
|
||||||
"ui-state-disabled" );
|
|
||||||
},
|
|
||||||
|
|
||||||
widget: function() {
|
|
||||||
return this.element;
|
|
||||||
},
|
|
||||||
|
|
||||||
option: function( key, value ) {
|
|
||||||
var options = key;
|
|
||||||
|
|
||||||
if ( arguments.length === 0 ) {
|
|
||||||
// don't return a reference to the internal hash
|
|
||||||
return $.extend( {}, this.options );
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof key === "string" ) {
|
|
||||||
if ( value === undefined ) {
|
|
||||||
return this.options[ key ];
|
|
||||||
}
|
|
||||||
options = {};
|
|
||||||
options[ key ] = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
this._setOptions( options );
|
|
||||||
|
|
||||||
return this;
|
|
||||||
},
|
|
||||||
_setOptions: function( options ) {
|
|
||||||
var self = this;
|
|
||||||
$.each( options, function( key, value ) {
|
|
||||||
self._setOption( key, value );
|
|
||||||
});
|
|
||||||
|
|
||||||
return this;
|
|
||||||
},
|
|
||||||
_setOption: function( key, value ) {
|
|
||||||
this.options[ key ] = value;
|
|
||||||
|
|
||||||
if ( key === "disabled" ) {
|
|
||||||
this.widget()
|
|
||||||
[ value ? "addClass" : "removeClass"](
|
|
||||||
this.widgetBaseClass + "-disabled" + " " +
|
|
||||||
"ui-state-disabled" )
|
|
||||||
.attr( "aria-disabled", value );
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
},
|
|
||||||
|
|
||||||
enable: function() {
|
|
||||||
return this._setOption( "disabled", false );
|
|
||||||
},
|
|
||||||
disable: function() {
|
|
||||||
return this._setOption( "disabled", true );
|
|
||||||
},
|
|
||||||
|
|
||||||
_trigger: function( type, event, data ) {
|
|
||||||
var prop, orig,
|
|
||||||
callback = this.options[ type ];
|
|
||||||
|
|
||||||
data = data || {};
|
|
||||||
event = $.Event( event );
|
|
||||||
event.type = ( type === this.widgetEventPrefix ?
|
|
||||||
type :
|
|
||||||
this.widgetEventPrefix + type ).toLowerCase();
|
|
||||||
// the original event may come from any element
|
|
||||||
// so we need to reset the target on the new event
|
|
||||||
event.target = this.element[ 0 ];
|
|
||||||
|
|
||||||
// copy original event properties over to the new event
|
|
||||||
orig = event.originalEvent;
|
|
||||||
if ( orig ) {
|
|
||||||
for ( prop in orig ) {
|
|
||||||
if ( !( prop in event ) ) {
|
|
||||||
event[ prop ] = orig[ prop ];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.element.trigger( event, data );
|
|
||||||
|
|
||||||
return !( $.isFunction(callback) &&
|
|
||||||
callback.call( this.element[0], event, data ) === false ||
|
|
||||||
event.isDefaultPrevented() );
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
}));
|
|
Loading…
Reference in a new issue