Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
2361df7da9 | ||
|
|
74a50ab8f6 | ||
|
|
0cf53e8fe0 | ||
|
|
c3f538fe15 | ||
|
|
629bbf090f | ||
|
|
c2d520b753 |
10
README.md
10
README.md
@ -6,9 +6,19 @@ Chrome расширение для улучшенного просмотра Ins
|
|||||||
|
|
||||||
- **Управление громкостью** - Вертикальный слайдер для точной настройки громкости видео
|
- **Управление громкостью** - Вертикальный слайдер для точной настройки громкости видео
|
||||||
- **Загрузка роликов** - Скачивайте рилсы одним кликом
|
- **Загрузка роликов** - Скачивайте рилсы одним кликом
|
||||||
|
- **Перемотка видео** - Используйте слайдер для перемотки рилса, если пропустили интересный момент, больше не придется пересматривать всё видео!
|
||||||
|
|
||||||
## Установка
|
## Установка
|
||||||
|
|
||||||
|
### Использование готового расширения
|
||||||
|
|
||||||
|
1. Скачайте последнюю версию расширения из [релизов на GitHub](https://github.com/ShiftyX1/reels-master/releases)
|
||||||
|
2. Распакуйте архив в удобное место на вашем компьютере
|
||||||
|
3. Откройте Chrome и перейдите на страницу расширений: `chrome://extensions/`
|
||||||
|
4. Включите "Режим разработчика" (Developer mode) в правом верхнем углу
|
||||||
|
5. Нажмите "Загрузить распакованное расширение" (Load unpacked)
|
||||||
|
6. Выберите папку `Reels Master` из распакованного архива
|
||||||
|
|
||||||
### Разработка
|
### Разработка
|
||||||
|
|
||||||
> [!NOTE]
|
> [!NOTE]
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "reels-master",
|
"name": "reels-master",
|
||||||
"version": "1.0.0",
|
"version": "1.1.2",
|
||||||
"description": "Chrome extension for Instagram Reels with volume control and download functionality",
|
"description": "Chrome extension for Instagram Reels with volume control and download functionality",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
@ -207,3 +207,77 @@
|
|||||||
.reels-master-spinner {
|
.reels-master-spinner {
|
||||||
animation: spin 1s linear infinite;
|
animation: spin 1s linear infinite;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Seeking Control - для overlay контейнера */
|
||||||
|
.reels-master-seeking {
|
||||||
|
width: 100%;
|
||||||
|
padding: 8px 16px 12px 16px;
|
||||||
|
background: linear-gradient(to top, rgba(0, 0, 0, 0.4), transparent);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 6px;
|
||||||
|
pointer-events: auto;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reels-master-seeking-slider {
|
||||||
|
-webkit-appearance: none;
|
||||||
|
appearance: none;
|
||||||
|
width: 100%;
|
||||||
|
height: 3px;
|
||||||
|
background: rgba(255, 255, 255, 0.3);
|
||||||
|
outline: none;
|
||||||
|
border-radius: 2px;
|
||||||
|
cursor: pointer;
|
||||||
|
position: relative;
|
||||||
|
pointer-events: auto;
|
||||||
|
touch-action: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reels-master-seeking-slider::-webkit-slider-thumb {
|
||||||
|
-webkit-appearance: none;
|
||||||
|
appearance: none;
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
background: white;
|
||||||
|
cursor: pointer;
|
||||||
|
border-radius: 50%;
|
||||||
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.5);
|
||||||
|
transition: all 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reels-master-seeking-slider::-webkit-slider-thumb:hover {
|
||||||
|
background: #f0f0f0;
|
||||||
|
transform: scale(1.3);
|
||||||
|
box-shadow: 0 3px 6px rgba(0, 0, 0, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
.reels-master-seeking-slider::-moz-range-thumb {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
background: white;
|
||||||
|
cursor: pointer;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: none;
|
||||||
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.5);
|
||||||
|
transition: all 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.reels-master-seeking-slider::-moz-range-thumb:hover {
|
||||||
|
background: #f0f0f0;
|
||||||
|
transform: scale(1.3);
|
||||||
|
box-shadow: 0 3px 6px rgba(0, 0, 0, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
.reels-master-time-display {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
color: white;
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 500;
|
||||||
|
text-shadow: 0 1px 2px rgba(0, 0, 0, 0.8);
|
||||||
|
user-select: none;
|
||||||
|
opacity: 0.9;
|
||||||
|
pointer-events: none;
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
|
||||||
|
}
|
||||||
|
|||||||
@ -8,6 +8,8 @@ class ReelsMaster {
|
|||||||
private processedContainers: WeakSet<HTMLElement> = new WeakSet();
|
private processedContainers: WeakSet<HTMLElement> = new WeakSet();
|
||||||
private videoVolumeListeners: WeakMap<HTMLVideoElement, boolean> = new WeakMap();
|
private videoVolumeListeners: WeakMap<HTMLVideoElement, boolean> = new WeakMap();
|
||||||
private domObserver: MutationObserver | null = null;
|
private domObserver: MutationObserver | null = null;
|
||||||
|
private processedOverlays: WeakSet<HTMLElement> = new WeakSet();
|
||||||
|
private videoSeekingListeners: WeakMap<HTMLVideoElement, Set<HTMLInputElement>> = new WeakMap();
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.init();
|
this.init();
|
||||||
@ -52,6 +54,7 @@ class ReelsMaster {
|
|||||||
private start(): void {
|
private start(): void {
|
||||||
console.log('Reels Master: Starting...');
|
console.log('Reels Master: Starting...');
|
||||||
this.injectControlsToAllContainers();
|
this.injectControlsToAllContainers();
|
||||||
|
this.injectSeekingToAllOverlays();
|
||||||
this.setupDOMObserver();
|
this.setupDOMObserver();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -133,6 +136,7 @@ class ReelsMaster {
|
|||||||
if (shouldCheck) {
|
if (shouldCheck) {
|
||||||
requestAnimationFrame(() => {
|
requestAnimationFrame(() => {
|
||||||
this.injectControlsToAllContainers();
|
this.injectControlsToAllContainers();
|
||||||
|
this.injectSeekingToAllOverlays();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -194,6 +198,37 @@ class ReelsMaster {
|
|||||||
'svg[aria-label="Speichern"]',
|
'svg[aria-label="Speichern"]',
|
||||||
'svg[aria-label="保存"]',
|
'svg[aria-label="保存"]',
|
||||||
].join(',');
|
].join(',');
|
||||||
|
|
||||||
|
private readonly FOLLOW_TEXTS = [
|
||||||
|
'Follow',
|
||||||
|
'Подписаться',
|
||||||
|
'Seguir',
|
||||||
|
'Suivre',
|
||||||
|
'Folgen',
|
||||||
|
'Following',
|
||||||
|
'Подписки',
|
||||||
|
'Siguiendo',
|
||||||
|
'Abonné(e)',
|
||||||
|
'Gefolgt',
|
||||||
|
'Requested',
|
||||||
|
'Запрос отправлен',
|
||||||
|
'Solicitado',
|
||||||
|
'Demandé',
|
||||||
|
'Anfrage gesendet',
|
||||||
|
'フォローする',
|
||||||
|
'关注',
|
||||||
|
];
|
||||||
|
|
||||||
|
private readonly AVATAR_SELECTORS = [
|
||||||
|
'img[alt*="profile picture"]',
|
||||||
|
'img[alt*="Фото профиля"]',
|
||||||
|
'img[alt*="фото профиля"]',
|
||||||
|
'img[alt*="Foto de perfil"]',
|
||||||
|
'img[alt*="Photo de profil"]',
|
||||||
|
'img[alt*="Profilbild"]',
|
||||||
|
'img[alt*="プロフィール写真"]',
|
||||||
|
'img[alt*="头像"]',
|
||||||
|
].join(',');
|
||||||
|
|
||||||
private findAllActionContainers(): HTMLElement[] {
|
private findAllActionContainers(): HTMLElement[] {
|
||||||
const containers: HTMLElement[] = [];
|
const containers: HTMLElement[] = [];
|
||||||
@ -460,6 +495,182 @@ class ReelsMaster {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
private injectSeekingToAllOverlays(): void {
|
||||||
|
if (!window.location.pathname.includes('/reels/')) return;
|
||||||
|
|
||||||
|
const overlayContainers = this.findAllOverlayContainers();
|
||||||
|
console.log(`Reels Master: Found ${overlayContainers.length} overlay containers`);
|
||||||
|
|
||||||
|
for (const container of overlayContainers) {
|
||||||
|
this.injectSeekingToOverlay(container);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private findAllOverlayContainers(): HTMLElement[] {
|
||||||
|
const containers: HTMLElement[] = [];
|
||||||
|
|
||||||
|
const followButtons = document.querySelectorAll('[role="button"]');
|
||||||
|
|
||||||
|
for (const button of followButtons) {
|
||||||
|
const text = button.textContent?.trim();
|
||||||
|
if (text && this.FOLLOW_TEXTS.includes(text)) {
|
||||||
|
let parent = button.parentElement;
|
||||||
|
let depth = 0;
|
||||||
|
const maxDepth = 15;
|
||||||
|
|
||||||
|
while (parent && depth < maxDepth) {
|
||||||
|
const hasAvatar = parent.querySelector(this.AVATAR_SELECTORS);
|
||||||
|
const hasFollow = parent.querySelector('[role="button"]');
|
||||||
|
|
||||||
|
if (hasAvatar && hasFollow && parent.children.length >= 2) {
|
||||||
|
if (!containers.includes(parent as HTMLElement)) {
|
||||||
|
containers.push(parent as HTMLElement);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
parent = parent.parentElement;
|
||||||
|
depth++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return containers;
|
||||||
|
}
|
||||||
|
|
||||||
|
private injectSeekingToOverlay(overlayContainer: HTMLElement): void {
|
||||||
|
if (this.processedOverlays.has(overlayContainer)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (overlayContainer.querySelector('.reels-master-seeking')) {
|
||||||
|
this.processedOverlays.add(overlayContainer);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const video = this.findVideoForOverlay(overlayContainer);
|
||||||
|
if (!video) {
|
||||||
|
console.log('Reels Master: Video not found for overlay');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const seekingControl = this.createSeekingControl(video);
|
||||||
|
overlayContainer.appendChild(seekingControl);
|
||||||
|
|
||||||
|
this.processedOverlays.add(overlayContainer);
|
||||||
|
console.log('Reels Master: Seeking control injected to overlay');
|
||||||
|
}
|
||||||
|
|
||||||
|
private findVideoForOverlay(overlayContainer: HTMLElement): HTMLVideoElement | null {
|
||||||
|
let parent = overlayContainer.parentElement;
|
||||||
|
|
||||||
|
while (parent) {
|
||||||
|
const video = parent.querySelector('video');
|
||||||
|
if (video) {
|
||||||
|
return video;
|
||||||
|
}
|
||||||
|
parent = parent.parentElement;
|
||||||
|
|
||||||
|
if (parent === document.body) break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.getClosestVideoToElement(overlayContainer);
|
||||||
|
}
|
||||||
|
|
||||||
|
private createSeekingControl(video: HTMLVideoElement): HTMLDivElement {
|
||||||
|
const seekingContainer = document.createElement('div');
|
||||||
|
seekingContainer.className = 'reels-master-seeking';
|
||||||
|
|
||||||
|
seekingContainer.addEventListener('click', (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
});
|
||||||
|
seekingContainer.addEventListener('mousedown', (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
});
|
||||||
|
seekingContainer.addEventListener('touchstart', (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
});
|
||||||
|
|
||||||
|
const timeDisplay = document.createElement('div');
|
||||||
|
timeDisplay.className = 'reels-master-time-display';
|
||||||
|
|
||||||
|
const currentTimeSpan = document.createElement('span');
|
||||||
|
currentTimeSpan.textContent = '0:00';
|
||||||
|
|
||||||
|
const durationSpan = document.createElement('span');
|
||||||
|
durationSpan.textContent = '0:00';
|
||||||
|
|
||||||
|
timeDisplay.appendChild(currentTimeSpan);
|
||||||
|
timeDisplay.appendChild(durationSpan);
|
||||||
|
|
||||||
|
const slider = document.createElement('input');
|
||||||
|
slider.type = 'range';
|
||||||
|
slider.min = '0';
|
||||||
|
slider.max = '100';
|
||||||
|
slider.value = '0';
|
||||||
|
slider.className = 'reels-master-seeking-slider';
|
||||||
|
|
||||||
|
slider.addEventListener('click', (e) => e.stopPropagation());
|
||||||
|
slider.addEventListener('mousedown', (e) => e.stopPropagation());
|
||||||
|
slider.addEventListener('mouseup', (e) => e.stopPropagation());
|
||||||
|
slider.addEventListener('touchstart', (e) => e.stopPropagation());
|
||||||
|
slider.addEventListener('touchend', (e) => e.stopPropagation());
|
||||||
|
slider.addEventListener('touchmove', (e) => e.stopPropagation());
|
||||||
|
|
||||||
|
const updateDuration = () => {
|
||||||
|
if (video.duration && !isNaN(video.duration) && video.duration !== Infinity) {
|
||||||
|
slider.max = String(video.duration);
|
||||||
|
durationSpan.textContent = this.formatTime(video.duration);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const updateTime = () => {
|
||||||
|
if (!isNaN(video.duration) && video.duration !== Infinity) {
|
||||||
|
slider.value = String(video.currentTime);
|
||||||
|
currentTimeSpan.textContent = this.formatTime(video.currentTime);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
video.addEventListener('loadedmetadata', updateDuration);
|
||||||
|
video.addEventListener('durationchange', updateDuration);
|
||||||
|
video.addEventListener('timeupdate', updateTime);
|
||||||
|
|
||||||
|
updateDuration();
|
||||||
|
updateTime();
|
||||||
|
|
||||||
|
let isSeeking = false;
|
||||||
|
|
||||||
|
slider.addEventListener('input', (e) => {
|
||||||
|
const time = parseFloat((e.target as HTMLInputElement).value);
|
||||||
|
currentTimeSpan.textContent = this.formatTime(time);
|
||||||
|
isSeeking = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
slider.addEventListener('change', (e) => {
|
||||||
|
const time = parseFloat((e.target as HTMLInputElement).value);
|
||||||
|
video.currentTime = time;
|
||||||
|
isSeeking = false;
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!this.videoSeekingListeners.has(video)) {
|
||||||
|
this.videoSeekingListeners.set(video, new Set());
|
||||||
|
}
|
||||||
|
this.videoSeekingListeners.get(video)!.add(slider);
|
||||||
|
|
||||||
|
seekingContainer.appendChild(timeDisplay);
|
||||||
|
seekingContainer.appendChild(slider);
|
||||||
|
|
||||||
|
return seekingContainer;
|
||||||
|
}
|
||||||
|
|
||||||
|
private formatTime(seconds: number): string {
|
||||||
|
if (isNaN(seconds) || seconds === Infinity) {
|
||||||
|
return '0:00';
|
||||||
|
}
|
||||||
|
|
||||||
|
const mins = Math.floor(seconds / 60);
|
||||||
|
const secs = Math.floor(seconds % 60);
|
||||||
|
return `${mins}:${secs.toString().padStart(2, '0')}`;
|
||||||
|
}}
|
||||||
|
|
||||||
new ReelsMaster();
|
new ReelsMaster();
|
||||||
|
|||||||
@ -1,8 +1,8 @@
|
|||||||
{
|
{
|
||||||
"manifest_version": 3,
|
"manifest_version": 3,
|
||||||
"name": "Reels Master",
|
"name": "Reels Master",
|
||||||
"version": "1.0.0",
|
"version": "1.1.2",
|
||||||
"description": "Instagram Reels volume control and download",
|
"description": "Enhance your Instagram experience with Reels Master - download reels, seek through videos, and more!",
|
||||||
"background": {
|
"background": {
|
||||||
"service_worker": "background/background.js"
|
"service_worker": "background/background.js"
|
||||||
},
|
},
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user