문제

PHP 기반 웹 사이트에서 사용자가 간단한 양식을 작성한 후 다운로드 패키지를 보내고 싶습니다.사이트에서 시작되는 다운로드는 "다운로드가 잠시 후 시작됩니다."라고 말하는 download.com과 같은 사이트와 유사해야 합니다.

몇 가지 가능한 접근법 브라우저 호환성에 대해 알고 있습니다(빠른 테스트를 기반으로 함).

1) window.open 새 파일을 가리킵니다.

- FireFox 3 blocks this.  
 - IE6 blocks this.  
 - IE7 blocks this.

2) 새 파일을 가리키는 iframe을 만듭니다.

- FireFox 3 seems to think this is OK. (Maybe it's because I already accepted it once?)  
 - IE6 blocks this.  
 - IE7 blocks this.

How can I do this so that at least these three browsers will not object? 

보너스:브라우저 조건문이 필요하지 않은 방법이 있습니까?

(나는 download.com이 조건부로 두 가지 방법을 모두 사용한다고 생각하지만 둘 중 하나를 작동시킬 수는 없습니다.)

답변 및 설명:

Q: "Why not point the current window to the file?"  
A: That might work, but in this particular case, I want to show them some other content while their download starts - for example, "would you like to donate to this project?"

업데이트:나는 이 접근법을 포기했습니다.이유는 아래 내 답변을 참조하세요.

도움이 되었습니까?

해결책

대부분의 브라우저가 지원하는 메타 새로 고침을 수행할 수도 있습니다.Download.com은 noscript 태그에 하나를 배치합니다.

<meta http-equiv="refresh" content="5;url=/download.php?doc=123.zip"/>

다른 팁

업데이트:나는 이 접근 방식을 포기하고 대신 사용자에게 실제 파일에 대한 링크를 제공하기로 결정했습니다.내 추론은 다음과 같습니다.

서버에서 시작하는 다운로드에 대한 초기 시도가 브라우저에 의해 차단되었습니다.그 때문에 다음과 같은 생각이 들었습니다."브라우저가 맞아요.어떻게 그것 이것이 합법적인 다운로드라는 것을 알고 계십니까?그것 ~해야 한다 분명히 사용자가 시작하지 않은 다운로드를 차단합니다."

서버 시작 다운로드에 사용할 수 있는 모든 방법은 또한 악성 코드를 보내려는 사람이 사용합니다.따라서 사용자가 링크를 클릭하여 파일을 구체적으로 요청할 때만 다운로드가 이루어져야 합니다.

동의하지 않으셔도 되며, 그래도 다운로드를 시작하고 싶다면 이 스레드가 도움이 되기를 바랍니다.

나는 일반적으로 적절한 Content-Type을 사용하여 파일을 브라우저에 직접 출력하는 PHP 스크립트를 가지고 있습니다.

if(file_exists($filename)) {
    header("Pragma:  public");
    header("Expires:  0");
    header("Cache-Control:  must-revalidate, pre-check=0");
    header("Cache-Control:  private", false);
    header("Content-Type:  " . $content-type);
    header("Content-Disposition:  attachment; filename=\"" . basename($filename) . "\";" );
    header("Content-Transfer-Encoding:  binary");
    header("Content-Length:  " . filesize($filename));

    readfile("$filename");
}else{
    print "ERROR:  the file " . basename($filename) . " could not be downloaded because it did not exist.";
}

유일한 단점은 이것이 HTTP 헤더를 설정하기 때문에 다른 출력이 있기 전에 호출된다는 것입니다.

그러나 PHP 다운로드 페이지에 대한 링크가 있으면 현재 페이지의 내용을 망칠 필요 없이 브라우저가 다운로드 상자를 팝업으로 표시하게 됩니다.

한 가지 문제는 헤더가 "올바르게" 설정되지 않은 경우 IE(특히 버전 6)에서 문제가 발생할 수 있다는 것입니다.

올바른 Content-Type을 설정했는지 확인하고 최소한 IE의 캐시 옵션을 다음으로 설정하는 것도 고려하십시오. 허용하다 캐싱.파일이 파일인 경우 사용자는 저장하는 대신 열 수 있습니다(예:MS Word 문서) IE의 초기 버전은 캐시에 다운로드된 파일을 가리키는 "열기" 요청을 해당 앱에 전달하므로 파일을 캐시해야 합니다.

관련 문제도 있습니다. IE6 사용자의 캐시가 가득 차면 파일이 제대로 저장되지 않습니다. 따라서 해당 앱이 파일을 열 때 파일이 손상되었다고 불평하게 됩니다.

다운로드 시 gzip 작업도 끄고 싶을 수도 있습니다(IE의 경우).

IE6/IE7 모두 대용량 다운로드에 문제가 있습니다(예:4.x Gigs...) IE에는 다운로드 관리자도 없기 때문에 가능성 있는 시나리오는 아니지만 주의해야 할 사항이 있습니다.

마지막으로, IE6은 중첩된 iframe 내에서 시작된 다운로드 "푸시"를 제대로 처리하지 못하는 경우가 있습니다.무엇이 문제를 유발하는지 정확히 모르겠지만 IE6을 사용하면 이 시나리오를 피하는 것이 더 쉽다는 것을 알았습니다.

호이!

@나단:나는 정확히 그렇게 하기로 결정했습니다:내 "getfile.php"가 필요한 모든 항목을 로드한 다음 다음을 수행하도록 합니다.

header("Location: ./$path/$filename");

브라우저 자체가 옳다고 생각하는 모든 작업을 파일에서 직접 수행하도록 허용합니다.이것은 Opera에서도 나와 잘 작동합니다.

그러나 이는 파일에 대한 직접적인 접근이 허용되지 않는 환경에서는 문제가 될 수 있으며, 이 경우 다른 방법을 찾아야 합니다!(Discordia에게 감사드립니다. 제 파일은 공개 PDF입니다!)

감사합니다, 바스티

새 파일을 가리키도록 위치를 변경하는 것은 어떻습니까?(예:window.location을 변경하여)

나는 항상 파일을 가리키는 iframe을 만들었습니다.

<iframe src="/download.exe" frameborder="0" height="0" width="0"><a href="/download.exe">Click here to download.</a></iframe>

현재 창에서 다운로드를 가리키지 않는 것과 관련됩니다.내 경험에 따르면 다운로드 시(올바른 헤더를 전송하는 한) 실제로 브라우저 창을 업데이트하지 않기 때문에 "기부해 주세요" 페이지를 계속 표시할 수 있습니다.내 사이트 중 하나에서 csv 내보내기를 위해 이 작업을 수행하는데, 사용자에 관한 한 안전한 파일 창이 팝업됩니다.그래서 나는 Soldarnal이 보여준 것처럼 간단한 메타 리디렉션을 권장합니다.

요약하자면 두 가지 목표가 있습니다.

  1. 다운로드 프로세스 시작
  2. 사용자에게 기부 옵션이 있는 페이지 표시

이를 달성하기 위해 다음을 수행합니다.

사용자가 양식을 제출하면 기부 옵션과 다운로드가 5초 안에 시작된다는 텍스트가 포함된 결과 페이지가 표시됩니다.그리고 이 페이지의 헤드 섹션에 Soldarnal이 말한 대로 META 코드를 입력합니다.<meta http-equiv="refresh" content="5;url=/download.php?doc=123.zip>

그리고 그게 다야.

<a href="normaldownload.zip" onclick="use_dhtml_or_ajax_to_display_page()">

다운로드가 저장되면 현재 페이지는 영향을 받지 않습니다.다운로드가 동일한 창에서 열리지 않는지 확인하십시오(적절한 MIME 유형 또는 Content-Disposition) 그러면 무엇이든 보여줄 수 있습니다.

더 많은 완전한 답변 보기

Javascript/jQuery를 사용하여 다운로드를 시작할 수 있습니다.여기에 예가 있습니다. Ajax 요청을 제거하고 setTimeout() 블록만 사용할 수 있습니다.

$("btnDownloadCSV").on('click', function() {
    $.ajax({
        url: "php_backend/get_download_url",
        type: 'post',
        contentType: "application/x-www-form-urlencoded",
        data: {somedata: "somedata"},
        success: function(data) {
            // If iFrame already exists, remove it.
            if($("[id^='iframeTempCSV_"]).length) { 
                $("[id^='iframeTempCSV_"]).remove();
            }
            setTimeout(function() {
                //  If I'm creating an iframe with the same id, it will permit download only the first time.
                // So randHashId appended to ID to trick the browser.
                var randHashId = Math.random().toString(36).substr(2);
                // Create a fresh iFrame for auto-downloading CSV
                $('<iframe id="iframeTempCSV_'+randHashId+'" style="display:none;" src="'+data.filepath+'"></iframe>').appendTo('body');
            }, 1000);
        },
        error: function(xhr, textStatus, errorThrown) {
           console.error("Error downloading...");
       }
    });
});
라이센스 : CC-BY-SA ~와 함께 속성
제휴하지 않습니다 StackOverflow
scroll top