// ==UserScript==
// @author Lete114
// @version 0.1
// @license MIT License
// @name Console Importer
// @description Easily import JS and CSS resources from Chrome console.
// @match *://*/*
// @icon 
// @namespace https://github.com/Lete114/my-tampermonkey-scripts
// @grant unsafeWindow
// ==/UserScript==
;(function () {
'use strict'
// https://github.com/pd4d10/console-importer
const window = unsafeWindow
const tiza = new Tiza()
const PREFIX_TEXT = '[$i]: '
const prefix = tiza.color('blue').text
const strong = tiza.color('blue').bold().text
const error = tiza.color('red').text
const log = (...args) => tiza.log(prefix(PREFIX_TEXT), ...args)
const logError = (...args) => tiza.log(error(PREFIX_TEXT), ...args)
/**
* @type { Set<string> | null }
*/
let lastGlobalVariableSet = null
/**
*
* @param { string } name
* @returns
*/
function createBeforeLoad(name) {
return () => {
lastGlobalVariableSet = new Set(Object.keys(window))
log(strong(name), ' is loading, please be patient...')
}
}
/**
*
* @param { string } name
* @param { string? } url
* @returns
*/
function createOnLoad(name, url) {
return () => {
const urlText = url ? `(${url})` : ''
log(strong(name), `${urlText} is loaded.`)
const currentGlobalVariables = Object.keys(window)
const newGlobalVariables = currentGlobalVariables.filter(
(key) => !lastGlobalVariableSet?.has(key)
)
if (newGlobalVariables.length > 0) {
log(
'The new global variables are as follows: ',
strong(newGlobalVariables.join(',')),
' . Maybe you can use them.'
)
} else {
// maybe css request or script loaded already
}
// Update global variable list
lastGlobalVariableSet = new Set(currentGlobalVariables)
}
}
/**
*
* @param { string } name
* @param { string? } url
* @returns
*/
function createOnError(name, url) {
return () => {
const urlText = url ? `(${strong(url)})` : ''
logError(
'Fail to load ',
strong(name),
', is this URL',
urlText,
' correct?'
)
}
}
// Try to remove referrer for security
// https://imququ.com/post/referrer-policy.html
// https://www.w3.org/TR/referrer-policy/
function addNoReferrerMeta() {
const originMeta =
document.querySelector < HTMLMetaElement > 'meta[name=referrer]'
if (originMeta) {
// If there is already a referrer policy meta tag, save origin content
// and then change it, call `remove` to restore it
const content = originMeta.content
originMeta.content = 'no-referrer'
return function remove() {
originMeta.content = content
}
} else {
// Else, create a new one, call `remove` to delete it
const meta = document.createElement('meta')
meta.name = 'referrer'
meta.content = 'no-referrer'
document.head.appendChild(meta)
return function remove() {
// Removing meta tag directly not work, should set it to default first
meta.content = 'no-referrer-when-downgrade'
document.head.removeChild(meta)
}
}
}
/**
* Insert script tag
* @param { sting } url
* @param { function } onload
* @param { function } onerror
*/
function injectScript(url, onload, onerror) {
const remove = addNoReferrerMeta()
const script = document.createElement('script')
script.src = url
script.onload = onload
script.onerror = onerror
document.body.appendChild(script)
remove()
document.body.removeChild(script)
}
/**
* Insert link tag
* @param { string } url
* @param { function } onload
* @param { function } onerror
*/
function injectStyle(url, onload, onerror) {
const remove = addNoReferrerMeta()
const link = document.createElement('link')
link.href = url
link.rel = 'stylesheet'
// https://developer.mozilla.org/en-US/docs/Web/HTML/Element/link#Stylesheet_load_events
link.onload = onload
link.onerror = onerror
document.head.appendChild(link)
remove()
// Should not remove <link> tag, unlike <script>
}
/**
*
* @param { string } url
* @param {*} beforeLoad
* @param {*} onload
* @param {*} onerror
* @returns
*/
function inject(
url,
beforeLoad = createBeforeLoad(url),
onload = createOnLoad(url),
onerror = createOnError(url)
) {
beforeLoad()
// Handle CSS
if (/\.css$/.test(url)) {
return injectStyle(url, onload, onerror)
}
// Handle JS
return injectScript(url, onload, onerror)
}
/**
* From cdnjs
* https://cdnjs.com/
* @param { string } name
*/
function cdnjs(name) {
log('Searching for ', strong(name), ', please be patient...')
fetch(`https://api.cdnjs.com/libraries?search=${name}`, {
referrerPolicy: 'no-referrer'
})
.then((res) => res.json())
.then(({ results }) => {
if (results.length === 0) {
logError(
'Sorry, ',
strong(name),
' not found, please try another keyword.'
)
return
}
const matchedResult = results.filter((item) => item.name === name)
const { name: exactName, latest: url } = matchedResult[0] || results[0]
if (name !== exactName) {
log(
strong(name),
' not found, import ',
strong(exactName),
' instead.'
)
}
inject(
url,
createBeforeLoad(exactName),
createOnLoad(exactName, url),
createOnError(exactName, url)
)
})
.catch(() => {
logError(
'There appears to be some trouble with your network. If you think this is a bug, please report an issue:'
)
logError('https://github.com/pd4d10/console-importer/issues')
})
}
/**
* From unpkg
* https://unpkg.com
* @param { string } name
*/
function unpkg(name) {
createBeforeLoad(name)()
const url = `https://unpkg.com/${name}`
injectScript(url, createOnLoad(name, url), createOnError(name, url))
}
/**
* https://www.jsdelivr.com/esm
* @param { string } name
* @returns
*/
async function esm(name) {
log(strong(name), '(esm) is loading, please be patient...')
const res = await import(`https://esm.run/${name}`)
return res
}
/**
* Entry
* @param { string } originName
* @returns
*/
function importer(originName) {
if (typeof originName !== 'string') {
throw new Error('Argument should be a string, please check it.')
}
// Trim string
const name = originName.trim()
// If it is a valid URL, inject it directly
if (/^https?:\/\//.test(name)) {
return inject(name)
}
// If version specified, try unpkg
if (name.indexOf('@') !== -1) {
return unpkg(name)
}
return cdnjs(name)
}
importer.cdnjs = cdnjs
importer.unpkg = unpkg
importer.esm = esm
// Do not output annoying ugly string of function content
importer.toString = () => '$i'
// Assign to console
console.$i = importer
// Do not break existing $i
const win = window
if (typeof win.$i === 'undefined') {
win.$i = importer
} else {
log('$i is already in use, please use `console.$i` instead')
}
function Tiza(currentStyles, texts, styles) {
if (currentStyles === void 0) {
currentStyles = []
}
if (texts === void 0) {
texts = []
}
if (styles === void 0) {
styles = []
}
var _this = this
// Get method
this.getCurrentStyles = function () {
return _this._currentStyles
}
this.getTexts = function () {
return _this._texts
}
this.getStyles = function () {
return _this._styles
}
// Push a style to current Styles
this.style = function (s) {
return new Tiza(
_this._currentStyles.concat([s]),
_this._texts,
_this._styles
)
}
// Alias for style method
this.color = function (c) {
return _this.style('color:' + c)
}
this.bgColor = function (c) {
return _this.style('background-color:' + c)
}
this.bold = function () {
return _this.style('font-weight:bold')
}
this.italic = function () {
return _this.style('font-style:italic')
}
this.size = function (n) {
var s = typeof n === 'number' ? n + 'px' : n // Convert number to px
return _this.style('font-size:' + s)
}
// Clear all current styles
this.reset = function () {
return new Tiza([], _this._texts, _this._styles)
}
this.text = function () {
var args = []
for (var _i = 0; _i < arguments.length; _i++) {
args[_i] = arguments[_i]
}
var texts = _this._texts.slice()
var styles = _this._styles.slice()
args.forEach(function (arg) {
if (arg instanceof Tiza) {
texts.push.apply(texts, arg.getTexts())
styles.push.apply(styles, arg.getStyles())
} else {
texts.push(arg)
styles.push(_this._currentStyles.join(';'))
}
})
return new Tiza(_this._currentStyles, texts, styles)
}
this.space = function (count) {
if (count === void 0) {
count = 1
}
return _this.text(repeat(' ', count))
}
this.newline = function (count) {
if (count === void 0) {
count = 1
}
return _this.text(repeat('\n', count))
}
this._output = function (type) {
return function () {
var args = []
for (var _i = 0; _i < arguments.length; _i++) {
args[_i] = arguments[_i]
}
var ins = _this.text.apply(_this, args)
console[type].apply(
console,
[
ins
.getTexts()
.map(function (t) {
return '%c' + t
})
.join('')
].concat(ins._styles)
)
return new Tiza(ins.getCurrentStyles())
}
}
this.log = this._output('log')
this.info = this._output('info')
this.warn = this._output('warn')
this.error = this._output('error')
this._currentStyles = currentStyles
this._texts = texts
this._styles = styles
}
})()