diff2html/docs/demo.js

654 lines
19 KiB
JavaScript
Raw Normal View History

2016-10-12 21:52:59 +00:00
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
(function(self) {
'use strict';
if (self.fetch) {
return
}
var support = {
searchParams: 'URLSearchParams' in self,
iterable: 'Symbol' in self && 'iterator' in Symbol,
blob: 'FileReader' in self && 'Blob' in self && (function() {
try {
new Blob()
return true
} catch(e) {
return false
}
})(),
formData: 'FormData' in self,
arrayBuffer: 'ArrayBuffer' in self
}
2016-12-11 15:21:58 +00:00
if (support.arrayBuffer) {
var viewClasses = [
'[object Int8Array]',
'[object Uint8Array]',
'[object Uint8ClampedArray]',
'[object Int16Array]',
'[object Uint16Array]',
'[object Int32Array]',
'[object Uint32Array]',
'[object Float32Array]',
'[object Float64Array]'
]
var isDataView = function(obj) {
return obj && DataView.prototype.isPrototypeOf(obj)
}
var isArrayBufferView = ArrayBuffer.isView || function(obj) {
return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1
}
}
2016-10-12 21:52:59 +00:00
function normalizeName(name) {
if (typeof name !== 'string') {
name = String(name)
}
if (/[^a-z0-9\-#$%&'*+.\^_`|~]/i.test(name)) {
throw new TypeError('Invalid character in header field name')
}
return name.toLowerCase()
}
function normalizeValue(value) {
if (typeof value !== 'string') {
value = String(value)
}
return value
}
// Build a destructive iterator for the value list
function iteratorFor(items) {
var iterator = {
next: function() {
var value = items.shift()
return {done: value === undefined, value: value}
}
}
if (support.iterable) {
iterator[Symbol.iterator] = function() {
return iterator
}
}
return iterator
}
2016-10-12 21:52:59 +00:00
function Headers(headers) {
this.map = {}
if (headers instanceof Headers) {
headers.forEach(function(value, name) {
this.append(name, value)
}, this)
} else if (headers) {
Object.getOwnPropertyNames(headers).forEach(function(name) {
this.append(name, headers[name])
}, this)
}
}
Headers.prototype.append = function(name, value) {
name = normalizeName(name)
value = normalizeValue(value)
2016-12-11 15:21:58 +00:00
var oldValue = this.map[name]
this.map[name] = oldValue ? oldValue+','+value : value
2016-10-12 21:52:59 +00:00
}
Headers.prototype['delete'] = function(name) {
delete this.map[normalizeName(name)]
}
Headers.prototype.get = function(name) {
2016-12-11 15:21:58 +00:00
name = normalizeName(name)
return this.has(name) ? this.map[name] : null
2016-10-12 21:52:59 +00:00
}
Headers.prototype.has = function(name) {
return this.map.hasOwnProperty(normalizeName(name))
}
Headers.prototype.set = function(name, value) {
2016-12-11 15:21:58 +00:00
this.map[normalizeName(name)] = normalizeValue(value)
2016-10-12 21:52:59 +00:00
}
Headers.prototype.forEach = function(callback, thisArg) {
2016-12-11 15:21:58 +00:00
for (var name in this.map) {
if (this.map.hasOwnProperty(name)) {
callback.call(thisArg, this.map[name], name, this)
}
}
2016-10-12 21:52:59 +00:00
}
Headers.prototype.keys = function() {
var items = []
this.forEach(function(value, name) { items.push(name) })
return iteratorFor(items)
}
Headers.prototype.values = function() {
var items = []
this.forEach(function(value) { items.push(value) })
return iteratorFor(items)
}
Headers.prototype.entries = function() {
var items = []
this.forEach(function(value, name) { items.push([name, value]) })
return iteratorFor(items)
}
if (support.iterable) {
Headers.prototype[Symbol.iterator] = Headers.prototype.entries
}
2016-10-12 21:52:59 +00:00
function consumed(body) {
if (body.bodyUsed) {
return Promise.reject(new TypeError('Already read'))
}
body.bodyUsed = true
}
function fileReaderReady(reader) {
return new Promise(function(resolve, reject) {
reader.onload = function() {
resolve(reader.result)
}
reader.onerror = function() {
reject(reader.error)
}
})
}
function readBlobAsArrayBuffer(blob) {
var reader = new FileReader()
2016-12-11 15:21:58 +00:00
var promise = fileReaderReady(reader)
2016-10-12 21:52:59 +00:00
reader.readAsArrayBuffer(blob)
2016-12-11 15:21:58 +00:00
return promise
2016-10-12 21:52:59 +00:00
}
function readBlobAsText(blob) {
var reader = new FileReader()
2016-12-11 15:21:58 +00:00
var promise = fileReaderReady(reader)
2016-10-12 21:52:59 +00:00
reader.readAsText(blob)
2016-12-11 15:21:58 +00:00
return promise
}
function readArrayBufferAsText(buf) {
var view = new Uint8Array(buf)
var chars = new Array(view.length)
for (var i = 0; i < view.length; i++) {
chars[i] = String.fromCharCode(view[i])
}
return chars.join('')
}
function bufferClone(buf) {
if (buf.slice) {
return buf.slice(0)
} else {
var view = new Uint8Array(buf.byteLength)
view.set(new Uint8Array(buf))
return view.buffer
}
2016-10-12 21:52:59 +00:00
}
function Body() {
this.bodyUsed = false
this._initBody = function(body) {
this._bodyInit = body
2016-12-11 15:21:58 +00:00
if (!body) {
this._bodyText = ''
} else if (typeof body === 'string') {
2016-10-12 21:52:59 +00:00
this._bodyText = body
} else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
this._bodyBlob = body
} else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
this._bodyFormData = body
} else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
this._bodyText = body.toString()
2016-12-11 15:21:58 +00:00
} else if (support.arrayBuffer && support.blob && isDataView(body)) {
this._bodyArrayBuffer = bufferClone(body.buffer)
// IE 10-11 can't handle a DataView body.
this._bodyInit = new Blob([this._bodyArrayBuffer])
} else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {
this._bodyArrayBuffer = bufferClone(body)
2016-10-12 21:52:59 +00:00
} else {
throw new Error('unsupported BodyInit type')
}
if (!this.headers.get('content-type')) {
if (typeof body === 'string') {
this.headers.set('content-type', 'text/plain;charset=UTF-8')
} else if (this._bodyBlob && this._bodyBlob.type) {
this.headers.set('content-type', this._bodyBlob.type)
} else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8')
2016-10-12 21:52:59 +00:00
}
}
}
if (support.blob) {
this.blob = function() {
var rejected = consumed(this)
if (rejected) {
return rejected
}
if (this._bodyBlob) {
return Promise.resolve(this._bodyBlob)
2016-12-11 15:21:58 +00:00
} else if (this._bodyArrayBuffer) {
return Promise.resolve(new Blob([this._bodyArrayBuffer]))
2016-10-12 21:52:59 +00:00
} else if (this._bodyFormData) {
throw new Error('could not read FormData body as blob')
} else {
return Promise.resolve(new Blob([this._bodyText]))
}
}
this.arrayBuffer = function() {
2016-12-11 15:21:58 +00:00
if (this._bodyArrayBuffer) {
return consumed(this) || Promise.resolve(this._bodyArrayBuffer)
2016-10-12 21:52:59 +00:00
} else {
2016-12-11 15:21:58 +00:00
return this.blob().then(readBlobAsArrayBuffer)
2016-10-12 21:52:59 +00:00
}
}
2016-12-11 15:21:58 +00:00
}
this.text = function() {
var rejected = consumed(this)
if (rejected) {
return rejected
}
if (this._bodyBlob) {
return readBlobAsText(this._bodyBlob)
} else if (this._bodyArrayBuffer) {
return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))
} else if (this._bodyFormData) {
throw new Error('could not read FormData body as text')
} else {
return Promise.resolve(this._bodyText)
2016-10-12 21:52:59 +00:00
}
}
if (support.formData) {
this.formData = function() {
return this.text().then(decode)
}
}
this.json = function() {
return this.text().then(JSON.parse)
}
return this
}
// HTTP methods whose capitalization should be normalized
var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']
function normalizeMethod(method) {
var upcased = method.toUpperCase()
return (methods.indexOf(upcased) > -1) ? upcased : method
}
function Request(input, options) {
options = options || {}
var body = options.body
2016-12-11 15:21:58 +00:00
if (typeof input === 'string') {
this.url = input
} else {
2016-10-12 21:52:59 +00:00
if (input.bodyUsed) {
throw new TypeError('Already read')
}
this.url = input.url
this.credentials = input.credentials
if (!options.headers) {
this.headers = new Headers(input.headers)
}
this.method = input.method
this.mode = input.mode
2016-12-11 15:21:58 +00:00
if (!body && input._bodyInit != null) {
2016-10-12 21:52:59 +00:00
body = input._bodyInit
input.bodyUsed = true
}
}
this.credentials = options.credentials || this.credentials || 'omit'
if (options.headers || !this.headers) {
this.headers = new Headers(options.headers)
}
this.method = normalizeMethod(options.method || this.method || 'GET')
this.mode = options.mode || this.mode || null
this.referrer = null
if ((this.method === 'GET' || this.method === 'HEAD') && body) {
throw new TypeError('Body not allowed for GET or HEAD requests')
}
this._initBody(body)
}
Request.prototype.clone = function() {
2016-12-11 15:21:58 +00:00
return new Request(this, { body: this._bodyInit })
2016-10-12 21:52:59 +00:00
}
function decode(body) {
var form = new FormData()
body.trim().split('&').forEach(function(bytes) {
if (bytes) {
var split = bytes.split('=')
var name = split.shift().replace(/\+/g, ' ')
var value = split.join('=').replace(/\+/g, ' ')
form.append(decodeURIComponent(name), decodeURIComponent(value))
}
})
return form
}
2016-12-11 15:21:58 +00:00
function parseHeaders(rawHeaders) {
var headers = new Headers()
rawHeaders.split('\r\n').forEach(function(line) {
var parts = line.split(':')
var key = parts.shift().trim()
if (key) {
var value = parts.join(':').trim()
headers.append(key, value)
}
2016-10-12 21:52:59 +00:00
})
2016-12-11 15:21:58 +00:00
return headers
2016-10-12 21:52:59 +00:00
}
Body.call(Request.prototype)
function Response(bodyInit, options) {
if (!options) {
options = {}
}
this.type = 'default'
2016-12-11 15:21:58 +00:00
this.status = 'status' in options ? options.status : 200
2016-10-12 21:52:59 +00:00
this.ok = this.status >= 200 && this.status < 300
2016-12-11 15:21:58 +00:00
this.statusText = 'statusText' in options ? options.statusText : 'OK'
this.headers = new Headers(options.headers)
2016-10-12 21:52:59 +00:00
this.url = options.url || ''
this._initBody(bodyInit)
}
Body.call(Response.prototype)
Response.prototype.clone = function() {
return new Response(this._bodyInit, {
status: this.status,
statusText: this.statusText,
headers: new Headers(this.headers),
url: this.url
})
}
Response.error = function() {
var response = new Response(null, {status: 0, statusText: ''})
response.type = 'error'
return response
}
var redirectStatuses = [301, 302, 303, 307, 308]
Response.redirect = function(url, status) {
if (redirectStatuses.indexOf(status) === -1) {
throw new RangeError('Invalid status code')
}
return new Response(null, {status: status, headers: {location: url}})
}
self.Headers = Headers
self.Request = Request
self.Response = Response
self.fetch = function(input, init) {
return new Promise(function(resolve, reject) {
2016-12-11 15:21:58 +00:00
var request = new Request(input, init)
2016-10-12 21:52:59 +00:00
var xhr = new XMLHttpRequest()
xhr.onload = function() {
var options = {
status: xhr.status,
2016-10-12 21:52:59 +00:00
statusText: xhr.statusText,
2016-12-11 15:21:58 +00:00
headers: parseHeaders(xhr.getAllResponseHeaders() || '')
2016-10-12 21:52:59 +00:00
}
2016-12-11 15:21:58 +00:00
options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL')
2016-10-12 21:52:59 +00:00
var body = 'response' in xhr ? xhr.response : xhr.responseText
resolve(new Response(body, options))
}
xhr.onerror = function() {
reject(new TypeError('Network request failed'))
}
xhr.ontimeout = function() {
reject(new TypeError('Network request failed'))
}
xhr.open(request.method, request.url, true)
if (request.credentials === 'include') {
xhr.withCredentials = true
}
if ('responseType' in xhr && support.blob) {
xhr.responseType = 'blob'
}
request.headers.forEach(function(value, name) {
xhr.setRequestHeader(name, value)
})
xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)
})
}
self.fetch.polyfill = true
})(typeof self !== 'undefined' ? self : this);
},{}],2:[function(require,module,exports){
2016-10-12 22:55:31 +00:00
/* global Diff2HtmlUI */
2016-10-12 21:52:59 +00:00
/*
* Example URLs:
*
* https://github.com/rtfpessoa/diff2html/commit/7d02e67f3b3386ac5d804f974d025cd7a1165839
* https://github.com/rtfpessoa/diff2html/pull/106
*
* https://gitlab.com/gitlab-org/gitlab-ce/commit/4e963fed42ad518caa7353d361a38a1250c99c41
* https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/6763
*
* https://bitbucket.org/atlassian/amps/commits/52c38116f12475f75af4a147b7a7685478b83eca
* https://bitbucket.org/atlassian/amps/pull-requests/236
*/
2016-10-12 22:55:31 +00:00
$(document).ready(function() {
2016-10-12 21:52:59 +00:00
// Improves browser compatibility
require('whatwg-fetch');
var searchParam = 'diff';
2016-10-12 21:52:59 +00:00
var $container = $('.container');
2016-10-12 22:55:31 +00:00
var $url = $('#url');
var $outputFormat = $('#diff-url-options-output-format');
var $showFiles = $('#diff-url-options-show-files');
var $matching = $('#diff-url-options-matching');
var $wordThreshold = $('#diff-url-options-match-words-threshold');
var $matchingMaxComparisons = $('#diff-url-options-matching-max-comparisons');
2016-10-12 21:52:59 +00:00
if (window.location.search) {
var url = getUrlFromSearch(window.location.search);
$url.val(url);
smartDraw(url);
2016-10-12 21:52:59 +00:00
}
bind();
2016-10-12 21:52:59 +00:00
$outputFormat
.add($showFiles)
.add($matching)
.add($wordThreshold)
.add($matchingMaxComparisons)
2016-10-12 22:55:31 +00:00
.change(function() {
smartDraw();
2016-10-12 21:52:59 +00:00
});
function getUrlFromSearch(search) {
try {
return search
.split('?')[1]
.split(searchParam + '=')[1]
.split('&')[0];
} catch (_ignore) {
}
return null;
}
2016-10-12 21:52:59 +00:00
function bind() {
2016-10-12 22:55:31 +00:00
$('#url-btn').click(function(e) {
2016-10-12 21:52:59 +00:00
e.preventDefault();
var url = $url.val();
smartDraw(url);
2016-10-12 21:52:59 +00:00
});
2016-10-12 22:17:39 +00:00
2016-10-12 22:55:31 +00:00
$url.on('paste', function(e) {
var url = e.originalEvent.clipboardData.getData('Text');
smartDraw(url);
2016-10-12 22:17:39 +00:00
});
2016-10-12 21:52:59 +00:00
}
function prepareUrl(url) {
var fetchUrl;
var headers = new Headers();
2016-10-12 21:52:59 +00:00
var githubCommitUrl = /^https?:\/\/(?:www\.)?github\.com\/(.*?)\/(.*?)\/commit\/(.*?)(?:\.diff)?(?:\.patch)?(?:\/.*)?$/;
var githubPrUrl = /^https?:\/\/(?:www\.)?github\.com\/(.*?)\/(.*?)\/pull\/(.*?)(?:\.diff)?(?:\.patch)?(?:\/.*)?$/;
var gitlabCommitUrl = /^https?:\/\/(?:www\.)?gitlab\.com\/(.*?)\/(.*?)\/commit\/(.*?)(?:\.diff)?(?:\.patch)?(?:\/.*)?$/;
var gitlabPrUrl = /^https?:\/\/(?:www\.)?gitlab\.com\/(.*?)\/(.*?)\/merge_requests\/(.*?)(?:\.diff)?(?:\.patch)?(?:\/.*)?$/;
var bitbucketCommitUrl = /^https?:\/\/(?:www\.)?bitbucket\.org\/(.*?)\/(.*?)\/commits\/(.*?)(?:\/raw)?(?:\/.*)?$/;
var bitbucketPrUrl = /^https?:\/\/(?:www\.)?bitbucket\.org\/(.*?)\/(.*?)\/pull-requests\/(.*?)(?:\/.*)?$/;
function gitLabUrlGen(userName, projectName, type, value) {
return 'https://crossorigin.me/https://gitlab.com/' + userName + '/' + projectName + '/' + type + '/' + value + '.diff';
}
function gitHubUrlGen(userName, projectName, type, value) {
headers.append('Accept', 'application/vnd.github.v3.diff');
return 'https://api.github.com/repos/' + userName + '/' + projectName + '/' + type + '/' + value;
2016-10-12 21:52:59 +00:00
}
function bitbucketUrlGen(userName, projectName, type, value) {
2016-10-12 22:55:31 +00:00
var baseUrl = 'https://bitbucket.org/api/2.0/repositories/';
if (type === 'pullrequests') {
return baseUrl + userName + '/' + projectName + '/pullrequests/' + value + '/diff';
2016-10-12 21:52:59 +00:00
}
2016-10-12 22:55:31 +00:00
return baseUrl + userName + '/' + projectName + '/diff/' + value;
2016-10-12 21:52:59 +00:00
}
var values;
if ((values = githubCommitUrl.exec(url))) {
fetchUrl = gitHubUrlGen(values[1], values[2], 'commits', values[3]);
2016-10-12 21:52:59 +00:00
} else if ((values = githubPrUrl.exec(url))) {
fetchUrl = gitHubUrlGen(values[1], values[2], 'pulls', values[3]);
2016-10-12 21:52:59 +00:00
} else if ((values = gitlabCommitUrl.exec(url))) {
fetchUrl = gitLabUrlGen(values[1], values[2], 'commit', values[3]);
2016-10-12 21:52:59 +00:00
} else if ((values = gitlabPrUrl.exec(url))) {
fetchUrl = gitLabUrlGen(values[1], values[2], 'merge_requests', values[3]);
2016-10-12 21:52:59 +00:00
} else if ((values = bitbucketCommitUrl.exec(url))) {
fetchUrl = bitbucketUrlGen(values[1], values[2], 'commit', values[3]);
2016-10-12 21:52:59 +00:00
} else if ((values = bitbucketPrUrl.exec(url))) {
fetchUrl = bitbucketUrlGen(values[1], values[2], 'pullrequests', values[3]);
2016-10-12 21:52:59 +00:00
} else {
2016-10-12 22:55:31 +00:00
console.info('Could not parse url, using the provided url.');
2016-10-15 11:52:16 +00:00
fetchUrl = 'https://crossorigin.me/' + url;
2016-10-12 21:52:59 +00:00
}
return {
2016-10-15 11:41:15 +00:00
originalUrl: url,
url: fetchUrl,
headers: headers
};
2016-10-12 21:52:59 +00:00
}
2016-10-15 11:41:15 +00:00
function smartDraw(urlOpt) {
var url = urlOpt || $url.val();
var req = prepareUrl(url);
draw(req);
}
function draw(req) {
2016-10-15 11:41:15 +00:00
if (!validateUrl(req.url)) {
2016-10-15 20:47:32 +00:00
console.error('Invalid url provided!');
2016-10-15 11:41:15 +00:00
return;
}
2016-10-15 11:56:55 +00:00
if (validateUrl(req.originalUrl)) updateUrl(req.originalUrl);
2016-10-15 11:41:15 +00:00
2016-10-12 21:52:59 +00:00
var outputFormat = $outputFormat.val();
2016-10-12 22:55:31 +00:00
var showFiles = $showFiles.is(':checked');
2016-10-12 21:52:59 +00:00
var matching = $matching.val();
var wordThreshold = $wordThreshold.val();
var matchingMaxComparisons = $matchingMaxComparisons.val();
fetch(req.url, {
method: 'GET',
headers: req.headers,
mode: 'cors',
cache: 'default'
})
2016-10-12 22:55:31 +00:00
.then(function(res) {
return res.text();
2016-10-12 21:52:59 +00:00
})
2016-10-12 22:55:31 +00:00
.then(function(data) {
2016-10-12 21:52:59 +00:00
var container = '#url-diff-container';
var diff2htmlUi = new Diff2HtmlUI({diff: data});
2016-10-12 22:55:31 +00:00
if (outputFormat === 'side-by-side') {
2016-10-15 21:18:16 +00:00
$container.css({'width': '100%'});
2016-10-12 21:52:59 +00:00
} else {
2016-10-15 21:18:16 +00:00
$container.css({'width': ''});
2016-10-12 21:52:59 +00:00
}
diff2htmlUi.draw(container, {
outputFormat: outputFormat,
showFiles: showFiles,
matching: matching,
matchWordsThreshold: wordThreshold,
2016-10-12 22:17:39 +00:00
matchingMaxComparisons: matchingMaxComparisons,
synchronisedScroll: true
2016-10-12 21:52:59 +00:00
});
diff2htmlUi.fileListCloseable(container, false);
diff2htmlUi.highlightCode(container);
});
}
2016-10-15 11:41:15 +00:00
function validateUrl(url) {
return /^(?:(?:(?:https?|ftp):)?\/\/)(?:\S+(?::\S*)?@)?(?:(?!(?:10|127)(?:\.\d{1,3}){3})(?!(?:169\.254|192\.168)(?:\.\d{1,3}){2})(?!172\.(?:1[6-9]|2\d|3[0-1])(?:\.\d{1,3}){2})(?:[1-9]\d?|1\d\d|2[01]\d|22[0-3])(?:\.(?:1?\d{1,2}|2[0-4]\d|25[0-5])){2}(?:\.(?:[1-9]\d?|1\d\d|2[0-4]\d|25[0-4]))|(?:(?:[a-z\u00a1-\uffff0-9]-*)*[a-z\u00a1-\uffff0-9]+)(?:\.(?:[a-z\u00a1-\uffff0-9]-*)*[a-z\u00a1-\uffff0-9]+)*(?:\.(?:[a-z\u00a1-\uffff]{2,})).?)(?::\d{2,5})?(?:[/?#]\S*)?$/i.test(url);
}
function updateUrl(url) {
var currentUrl = getUrlFromSearch(window.location.search);
if (currentUrl === url) return;
window.location = 'demo.html?' + searchParam + '=' + url;
2016-10-15 11:41:15 +00:00
}
2016-10-12 21:52:59 +00:00
});
},{"whatwg-fetch":1}]},{},[2]);