[javascript] 상대 URL에서 절대 URL 가져 오기. (IE6 문제)

현재 상대 URL을 절대 URL로 ‘변환’하기 위해 다음 함수를 사용하고 있습니다.

function qualifyURL(url) {
    var a = document.createElement('a');
    a.href = url;
    return a.href;
}

이것은 대부분의 브라우저에서 잘 작동하지만 IE6는 여전히 상대 URL을 반환하도록 고집합니다! getAttribute ( ‘href’)를 사용하면 동일합니다.

IE6에서 정규화 된 URL을 얻을 수 있었던 유일한 방법은 img 요소를 만들고 ‘src’속성을 쿼리하는 것입니다. 문제는 서버 요청을 생성한다는 것입니다. 피하고 싶은 것.

그래서 내 질문은 : (서버 요청없이) 상대 URL에서 IE6에서 정규화 된 URL을 얻는 방법이 있습니까?


빠른 정규식 / 문자열 수정을 권장하기 전에 그렇게 간단하지 않다고 확신합니다. 기본 요소 + 이중 기간 상대 URL + 수많은 다른 잠재적 변수가 정말 지옥을 만듭니다!

거대한 정규식 솔루션을 만들지 않고도 할 수있는 방법이 있어야합니다.



답변

얼마나 이상해! 그러나 IE는 DOM 메서드 대신 innerHTML을 사용할 때이를 이해합니다.

function escapeHTML(s) {
    return s.split('&').join('&amp;').split('<').join('&lt;').split('"').join('&quot;');
}
function qualifyURL(url) {
    var el= document.createElement('div');
    el.innerHTML= '<a href="'+escapeHTML(url)+'">x</a>';
    return el.firstChild.href;
}

조금 못 생겼지 만 스스로하는 것보다 더 간결합니다.


답변

브라우저가 <base> 태그를 올바르게 구현하는 한 브라우저는 다음을 수행합니다.

function resolve(url, base_url) {
  var doc      = document
    , old_base = doc.getElementsByTagName('base')[0]
    , old_href = old_base && old_base.href
    , doc_head = doc.head || doc.getElementsByTagName('head')[0]
    , our_base = old_base || doc_head.appendChild(doc.createElement('base'))
    , resolver = doc.createElement('a')
    , resolved_url
    ;
  our_base.href = base_url || '';
  resolver.href = url;
  resolved_url  = resolver.href; // browser magic at work here

  if (old_base) old_base.href = old_href;
  else doc_head.removeChild(our_base);
  return resolved_url;
}

실험 할 수있는 jsfiddle은 다음과 같습니다. http://jsfiddle.net/ecmanaut/RHdnZ/


답변

요소를 복제하기 만하면 IE6에서 작동하도록 만들 수 있습니다.

function qualifyURL(url) {
    var a = document.createElement('a');
    a.href = url;
    return a.cloneNode(false).href;
}

(IE6 및 IE5.5 모드에서 IETester를 사용하여 테스트)


답변

블로그 에서 찾았습니다 @bobince 솔루션과 같은 또 다른 방법을 .

function canonicalize(url) {
    var div = document.createElement('div');
    div.innerHTML = "<a></a>";
    div.firstChild.href = url; // Ensures that the href is properly escaped
    div.innerHTML = div.innerHTML; // Run the current innerHTML back through the parser
    return div.firstChild.href;
}

나는 그것이 큰 문제가 아니라 조금 더 우아하다는 것을 알았습니다.


답변

URI.js 가 문제를 해결하는 것 같습니다.

URI("../foobar.html").absoluteTo("http://example.org/hello/world.html").toString()

http://medialize.github.io/URI.js/docs.html#absoluteto 참조

IE6에서는 테스트되지 않았지만 일반적인 문제를 검색하는 다른 사용자에게 도움이 될 수 있습니다.


답변

나는 실제로 원본 문서를 수정할 필요가 없지만 (일시적으로도 아님) 브라우저의 내장 URL 파싱 등을 사용하는 접근 방식을 원했습니다. 또한 내 자신의 기반을 제공 할 수 있기를 원했습니다 (예 : ecmanaught의 답변). 다소 간단하지만 createHTMLDocument를 사용합니다 (좀 더 호환되도록 createDocument로 대체 할 수 있음).

function absolutize(base, url) {
    d = document.implementation.createHTMLDocument();
    b = d.createElement('base');
    d.head.appendChild(b);
    a = d.createElement('a');
    d.body.appendChild(a);
    b.href = base;
    a.href = url;
    return a.href;
}

http://jsfiddle.net/5u6j403k/


답변

이 솔루션은 모든 브라우저에서 작동합니다.

/**
 * Given a filename for a static resource, returns the resource's absolute
 * URL. Supports file paths with or without origin/protocol.
 */
function toAbsoluteURL (url) {
  // Handle absolute URLs (with protocol-relative prefix)
  // Example: //domain.com/file.png
  if (url.search(/^\/\//) != -1) {
    return window.location.protocol + url
  }

  // Handle absolute URLs (with explicit origin)
  // Example: http://domain.com/file.png
  if (url.search(/:\/\//) != -1) {
    return url
  }

  // Handle absolute URLs (without explicit origin)
  // Example: /file.png
  if (url.search(/^\//) != -1) {
    return window.location.origin + url
  }

  // Handle relative URLs
  // Example: file.png
  var base = window.location.href.match(/(.*\/)/)[0]
  return base + url

그러나 “../file.png”와 같이 “..”가 포함 된 상대 URL은 지원하지 않습니다.