PixelCat

Making the pixels speak

  1. // ==UserScript==
  2. // @name PixelCat
  3. // @namespace https://github.com/pixelcattt/core
  4. // @version 0.0.1
  5. // @description Making the pixels speak
  6. // @author Timothy Lau
  7. // @match https://www.bilibili.com/video/*
  8. // @match https://www.youtube.com/watch?v=*
  9. // @license MIT
  10. // @icon 
  11. // @grant none
  12. // ==/UserScript==
  13.  
  14. // canvas width & height
  15. const width = 640;
  16. const height = 360;
  17.  
  18. // ==! DO NOT CHANGE !==
  19. const sigma = 10;
  20. const rho = 28;
  21. const beta = 8 / 3;
  22. // ==! DO NOT CHANGE !==
  23.  
  24. /**
  25. * Equivalent to the key
  26. * !! You should ensure that the variable of x,y,z is in the range of 0-1.
  27. */
  28. let x = 0.1;
  29. let y = 0;
  30. let z = 0;
  31.  
  32. const chaoticSequence = [];
  33.  
  34. // Generate chaoticSequence
  35. const simulateLorenzAttractor = () => {
  36. const dt = 0.01;
  37. const dx = sigma * (y - x) * dt;
  38. const dy = (x * (rho - z) - y) * dt;
  39. const dz = (x * y - beta * z) * dt;
  40.  
  41. x = x + dx;
  42. y = y + dy;
  43. z = z + dz;
  44.  
  45. chaoticSequence.push(Math.abs(x));
  46. };
  47.  
  48. for (let i = 0; i < width * height * 4; i++) {
  49. simulateLorenzAttractor();
  50. }
  51.  
  52. function decrypt(imageData, seq) {
  53. let index = imageData.data.length / 4;
  54. let pixel_count = imageData.data.length;
  55. for (let i = imageData.data.length; i > 0; i -= 4) {
  56. const newIndex = Math.floor((Math.floor(seq[index] * pixel_count) % pixel_count) / 4) * 4;
  57.  
  58. [imageData.data[i], imageData.data[newIndex]] = [imageData.data[newIndex], imageData.data[i]];
  59. [imageData.data[i + 1], imageData.data[newIndex + 1]] = [imageData.data[newIndex + 1], imageData.data[i + 1]];
  60. [imageData.data[i + 2], imageData.data[newIndex + 2]] = [imageData.data[newIndex + 2], imageData.data[i + 2]];
  61.  
  62. index--;
  63. }
  64. }
  65.  
  66. function injectBilibili() {
  67. var observer = new MutationObserver(mutations => {
  68. const video = document.querySelector('video');
  69. const canvas = document.querySelector('#canvas');
  70. if (video != null && canvas == null) {
  71. let _c = document.createElement('canvas');
  72. _c.id = 'canvas';
  73. _c.style = 'display: block;z-index: 0;position: absolute;right: 0px;bottom: 0px; width: 100%; height: 100%';
  74. video.parentElement.appendChild(_c);
  75. video.style = 'display: none';
  76. video.addEventListener('play', () => {
  77. const _ctx = _c.getContext('2d');
  78. _c.width = width;
  79. _c.height = height;
  80. const loop = function () {
  81. if (!video.paused && !video.ended) {
  82. _ctx.drawImage(video, 0, 0, video.videoWidth, video.videoHeight, 0, 0, _c.width, _c.height);
  83. const imageData = _ctx.getImageData(0, 0, _c.width, _c.height);
  84. decrypt(imageData, chaoticSequence);
  85. _ctx.putImageData(imageData, 0, 0);
  86. requestAnimationFrame(loop);
  87. }
  88. };
  89. loop();
  90. });
  91. }
  92. });
  93. observer.observe(document.body, { childList: true, subtree: true });
  94. }
  95.  
  96. function injectYoutube() {
  97. // Todo
  98. }
  99.  
  100. (function () {
  101. 'use strict';
  102. var strategy = {
  103. 0: injectBilibili,
  104. 1: injectYoutube
  105. }[
  106. GM_info.script.matches
  107. .map(rule => rule.replace(/\.|\*|\/|\?/g, match => ({ '.': '\\.', '*': '.*', '/': '\\/', '?': '\\?' }[match])))
  108. .map(rule => new RegExp(rule))
  109. .map((regExp, index) => (regExp.test(window.location.href) ? index : null))
  110. .filter(index => index != null)
  111. .join()
  112. ];
  113. strategy();
  114. })();