jQuery $.fn 不是一个函数
2016-09-18
4448
直接投反对票?
我是否遗漏了导致
$.fn.
无法正常工作的任何内容。而我有其他模块,此功能运行良好。
我在 jQuery 中遇到了一些奇怪的错误。我正在使用 jquery.jrac.js 文件裁剪图像,不幸的是,当我以这种方式访问​​函数时,我得到了
TypeError: $.fn.jracIMG 不是一个函数
$.fn.jracIMG("", param);
我已经正确添加了 jquery.js 文件。
如果我从
jracIMG
中删除
$.fn.
,在库文件本身中,它会给出一些其他奇怪的错误,如
$(elem).find("img").jrac({
下面是我正在使用的函数
(function( $ ){
$.fn.jrac = function(options) {
// Default settings
var settings = {
'crop_width': 200,
'crop_height': 100,
// The two following properties define the crop position (relative to the
// image).
'crop_x': 0,
'crop_y': 0,
'crop_resize': true,
'crop_aspect_ratio': null,
'image_width': null,
'image_height': null,
'zoom_min': 100,
'zoom_max': 3000,
'zoom_label': '',
'viewport_image_surrounding': false, // Set the viewport to surround the image on load
'viewport_width': null,
'viewport_height': null,
'viewport_resize': true,
// The two following properties allow to position the content (negative
// value allowed). It can be use to focus the viewport on the cropped
// part of the image.
'viewport_content_left': 0,
'viewport_content_top': 0,
// Submit here a callback function (context is the viewport).
'viewport_onload': null
};
// Apply the resize and crop tools to each images
return this.each(function() {
if (!$(this).is('img')) {
return;
}
// Read options
var destroy = false;
if ( typeof(options) == 'object' ) {
$.extend( settings, options );
}
else if (options == 'destroy') {
destroy = true;
}
var $image = $(this);
// Test if jrac was previously run
var jrac_loaded = false;
if ($image.parent().hasClass('jrac_viewport')) {
jrac_loaded = true;
}
if (jrac_loaded && destroy) {
// Unload jrac if asked
$image.draggable("destroy");
$image.parent().find('.jrac_crop').remove();
$image.parent().parent().find('.jrac_zoom_slider').remove();
$image.parent().append($image.data('original')); // restore original image
$image.parent().unwrap(); // remove container
$image.unwrap(); // remove viewport
$image.remove(); // remove image which was altered (in its css attributes)
}
else {
// Record image before some of its attributes get altered by some
// jrac css or some user size or proportion changes.
$image.data('original', $image.clone());
}
// Do nothing more if destroy is asked
if (destroy) {
return;
}
// Prepare image
if (!jrac_loaded) {
$image.hide().css('position','absolute').wrap('<div class="jrac_container"><div class="jrac_viewport"></div></div>');
}
// Get viewport and container references
var $viewport = $image.parent();
var $container = $viewport.parent();
// Add a waiting on load input image
var $loading = $('<div class="jrac_loading" />');
$viewport.append($loading);
// The following procedure hold business intend to be run once the image
// is loaded (load event).
var image_load_handler = function(){
// Add some custom properties to $image
$.extend($image, {
scale_proportion_locked: true,
originalWidth: $image.width(),
originalHeight: $image.height()
});
// Set given optional image size
$image.width(settings.image_width).height(settings.image_height);
// Apply the viewport image surrounding is asked
if (settings.viewport_image_surrounding) {
$viewport.width($image.width()).height($image.height());
}
else {
$viewport.width(settings.viewport_width).height(settings.viewport_height);
}
// Set the viewport content position for the image
$image.css({'left': settings.viewport_content_left, 'top': settings.viewport_content_top});
// Create the zoom widget which permit to resize the image
if (!jrac_loaded) {
if (settings.zoom_label) {
var $zoom_label = $('<div class="jrac_zoom_slider_label">'+settings.zoom_label+'</div>');
$container.append($zoom_label);
}
var $zoom_widget = $('<div class="jrac_zoom_slider"><div class="ui-slider-handle"></div></div>')
.slider({
value: $image.width(),
min: settings.zoom_min,
max: settings.zoom_max,
start: function(event, ui) {
$.extend($zoom_widget,{
on_start_width_value: ui.value,
on_start_height_value: $image.height()
})
},
slide: function(event, ui) {
var height = Math.round($zoom_widget.on_start_height_value * ui.value / $zoom_widget.on_start_width_value);
$image.height(height);
$image.width(ui.value);
$viewport.observator.notify('jrac_image_height', height);
$viewport.observator.notify('jrac_image_width', ui.value);
}
});
$container.append($zoom_widget);
// Make the viewport resizeable
if (settings.viewport_resize) {
$viewport.resizable();
}
// Enable the image draggable interaction
$image.draggable({
drag: function(event, ui) {
if (ui.position.left != ui.originalPosition.left) {
$viewport.observator.notify('jrac_crop_x', $viewport.observator.crop_position_x());
}
if (ui.position.top != ui.originalPosition.top) {
$viewport.observator.notify('jrac_crop_y', $viewport.observator.crop_position_y());
}
}
});
// Build the crop element
var $crop = $('<div class="jrac_crop"><div class="jrac_crop_drag_handler"></div></div>')
.css({
'width': settings.crop_width,
'height': settings.crop_height,
'left':settings.crop_x+settings.viewport_content_left,
'top':settings.crop_y+settings.viewport_content_top
}).draggable({
containment: $viewport,
handle: 'div.jrac_crop_drag_handler',
drag: function(event, ui) {
if (ui.position.left != ui.originalPosition.left) {
$viewport.observator.notify('jrac_crop_x', $viewport.observator.crop_position_x());
}
if (ui.position.top != ui.originalPosition.top) {
$viewport.observator.notify('jrac_crop_y', $viewport.observator.crop_position_y());
}
}
});
if (settings.crop_resize) {
$crop.resizable({
containment: $viewport,
aspectRatio: settings.crop_aspect_ratio,
resize: function(event, ui) {
if (ui.size.width != ui.originalSize.width) {
$viewport.observator.notify('jrac_crop_width', $crop.width());
}
if (ui.size.height != ui.originalSize.height) {
$viewport.observator.notify('jrac_crop_height', $crop.height());
}
}
})
}
$viewport.append($crop);
}
// Extend viewport witch usefull objects as it will be exposed to user
// functions interface
$.extend($viewport, {
$container: $container,
$image: $image,
$crop: $crop,
$zoom_widget: $zoom_widget,
// Let me introduce the following Terminator's friend which handle the
// creation of the viewport events.
observator: {
items: {},
// Register an event with a given element
register: function(event_name, element, onevent_callback) {
if (event_name && element) {
this.items[event_name] = {
element: element,
callback: onevent_callback
};
}
},
// Unregister an event
unregister: function(event_name) {
delete this.items[event_name];
},
// Trigger an event and optionally supply a value
notify: function(event_name, value) {
if (this.items[event_name]) {
var element = this.items[event_name].element;
var onevent_callback = this.items[event_name].callback;
element.trigger(event_name,[$viewport, value]);
if ($.isFunction(onevent_callback)) {
onevent_callback.call($viewport, event_name, element, value);
}
}
$image.trigger('jrac_events',[$viewport]);
},
notify_all: function() {
this.notify('jrac_crop_x', this.crop_position_x());
this.notify('jrac_crop_y', this.crop_position_y());
this.notify('jrac_crop_width', $crop.width());
this.notify('jrac_crop_height', $crop.height());
this.notify('jrac_image_width', $image.width());
this.notify('jrac_image_height', $image.height());
},
// Return crop x position relative to $image
crop_position_x: function() {
return $crop.position().left - $image.position().left;
},
// Return crop y position relative to $image
crop_position_y: function() {
return $crop.position().top - $image.position().top;
},
// Does the crop is completely inside the image?
crop_consistent: function() {
return this.crop_position_x()>=0 && this.crop_position_y()>=0
&& this.crop_position_x() + $crop.width() + parseInt($crop.css('border-width'),10)*2 <=$image.width()
&& this.crop_position_y() + $crop.height() + parseInt($crop.css('border-width'),10)*2 <=$image.height();
},
// Set a property (which his name is one of the event) with a given
// value then notify this operation
set_property: function(that, value) {
value = parseInt(value);
if (isNaN(value)) {
return;
}
switch (that) {
case 'jrac_crop_x':
$crop.css('left',value + $image.position().left);
break;
case 'jrac_crop_y':
$crop.css('top',value + $image.position().top);
break;
case 'jrac_crop_width':
$crop.width(value);
break;
case 'jrac_crop_height':
$crop.height(value);
break;
case 'jrac_image_width':
if ($image.scale_proportion_locked) {
var image_height = Math.round($image.height() * value / $image.width());
$image.height(image_height);
this.notify('jrac_image_height', image_height);
}
$image.width(value);
$zoom_widget.slider('value', value);
break;
case 'jrac_image_height':
if ($image.scale_proportion_locked) {
var image_width = Math.round($image.width() * value / $image.height());
$image.width(image_width);
this.notify('jrac_image_width', image_width);
$zoom_widget.slider('value', image_width);
}
$image.height(value);
break;
}
this.notify(that, value);
}
}
});
// Hide the loading notice
$loading.hide();
$loading.remove();
// Finally display the image
$image.show();
// Trigger the viewport_onload callback
if ($.isFunction(settings.viewport_onload)) {
settings.viewport_onload.call($viewport);
$viewport.observator.notify_all();
}
};
// When an image is using an src image "data" URL scheme then it appear
// that the image load event never get fired. Then fire directly
// image_load_handler() in that case.
var src = $image.attr('src');
if (/^data:image/.test(src)) {
image_load_handler();
}
else {
src = src + (src.search(/\?/)<0?'?':'&') + 'jracrandom=' + (new Date()).getTime();
$('<img>').attr('src', src).load(image_load_handler);
}
});
};
})( jQuery );
还有这个
$.fn.jracIMG = function(elem,param) {
$(elem).find("img").jrac({
'crop_resize':param["crop_resize"],
'viewport_resize': param["viewport_resize"],
'crop_width': param["crop_width"],
'crop_height': param["crop_height"],
'crop_x': param["crop_x"],
'crop_y': param["crop_y"],
'image_width' : param["image_width"] ? param["image_width"] : 550,
'viewport_onload': function() {
var $viewport = this;
//$viewport.$image.draggable("destroy");
var inputs = $viewport.$container.parent('.pane').find('.coords input:hidden');
var events = ['jrac_crop_x', 'jrac_crop_y', 'jrac_crop_width', 'jrac_crop_height', 'jrac_image_width', 'jrac_image_height'];
for (var i = 0; i < events.length; i++) {
var event_name = events[i];
// Register an event with an element.
$viewport.observator.register(event_name, inputs.eq(i));
// Attach a handler to that event for the element.
inputs.eq(i).bind(event_name, function(event, $viewport, value) {
$(this).val(value);
})
// Attach a handler for the built-in jQuery change event, handler
// which read user input and apply it to relevent viewport object.
.change(event_name, function(event) {
var event_name = event.data;
$viewport.$image.scale_proportion_locked = $viewport.$container.parent('.pane').find('.coords input:checkbox').is(':checked');
$viewport.observator.set_property(event_name, $(this).val());
});
}
$viewport.$container.find(".imgnatsize").remove();
$viewport.$container.append('<div class=imgnatsize>Image natual size: '
+ $viewport.$image.originalWidth + ' x '
+ $viewport.$image.originalHeight + '</div>')
}
})
// React on all viewport events.
.bind('jrac_events', function(event, $viewport) {
var inputs = $(this).parents('.pane').find('.coords input');
//inputs.css('background-color', ($viewport.observator.crop_consistent()) ? 'chartreuse' : 'salmon');
if ($viewport.observator.crop_consistent()) {
inputs.removeClass('_err').addClass('_suc');
} else {
inputs.removeClass('_suc').addClass('_err');
}
});
}
$.fn.containerFix = function(container) {
container = $(container);
var mTop = container.outerHeight() / 2;
container.css({
'margin-top': -mTop,
});
};
$.fn.createCropPanel = function() {
var elem;
elem = "<div class='pane clearfix'>";
elem += "<img class='cropPreview' />";
elem += "<form class='cropPanelForm' method='post'>";
elem += "<div class='coords'>";
elem += "<input type='hidden' name='cropx'/>";
elem += "<input type='hidden' name='cropy'/>";
elem += "<input type='hidden' name='cropw'/>";
elem += "<input type='hidden' name='croph'/>";
elem += "<input type='hidden' name='imagew'/>";
elem += "<input type='hidden' name='imageh'/>";
elem += "<input type='hidden' name='imagefilename'/>";
elem += "<input type='hidden' name='imagefiledirectory'/>";
elem += "<input type='hidden' name='is_dashboard' value='0'/>";
elem += "<input type='submit' value='Crop & Save' class='btn-cta btn-cta-sm b-styled' />";
elem += "<input type='hidden' name='logofilename'/>";
elem += "<input type='hidden' name='logofiledirectory'/>";
elem += "<input type='hidden' name='sliderfilename'/>";
elem += "<input type='hidden' name='sliderfiledirectory'/>";
elem += "</div>";
elem += "</form>";
elem += "</div>";
return elem;
};
$.fn.createCropPanelOnboard = function() {
var elem;
elem = "<div class='pane clearfix'>";
elem += "<img class='cropPreview' />";
elem += "<form class='cropPanelForm' method='post'>";
elem += "<div class='coords'>";
elem += "<input type='hidden' name='cropx'/>";
elem += "<input type='hidden' name='cropy'/>";
elem += "<input type='hidden' name='cropw'/>";
elem += "<input type='hidden' name='croph'/>";
elem += "<input type='hidden' name='imagew'/>";
elem += "<input type='hidden' name='imageh'/>";
elem += "<input type='hidden' name='imagefilename'/>";
elem += "<input type='hidden' name='imagefiledirectory'/>";
elem += "<input type='hidden' name='is_dashboard' value='0'/>";
//elem += "<input type='submit' id='crop_btn' value='Crop & Save' class='btn-cta btn-cta-sm b-styled' />";
elem += "<a href='javascript:;' class='btn-cta btn-cta-sm b-styled' id='crop_btn' >Crop & Save</a>";
elem += "<input type='hidden' name='logofilename'/>";
elem += "<input type='hidden' name='logofiledirectory'/>";
elem += "<input type='hidden' name='sliderfilename'/>";
elem += "<input type='hidden' name='sliderfiledirectory'/>";
elem += "</div>";
elem += "</form>";
elem += "</div>";
return elem;
};
如能提供任何帮助,将不胜感激。
解决方案
我把那些函数定义放在
$(document).ready(function(){
//Both files code goes here
});
现在它运行良好。
2个回答
我将这些函数定义、函数放在文件中,
$(document).ready(function(){
//Both files code goes here
});
现在它运行得很好。
Smile
2016-09-18
我以前也遇到过同样的问题,你试过使用 CDN 而不是自己托管代码吗?
我建议使用 Google 的 CDN,因为他们的服务器速度非常快,不会减慢客户端的速度。您可以使用
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.1.0/jquery.min.js"></script>
代替您的文件。
希望这对您有所帮助。
Jarryd Lisher
2016-09-18