// Process @[youtube](youtubeVideoID) // Process @[vimeo](vimeoVideoID) // Process @[vine](vineVideoID) // Process @[prezi](preziID) // Process @[osf](guid) // Process @[video](video-url) // Process @[video-iframe](video-url) const ytRegex = /^.*((youtu.be\/)|(v\/)|(\/u\/\w\/)|(embed\/)|(watch\?))\??v?=?([^#&?]*).*/; function youtubeParser(url: string) { const match = url.match(ytRegex); return match && match[7].length === 11 ? match[7] : url; } /* eslint-disable max-len */ const vimeoRegex = /https?:\/\/(?:www\.|player\.)?vimeo.com\/(?:channels\/(?:\w+\/)?|groups\/([^/]*)\/videos\/|album\/(\d+)\/video\/|)(\d+)(?:$|\/|\?)/; /* eslint-enable max-len */ function vimeoParser(url: string) { const match = url.match(vimeoRegex); return match && typeof match[3] === 'string' ? match[3] : url; } const vineRegex = /^http(?:s?):\/\/(?:www\.)?vine\.co\/v\/([a-zA-Z0-9]{1,13}).*/; function vineParser(url: string) { const match = url.match(vineRegex); return match && match[1].length === 11 ? match[1] : url; } const preziRegex = /^https:\/\/prezi.com\/(.[^/]+)/; function preziParser(url: string) { const match = url.match(preziRegex); return match ? match[1] : url; } // TODO: Write regex for staging and local servers. const mfrRegex = /^http(?:s?):\/\/(?:www\.)?mfr\.osf\.io\/render\?url=http(?:s?):\/\/osf\.io\/([a-zA-Z0-9]{1,5})\/\?action=download/; function mfrParser(url: string) { const match = url.match(mfrRegex); return match ? match[1] : url; } const EMBED_REGEX = /@\[([a-zA-Z].+)]\([\s]*(.*?)[\s]*[)]/im; function videoEmbed(md: any, options: any) { function videoReturn(state: any, silent: boolean) { let token; const theState = state; const oldPos = state.pos; if (state.src.charCodeAt(oldPos) !== 0x40 /* @ */ || state.src.charCodeAt(oldPos + 1) !== 0x5b /* [ */) { return false; } const match = EMBED_REGEX.exec(state.src.slice(state.pos, state.src.length)); if (!match || match.length < 3) { return false; } const service = match[1]; let videoID = match[2]; const serviceLower = service.toLowerCase(); if (serviceLower === 'youtube') { videoID = youtubeParser(videoID); } else if (serviceLower === 'vimeo') { videoID = vimeoParser(videoID); } else if (serviceLower === 'vine') { videoID = vineParser(videoID); } else if (serviceLower === 'prezi') { videoID = preziParser(videoID); } else if (serviceLower === 'osf') { videoID = mfrParser(videoID); } else if (!options[serviceLower]) { return false; } // If the videoID field is empty, regex currently make it the close parenthesis. if (videoID === ')') { videoID = ''; } const serviceStart = oldPos + 2; const serviceEnd = md.helpers.parseLinkLabel(state, oldPos + 1, false); // // We found the end of the link, and know for a fact it's a valid link; // so all that's left to do is to call tokenizer. // if (!silent) { theState.pos = serviceStart; theState.service = theState.src.slice(serviceStart, serviceEnd); const newState = new theState.md.inline.State(service, theState.md, theState.env, []); newState.md.inline.tokenize(newState); token = theState.push('video', ''); token.videoID = videoID; token.service = service; token.url = match[2]; // eslint-disable-line token.level = theState.level; } theState.pos += theState.src.indexOf(')', theState.pos); return true; } return videoReturn; } function extractVideoParameters(url: string) { const parameterMap = new Map(); const params = url.replace(/&/gi, '&').split(/[#?&]/); if (params.length > 1) { for (let i = 1; i < params.length; i += 1) { const keyValue = params[i].split('='); if (keyValue.length > 1) parameterMap.set(keyValue[0], keyValue[1]); } } return parameterMap; } function videoUrl( service: string, videoID: string, url: string, options: { youtube: { parameters: Record; nocookie: any }; vine: { embed: any }; } ) { switch (service) { case 'youtube': { const parameters = extractVideoParameters(url); if (options.youtube.parameters) { Object.keys(options.youtube.parameters).forEach((key) => { parameters.set(key, options.youtube.parameters[key]); }); } // Start time parameter can have the format t=0m10s or t= in share URLs, // but in embed URLs the parameter must be called 'start' and time must be in seconds const timeParameter = parameters.get('t'); if (timeParameter !== undefined) { let startTime = 0; const timeParts = timeParameter.match(/[0-9]+/g); let j = 0; while (timeParts.length > 0) { /* eslint-disable no-restricted-properties */ startTime += Number(timeParts.pop()) * 60 ** j; /* eslint-enable no-restricted-properties */ j += 1; } parameters.set('start', startTime); parameters.delete('t'); } parameters.delete('v'); parameters.delete('feature'); parameters.delete('origin'); const parameterArray = Array.from(parameters, (p) => p.join('=')); const parameterPos = videoID.indexOf('?'); let finalUrl = 'https://www.youtube'; if (options.youtube.nocookie || url.indexOf('youtube-nocookie.com') > -1) finalUrl += '-nocookie'; finalUrl += `.com/embed/${parameterPos > -1 ? videoID.substr(0, parameterPos) : videoID}`; if (parameterArray.length > 0) finalUrl += `?${parameterArray.join('&')}`; return finalUrl; } case 'vimeo': return `https://player.vimeo.com/video/${videoID}`; case 'vine': return `https://vine.co/v/${videoID}/embed/${options.vine.embed}`; case 'prezi': return ( `https://prezi.com/embed/${videoID}/?bgcolor=ffffff&lock_to_path=0&autoplay=0&autohide_ctrls=0&` + `landing_data=bHVZZmNaNDBIWnNjdEVENDRhZDFNZGNIUE43MHdLNWpsdFJLb2ZHanI5N1lQVHkxSHFxazZ0UUNCRHloSXZROHh3PT0&` + `landing_sign=1kD6c0N6aYpMUS0wxnQjxzSqZlEB8qNFdxtdjYhwSuI` ); case 'osf': return `https://mfr.osf.io/render?url=https://osf.io/${videoID}/?action=download`; default: return service; } } function tokenizeVideo(md: any, options: any) { function tokenizeReturn(tokens: any, idx: any) { const videoID = md.utils.escapeHtml(tokens[idx].videoID); const service = md.utils.escapeHtml(tokens[idx].service).toLowerCase(); const checkUrl = /http(?:s?):\/\/(?:www\.)?[a-zA-Z0-9-:.]{1,}\/render(?:\/)?[a-zA-Z0-9.&;?=:%]{1,}url=http(?:s?):\/\/[a-zA-Z0-9 -:.]{1,}\/[a-zA-Z0-9]{1,5}\/\?[a-zA-Z0-9.=:%]{1,}/; let num; if (service === 'video') { return `