วิธีการปรับขนาดภาพตามสัดส่วน / รักษาอัตราส่วนภาพ?


167

ฉันมีภาพที่จะมีขนาดค่อนข้างใหญ่และฉันต้องการลดขนาดลงด้วย jQuery ในขณะที่รักษาสัดส่วนที่ จำกัด ไว้เช่นอัตราส่วนภาพเดียวกัน

ใครสามารถชี้ให้ฉันดูรหัสหรืออธิบายเหตุผลได้บ้าง


4
คุณช่วยอธิบายได้ไหมว่าทำไมต้องใช้ jQuery มีวิธี CSS-เท่านั้น (ดูที่เป็นคำตอบของฉัน ): การตั้งค่าของมันmax-widthและเพื่อmax-height 100%
Dan Dascalescu

9
ในกรณีที่ไม่มีใครรู้ถ้าคุณตั้งค่าเพียงมิติเดียวของรูปภาพ (ทั้งความกว้างหรือความสูง) มันจะปรับขนาดตามสัดส่วน เป็นเช่นนี้มาตั้งแต่รุ่งอรุณของเว็บ ตัวอย่างเช่น:<img src='image.jpg' width=200>
GetFree

2
นอกจากนี้คุณอาจลองใช้บางอย่างเช่นslimmage.jsเพื่อประหยัดแบนด์วิดท์และ RAM อุปกรณ์มือถือ
ลิลิ ธ ริเวอร์

คำตอบ:


188

ดูรหัสชิ้นนี้ได้จากhttp://ericjuden.com/2009/07/jquery-image-resize/

$(document).ready(function() {
    $('.story-small img').each(function() {
        var maxWidth = 100; // Max width for the image
        var maxHeight = 100;    // Max height for the image
        var ratio = 0;  // Used for aspect ratio
        var width = $(this).width();    // Current image width
        var height = $(this).height();  // Current image height

        // Check if the current width is larger than the max
        if(width > maxWidth){
            ratio = maxWidth / width;   // get ratio for scaling image
            $(this).css("width", maxWidth); // Set new width
            $(this).css("height", height * ratio);  // Scale height based on ratio
            height = height * ratio;    // Reset height to match scaled image
            width = width * ratio;    // Reset width to match scaled image
        }

        // Check if current height is larger than max
        if(height > maxHeight){
            ratio = maxHeight / height; // get ratio for scaling image
            $(this).css("height", maxHeight);   // Set new height
            $(this).css("width", width * ratio);    // Scale width based on ratio
            width = width * ratio;    // Reset width to match scaled image
            height = height * ratio;    // Reset height to match scaled image
        }
    });
});

1
ขออภัยไม่มีตรรกะทางคณิตศาสตร์บางอย่าง ... เกิดอะไรขึ้นเมื่อคุณต้องการเพิ่มมันทั้งหมด (สมมติว่าคุณกำลังเพิ่ม maxHeight)?
เบ็น

4
สิ่งนี้สามารถทำได้ด้วย CSS เพียงอย่างเดียว? (ความกว้างสูงสุดความสูง: อัตโนมัติ ฯลฯ )
Tronathan

11
ไม่แน่ใจว่าทำไม jQuery ถึงต้องการสำหรับสิ่งนี้ การหดตัวของภาพตามสัดส่วนที่ลูกค้าสามารถทำได้ด้วย CSS และมันน่ารำคาญ: เพียงแค่ตั้งค่าของตนmax-widthและเพื่อmax-height jsfiddle.net/9EQ5c100%
Dan Dascalescu

10
สิ่งนี้ไม่สามารถทำได้ด้วย CSS เนื่องจาก IF STATEMENT ฉันเชื่อว่าจุดสำคัญคือการเติมภาพขนาดย่อ หากภาพสูงเกินไปจะต้องมีความกว้างสูงสุดหากภาพกว้างเกินไปจะต้องมีความสูงสูงสุด หากคุณทำ CSS ความกว้างสูงสุดความสูงสูงสุดคุณจะได้ภาพขนาดย่อที่มีช่องว่างแทนการเติมเต็ม
ntgCleaner

รหัสนี้อาจทำให้เกิดปัญหาในเบราว์เซอร์ล่มหรือช้าลงหรือไม่?
เดจาบอนด์

444

ฉันคิดว่านี่เป็นวิธีที่ยอดเยี่ยมจริงๆ:

 /**
  * Conserve aspect ratio of the original region. Useful when shrinking/enlarging
  * images to fit into a certain area.
  *
  * @param {Number} srcWidth width of source image
  * @param {Number} srcHeight height of source image
  * @param {Number} maxWidth maximum available width
  * @param {Number} maxHeight maximum available height
  * @return {Object} { width, height }
  */
function calculateAspectRatioFit(srcWidth, srcHeight, maxWidth, maxHeight) {

    var ratio = Math.min(maxWidth / srcWidth, maxHeight / srcHeight);

    return { width: srcWidth*ratio, height: srcHeight*ratio };
 }

33
คำตอบที่ยอดเยี่ยมมาก! คำตอบที่ถูกต้องอยู่บนใบหน้าถ้าความสูงและความกว้างนั้นกว้างกว่า หนวดดีจริงๆและหนวดก็ดี
Starkers

1
คุณสิทธิกำลัง @sstauross ว่าพิกเซลทศนิยมสามารถมีเล็กน้อยผลที่ไม่คาดคิด ในกรณีที่ใช้งานของฉัน แต่มันก็เล็กน้อย ฉันคิดว่าMath.floorจะช่วยในเรื่อง การออกแบบที่สมบูรณ์แบบของพิกเซล :-)
Jason J. Nathan

1
ขอบคุณฉันต้องการสิ่งนี้เกือบ "หนึ่งซับ"
Hernán

1
ขอบคุณ Jason คำตอบนี้ช่วยฉันได้จริงๆ
Ashok Shah

4
นี่เป็นวิธีที่ยอดเยี่ยมในการจัดการปัญหานี้! ฉันปรับแต่งเล็กน้อยสำหรับองค์ประกอบ img + ป้องกันการขยายภาพ:function imgSizeFit(img, maxWidth, maxHeight){ var ratio = Math.min(1, maxWidth / img.naturalWidth, maxHeight / img.naturalHeight); img.style.width = img.naturalWidth * ratio + 'px'; img.style.height = img.naturalHeight * ratio + 'px'; }
oriadam

70

หากฉันเข้าใจคำถามอย่างถูกต้องคุณไม่จำเป็นต้องมี jQuery การหดตัวของภาพตามสัดส่วนที่ลูกค้าสามารถทำได้ด้วย CSS เพียงอย่างเดียว: เพียงแค่ตั้งค่าของตนmax-widthและเพื่อmax-height100%

<div style="height: 100px">
<img src="http://www.getdigital.de/images/produkte/t4/t4_css_sucks2.jpg"
    style="max-height: 100%; max-width: 100%">
</div>​

นี่คือซอ: http://jsfiddle.net/9EQ5c/


2
นี่เป็นคำตอบที่ง่ายกว่าข้างบน ขอบคุณ btw คุณได้รับลิงก์ "คำตอบของฉัน" เพื่อเลื่อนลงไปที่โพสต์ของคุณได้อย่างไร
SnareChops

@SnareChops: มันเป็นเพียงการยึด HTML
Dan Dascalescu

1
@SnareChops: ถ้าคุณใช้ลิงก์ที่กำหนดโดยลิงก์ "share" ใต้คำตอบมันจะเลื่อนไปที่คำตอบ
Flimm

1
@Flimm เนื่องจากช่วงไม่แสดงผล: บล็อกโดยค่าเริ่มต้น เพียงเพิ่ม display: block หรือตั้งเป็น div
mahemoff

1
ในกรณีของฉัน IMG ถูกสร้างการแสดงผลด้วย WordPress ดังนั้นจึงตั้งค่าความกว้างและความสูงของหมวก ใน CSS ฉันยังมีการตั้งค่าwidth: auto; height: auto;ที่จะได้รับรหัสของคุณทำงาน :)
lippoliv

12

ในการกำหนดอัตราส่วนกว้างยาวคุณจำเป็นต้องมีอัตราส่วนเพื่อมุ่งสู่

ความสูง

function getHeight(length, ratio) {
  var height = ((length)/(Math.sqrt((Math.pow(ratio, 2)+1))));
  return Math.round(height);
}

ความกว้าง

function getWidth(length, ratio) {
  var width = ((length)/(Math.sqrt((1)/(Math.pow(ratio, 2)+1))));
  return Math.round(width);
}

ในตัวอย่างนี้ฉันใช้16:10ตั้งแต่นี้อัตราส่วนจอภาพทั่วไป

var ratio = (16/10);
var height = getHeight(300,ratio);
var width = getWidth(height,ratio);

console.log(height);
console.log(width);

ผลลัพธ์จากด้านบนจะเป็น147และ300


เมื่อพิจารณา, 300 = ความกว้างในแนวทแยง = ความสูง * อัตราส่วนและความสูงเท่ากับที่คุณบอกไว้
Johny Pie

6

ที่จริงฉันเพิ่งเจอปัญหานี้และวิธีแก้ปัญหาที่ฉันพบนั้นเรียบง่ายและแปลก

$("#someimage").css({height:<some new height>})

และภาพจะถูกปรับขนาดเป็นความสูงใหม่อย่างน่าอัศจรรย์และรักษาอัตราส่วนเดิมไว้อย่างน่าอัศจรรย์!


1
ฉันคิดว่ามันมีประโยชน์ - แต่ฉันคิดว่ามันจะไม่ จำกัด ภาพถ้าพูดได้กว้างมากถึงความกว้างสูงสุด ...
stephendwolff

สิ่งนี้ใช้ได้เมื่อคุณไม่ได้ตั้งค่าคุณลักษณะอื่น (ความกว้างในกรณีนี้)
NoobishPro

4

ปัญหานี้มี 4 พารามิเตอร์

  1. ความกว้างของภาพปัจจุบัน iX
  2. ความสูงของภาพปัจจุบัน iY
  3. ความกว้างวิวพอร์ตเป้าหมาย cX
  4. เป้าหมายความสูงวิวพอร์ต cY

และมีพารามิเตอร์เงื่อนไข 3 แบบ

  1. cX> cY?
  2. iX> cX?
  3. ฉัน> cY?

สารละลาย

  1. ค้นหาด้านที่เล็กกว่าของพอร์ตมุมมองเป้าหมาย F
  2. ค้นหาด้านที่ใหญ่กว่าของพอร์ตมุมมองปัจจุบัน L
  3. ค้นหาปัจจัยทั้ง F / L = factor
  4. ทวีคูณทั้งสองด้านของพอร์ตปัจจุบันด้วยปัจจัยคือ fX = iX * factor fY = iY * factor

นั่นคือทั้งหมดที่คุณต้องทำ

//Pseudo code


iX;//current width of image in the client
iY;//current height of image in the client
cX;//configured width
cY;//configured height
fX;//final width
fY;//final height

1. check if iX,iY,cX,cY values are >0 and all values are not empty or not junk

2. lE = iX > iY ? iX: iY; //long edge

3. if ( cX < cY )
   then
4.      factor = cX/lE;     
   else
5.      factor = cY/lE;

6. fX = iX * factor ; fY = iY * factor ; 

นี่เป็นฟอรัมสำหรับผู้ใหญ่ฉันไม่ได้ให้รหัสสำหรับคุณ :)


2
การโพสต์วิธีการนี้ดีมาก แต่ฉันทำเครื่องหมายว่าไม่ช่วยผู้ใช้ด้วยการโพสต์โค้ด ดูเหมือนจะเป็นสิ่งกีดขวางเล็กน้อย
Doidgey

6
"ใครบางคนชี้ให้ฉันไปที่รหัสบางส่วนหรืออธิบายตรรกะ?" - เห็นได้ชัดว่าเขาก็โอเคกับวิธีการอธิบายให้เขาเท่านั้น โดยส่วนตัวแล้วฉันคิดว่านี่จะเป็นวิธีที่ดีกว่าในการช่วยเหลือใครสักคนเพื่อช่วยให้พวกเขาเข้าใจวิธีการแทนที่จะคัดลอกและวางรหัส
JessMcintosh

@JessMcintosh สงสารแก้ไข bazillion ไปที่คำถามเดิมที่แสดงความคิดเห็นของคุณออกไปจากบริบท :)
เจสันเจนาธาน

4

ไม่<img src="/path/to/pic.jpg" style="max-width:XXXpx; max-height:YYYpx;" >ช่วยเหลือ?

เบราว์เซอร์จะดูแลการรักษาอัตราส่วนภาพให้เหมือนเดิม

คือmax-widthเตะในเมื่อความกว้างของภาพมากกว่าความสูงและความสูงของมันจะถูกคำนวณตามสัดส่วน ในทำนองเดียวกันmax-heightจะมีผลเมื่อความสูงมากกว่าความกว้าง

คุณไม่ต้องการ jQuery หรือ javascript ใด ๆ

รองรับ IE7 + และเบราว์เซอร์อื่น ๆ ( http://caniuse.com/minmaxwh )


สุดยอดเคล็ดลับ! เพียงแค่ใส่ CSS ลงในไฟล์ CSS และไม่ได้อยู่ในโค้ด html โดยตรง
Mark

ฉันคิดว่าปัญหาของเรื่องนี้คือมันจะไม่ทำงานเมื่อคุณไม่ทราบว่าความกว้างสูงสุดและความสูงสูงสุดคืออะไรจนกว่าหน้าจะโหลด นี่คือเหตุผลที่ต้องการโซลูชัน JS โดยปกติจะเป็นกรณีของไซต์ที่ตอบสนอง
เจสันเจนาธาน

2

สิ่งนี้จะทำงานกับภาพที่มีสัดส่วนเท่าที่เป็นไปได้ทั้งหมด

$(document).ready(function() {
    $('.list img').each(function() {
        var maxWidth = 100;
        var maxHeight = 100;
        var width = $(this).width();
        var height = $(this).height();
        var ratioW = maxWidth / width;  // Width ratio
        var ratioH = maxHeight / height;  // Height ratio

        // If height ratio is bigger then we need to scale height
        if(ratioH > ratioW){
            $(this).css("width", maxWidth);
            $(this).css("height", height * ratioW);  // Scale height according to width ratio
        }
        else{ // otherwise we scale width
            $(this).css("height", maxHeight);
            $(this).css("width", height * ratioH);  // according to height ratio
        }
    });
});

2

นี่คือการแก้ไขคำตอบของ Mehdiway ความกว้างและ / หรือความสูงใหม่ไม่ได้ถูกตั้งค่าเป็นค่าสูงสุด กรณีทดสอบที่ดีเป็นดังต่อไปนี้ (1768 x 1075 พิกเซล): http://spacecoastsports.com/wp-content/uploads/2014/06/sportsballs1.png (ฉันไม่สามารถแสดงความคิดเห็นได้เนื่องจากการขาดคะแนนชื่อเสียง)

  // Make sure image doesn't exceed 100x100 pixels
  // note: takes jQuery img object not HTML: so width is a function
  // not a property.
  function resize_image (image) {
      var maxWidth = 100;           // Max width for the image
      var maxHeight = 100;          // Max height for the image
      var ratio = 0;                // Used for aspect ratio

      // Get current dimensions
      var width = image.width()
      var height = image.height(); 
      console.log("dimensions: " + width + "x" + height);

      // If the current width is larger than the max, scale height
      // to ratio of max width to current and then set width to max.
      if (width > maxWidth) {
          console.log("Shrinking width (and scaling height)")
          ratio = maxWidth / width;
          height = height * ratio;
          width = maxWidth;
          image.css("width", width);
          image.css("height", height);
          console.log("new dimensions: " + width + "x" + height);
      }

      // If the current height is larger than the max, scale width
      // to ratio of max height to current and then set height to max.
      if (height > maxHeight) {
          console.log("Shrinking height (and scaling width)")
          ratio = maxHeight / height;
          width = width * ratio;
          height = maxHeight;
          image.css("width", width);
          image.css("height", height);
          console.log("new dimensions: " + width + "x" + height);
      }
  }

2
$('#productThumb img').each(function() {
    var maxWidth = 140; // Max width for the image
    var maxHeight = 140;    // Max height for the image
    var ratio = 0;  // Used for aspect ratio
    var width = $(this).width();    // Current image width
    var height = $(this).height();  // Current image height
    // Check if the current width is larger than the max
    if(width > height){
        height = ( height / width ) * maxHeight;

    } else if(height > width){
        maxWidth = (width/height)* maxWidth;
    }
    $(this).css("width", maxWidth); // Set new width
    $(this).css("height", maxHeight);  // Scale height based on ratio
});

5
โปรดพิจารณาการเพิ่มคำอธิบายไม่ใช่รหัสเฉพาะเมื่อตอบรับโพสต์
Jørgen R

1

หากภาพเป็นสัดส่วนแล้วรหัสนี้จะเติมเสื้อคลุมด้วยภาพ หากภาพไม่ได้อยู่ในสัดส่วนความกว้าง / ความสูงพิเศษจะถูกครอบตัด

    <script type="text/javascript">
        $(function(){
            $('#slider img').each(function(){
                var ReqWidth = 1000; // Max width for the image
                var ReqHeight = 300; // Max height for the image
                var width = $(this).width(); // Current image width
                var height = $(this).height(); // Current image height
                // Check if the current width is larger than the max
                if (width > height && height < ReqHeight) {

                    $(this).css("min-height", ReqHeight); // Set new height
                }
                else 
                    if (width > height && width < ReqWidth) {

                        $(this).css("min-width", ReqWidth); // Set new width
                    }
                    else 
                        if (width > height && width > ReqWidth) {

                            $(this).css("max-width", ReqWidth); // Set new width
                        }
                        else 
                            (height > width && width < ReqWidth)
                {

                    $(this).css("min-width", ReqWidth); // Set new width
                }
            });
        });
    </script>

1

โดยไม่ต้องเพิ่ม temp-vars หรือวงเล็บ

    var width= $(this).width(), height= $(this).height()
      , maxWidth=100, maxHeight= 100;

    if(width > maxWidth){
      height = Math.floor( maxWidth * height / width );
      width = maxWidth
      }
    if(height > maxHeight){
      width = Math.floor( maxHeight * width / height );
      height = maxHeight;
      }

โปรดทราบ: เครื่องมือค้นหาไม่ชอบถ้าแอตทริบิวต์ความกว้างและความสูงไม่พอดีกับภาพ แต่พวกเขาไม่รู้ JS


1

หลังจากการทดลองและข้อผิดพลาดฉันมาถึงวิธีนี้:

function center(img) {
    var div = img.parentNode;
    var divW = parseInt(div.style.width);
    var divH = parseInt(div.style.height);
    var srcW = img.width;
    var srcH = img.height;
    var ratio = Math.min(divW/srcW, divH/srcH);
    var newW = img.width * ratio;
    var newH = img.height * ratio;
    img.style.width  = newW + "px";
    img.style.height = newH + "px";
    img.style.marginTop = (divH-newH)/2 + "px";
    img.style.marginLeft = (divW-newW)/2 + "px";
}

1

การปรับขนาดสามารถทำได้ (รักษาอัตราส่วนภาพ) โดยใช้ CSS นี่เป็นคำตอบที่ง่ายขึ้นโดยได้แรงบันดาลใจจากโพสต์ของ Dan Dascalescu

http://jsbin.com/viqare

img{
     max-width:200px;
 /*Or define max-height*/
  }
<img src="http://e1.365dm.com/13/07/4-3/20/alastair-cook-ashes-profile_2967773.jpg"  alt="Alastair Cook" />

<img src="http://e1.365dm.com/13/07/4-3/20/usman-khawaja-australia-profile_2974601.jpg" alt="Usman Khawaja"/>


1

2 ขั้นตอน:

ขั้นตอนที่ 1) คำนวณอัตราส่วนของความกว้างดั้งเดิม / ความสูงดั้งเดิมของรูปภาพ

ขั้นตอนที่ 2) คูณอัตราส่วน original_width / original_height ด้วยความสูงที่ต้องการใหม่เพื่อให้ได้ความกว้างใหม่ที่สอดคล้องกับความสูงใหม่



-4

ทั้งหมดนี้ใช้ได้สำหรับฉันสำหรับรายการที่สามารถลากได้ - widthRatio: จริง

.appendTo(divwrapper).resizable({
    aspectRatio: true,
    handles: 'se',
    stop: resizestop 
})
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.