| OLD | NEW |
| 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
| 4 | 4 |
| 5 #library('uri_extras'); | 5 #library('uri_extras'); |
| 6 | 6 |
| 7 #import('dart:math'); | 7 #import('dart:math'); |
| 8 #import('dart:uri'); | 8 #import('dart:uri'); |
| 9 | 9 |
| 10 String relativize(Uri base, Uri uri) { | 10 String relativize(Uri base, Uri uri, bool isWindows) { |
| 11 if (base.scheme == 'file' && | 11 if (!base.path.startsWith('/')) { |
| 12 base.scheme == uri.scheme && | 12 // Also throw an exception if [base] or base.path is null. |
| 13 throw new IllegalArgumentException('Expected absolute path: ${base.path}'); |
| 14 } |
| 15 if (!uri.path.startsWith('/')) { |
| 16 // Also throw an exception if [uri] or uri.path is null. |
| 17 throw new IllegalArgumentException('Expected absolute path: ${uri.path}'); |
| 18 } |
| 19 bool equalsNCS(String a, String b) { |
| 20 return a.toLowerCase() == b.toLowerCase(); |
| 21 } |
| 22 |
| 23 String normalize(String path) { |
| 24 if (isWindows) { |
| 25 return path.toLowerCase(); |
| 26 } else { |
| 27 return path; |
| 28 } |
| 29 } |
| 30 |
| 31 if (equalsNCS(base.scheme, 'file') && |
| 32 equalsNCS(base.scheme, uri.scheme) && |
| 13 base.userInfo == uri.userInfo && | 33 base.userInfo == uri.userInfo && |
| 14 base.domain == uri.domain && | 34 equalsNCS(base.domain, uri.domain) && |
| 15 base.port == uri.port && | 35 base.port == uri.port && |
| 16 uri.query == "" && uri.fragment == "") { | 36 uri.query == "" && uri.fragment == "") { |
| 17 if (uri.path.startsWith(base.path)) { | 37 if (normalize(uri.path).startsWith(normalize(base.path))) { |
| 18 return uri.path.substring(base.path.length); | 38 return uri.path.substring(base.path.length); |
| 19 } | 39 } |
| 20 List<String> uriParts = uri.path.split('/'); | 40 List<String> uriParts = uri.path.split('/'); |
| 21 List<String> baseParts = base.path.split('/'); | 41 List<String> baseParts = base.path.split('/'); |
| 22 int common = 0; | 42 int common = 0; |
| 23 int length = min(uriParts.length, baseParts.length); | 43 int length = min(uriParts.length, baseParts.length); |
| 24 while (common < length && uriParts[common] == baseParts[common]) { | 44 while (common < length && |
| 45 normalize(uriParts[common]) == normalize(baseParts[common])) { |
| 25 common++; | 46 common++; |
| 26 } | 47 } |
| 48 if (common == 1 || (isWindows && common == 2)) { |
| 49 // The first part will always be an empty string because the |
| 50 // paths are absolute. On Windows, we must also consider drive |
| 51 // letters or hostnames. |
| 52 return uri.path; |
| 53 } |
| 27 StringBuffer sb = new StringBuffer(); | 54 StringBuffer sb = new StringBuffer(); |
| 28 for (int i = common + 1; i < baseParts.length; i++) { | 55 for (int i = common + 1; i < baseParts.length; i++) { |
| 29 sb.add('../'); | 56 sb.add('../'); |
| 30 } | 57 } |
| 31 for (int i = common; i < uriParts.length - 1; i++) { | 58 for (int i = common; i < uriParts.length - 1; i++) { |
| 32 sb.add('${uriParts[i]}/'); | 59 sb.add('${uriParts[i]}/'); |
| 33 } | 60 } |
| 34 sb.add('${uriParts.last()}'); | 61 sb.add('${uriParts.last()}'); |
| 35 return sb.toString(); | 62 return sb.toString(); |
| 36 } | 63 } |
| 37 return uri.toString(); | 64 return uri.toString(); |
| 38 } | 65 } |
| OLD | NEW |