// ==UserScript==
// @name Camwhores.tv Utilities Mod
// @namespace https://sleazyfork.org/users/1281730-vipprograms
// @version 1.1.1
// @description Information preview inside messages, removal of private lock from videos uploaded by friends, and more
// @author vipprograms
// @match https://www.camwhores.tv/*
// @grant GM_xmlhttpRequest
// @grant GM.getValue
// @grant GM_getValue
// @grant GM.setValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @grant GM.registerMenuCommand
// @grant window.close
// @icon 
// ==/UserScript==
(function() {
'use strict';
var _GM_registerMenuCommand, options;
const refreshMessagesTimer = 2500;
function isArrayEmpty(arr) {
return arr.length === 0;
}
function retrieveValueFromStorage(key) {
if (typeof GM_setValue === "function") {
return GM_getValue(key, false); // Default value set to false for boolean
}
else if (typeof GM === "object" && typeof GM.getValue === "function" && typeof GM.setValue === "function") {
return GM.getValue(key, false).then(function(value) {
return value;
});
}
else {
console.error("Unsupported userscript manager.");
return undefined;
}
}
function saveValue(key, array) {
GM.setValue(key, array).then(function() {
console.log("Array saved successfully.");
}).catch(function(error) {
console.error("Error saving array:", error);
});
}
if (typeof GM_registerMenuCommand !== 'undefined') {
_GM_registerMenuCommand = GM_registerMenuCommand;
} else if (typeof GM !== 'undefined' && typeof GM.registerMenuCommand !== 'undefined') {
_GM_registerMenuCommand = GM.registerMenuCommand;
} else {
_GM_registerMenuCommand = (s, f) => { debug(s); debug(f); };
}
function toggleChange(options, optionName, defaultValue = false){
var updatedOptions;
var currentState = options[optionName] !== undefined ? options[optionName] : defaultValue;
_GM_registerMenuCommand((currentState ? "Disable " : "Enable ") + optionName, () => {
currentState = !currentState;
updatedOptions[optionName] = currentState;
saveValue("options", updatedOptions);
setTimeout(() => {
location.reload();
}, 500);
});
updatedOptions = retrieveValueFromStorage("options") || {}; // Ensure options exist
if (updatedOptions[optionName] === undefined) {
updatedOptions[optionName] = defaultValue;
saveValue("options", updatedOptions);
}
}
options = retrieveValueFromStorage("options")
toggleChange(options, "auto close window after friend confirm")
var closeWindowReject = retrieveValueFromStorage("options")["auto close window after friend reject"];
toggleChange(options, "auto close window after friend reject")
var closeWindowConfirm = retrieveValueFromStorage("options")["auto close window after friend confirm"];
toggleChange(options, "alternative thumbnails", true)
var altThumbs = retrieveValueFromStorage("options")["alternative thumbnails"];
toggleChange(options, "auto refresh message page", true)
var refreshMessagePage = retrieveValueFromStorage("options")["auto refresh message page"];
// DON'T CHANGE THE TEXT STRINGS HERE! CHANGE THEM IN THE VALUES SECTION!
const auto_replies_default = ["Next time you send me a request with 0 videos, I'll block you",
"Very nice videos",
"Why? What's wrong?",
"Sorry, I don't like your videos",
"You don't have any videos"];
// DON'T CHANGE THE TEXT STRINGS HERE! CHANGE THEM IN THE VALUES SECTION!
const highlight_keywords_default = ['joi', 'cei', 'fuck', "cumshot"];
// DON'T CHANGE THE TEXT STRINGS HERE! CHANGE THEM IN THE VALUES SECTION!
var auto_replies = retrieveValueFromStorage("auto_replies");
var highlight_keywords = retrieveValueFromStorage("highlight_keywords");
if (!auto_replies) {
auto_replies = auto_replies_default;
saveValue("auto_replies", auto_replies_default);
}
if (!highlight_keywords) {
highlight_keywords = highlight_keywords_default;
saveValue("highlight_keywords", highlight_keywords_default);
}
const style = document.createElement('style');
style.textContent = `
ul > li.next,
ul > li.prev {
display: list-item !important;
}
.item:hover > a > div.img > .friends-tag,
.item:hover > a > div.img > .videos-tag{
background-color:#1a1a1a !important;
}
.button{
color: rgb(183, 176, 167);
text-align: center;
border: 1px solid transparent;
font-size: 14px;
padding: 5px 10px;
cursor: pointer;
background: linear-gradient(to bottom, #ffffff 0%, #cccccc 100%);
border-radius: 3px;
display: inline-block;
margin: 0 .3rem .3rem 0;
color: dimgrey;
}
.button:hover{
color: #f56c08;
border: 1px solid transparent;
background: #1e1e1e;
}
div.img span.unread-notification{
background: #c45606ab;
backdrop-filter: blur(5px) brightness(1);
top: 0px;
left: 0px;
border-bottom-left-radius: 3px;
border-bottom-right-radius: 3px;
outline: 3px solid #f56c08;
animation: glow 5s infinite;
}
@keyframes glow {
0% {
outline-color: #f56c08d6;
}
50% {
outline-color: transparent;
}
100% {
outline-color: #f56c08d6;
}
}
`;
document.head.appendChild(style);
if ((window.location.href.endsWith("?action=reject_add_to_friends_done") && closeWindowReject) ||
(window.location.href.endsWith("?action=confirm_add_to_friends_done") && closeWindowConfirm)) {
window.close();
}else if (window.location.href == "https://www.camwhores.tv/my/messages/") {
// console.log('(window.location.href == "https://www.camwhores.tv/my/messages/")')
if(refreshMessagePage){
$(document).ready(function() {
var linkClicked = localStorage.getItem("linkClicked") === "true";
console.log("Initial linkClicked value:", linkClicked);
$(window).on("focus", function() {
localStorage.setItem("linkClicked", "false");
linkClicked = false;
console.log("Window is focused");
if (interval !== null) {
clearInterval(interval);
}
});
if (linkClicked) {
console.log("Reloading...");
var interval = setInterval(function() {
location.reload();
}, refreshMessagesTimer);
}
$(document).on("click auxclick", "a", function(event) {
var href = $(this).attr("href");
if (event.button === 0 || event.button === 1) {
// alert("You're trying to go to " + href);
localStorage.setItem("linkClicked", "true");
linkClicked = true;
console.log("Link clicked. Will refresh");
}
});
$(window).on("blur", function() {
console.log("Window blur event");
if (linkClicked) {
console.log("Reloading...");
var interval = setInterval(function() {
location.reload();
}, refreshMessagesTimer);
}
});
});
}
function processThumbnails() {
// console.log("Processing thumbnails");
var thumbnails = document.querySelectorAll('strong.title');
if (thumbnails.length === 0) {
console.log("Thumbnails not found. Retrying in 1000 milliseconds.");
setTimeout(function() {
processThumbnails(); // Retry processing thumbnails
}, 1000);
return false; // Return false if thumbnails not found
}
thumbnails.forEach(function(thumbnail) {
var usernameText = thumbnail.textContent.trim();
if (usernameText) {
var usernameLink = document.querySelector('a[title="' + usernameText + '"]');
if (usernameLink) {
var href = usernameLink.getAttribute('href');
var userIdMatch = href.match(/\/(\d+)\/$/);
if (userIdMatch && userIdMatch.length > 1) {
var userId = userIdMatch[1];
var usernameUrl = "https://www.camwhores.tv/members/" + userId + "/";
GM_xmlhttpRequest({
method: "GET",
url: usernameUrl,
onload: function(response) {
var parser = new DOMParser();
var htmlDoc = parser.parseFromString(response.responseText, "text/html");
var infoMessage = htmlDoc.querySelector('.info-message');
var usernameLink = document.querySelector('a[title="' + usernameText + '"]');
// console.log(usernameText + ": " + usernameUrl)
// console.log(usernameLink)
if (infoMessage && infoMessage.textContent.includes("is in your friends list.")) {
if (usernameLink) {
var divElement = document.createElement('div');
divElement.classList.add("friends-tag");
divElement.textContent = "Friends ✅";
divElement.style.position = "absolute";
divElement.style.top = "0";
divElement.style.right = "0";
divElement.style.backgroundColor = "#414141"; //#1a1a1a
divElement.style.borderBottomLeftRadius = "3px";
divElement.style.color = "white";
divElement.style.padding = "3px 5px";
if (usernameLink) {
var firstDivElement = usernameLink.querySelector('div.img');
if (firstDivElement) {
firstDivElement.appendChild(divElement);
}
}
}
}
if(infoMessage && infoMessage.textContent.includes("wants to be your friend.")){
divElement = document.createElement('div');
divElement.textContent = "👀";
divElement.title = "Wants to be your friend!";
divElement.style.position = "absolute";
divElement.style.bottom = "0";
divElement.style.left = "0";
// divElement.style.backgroundColor = "aquamarine";
divElement.style.color = "white";
divElement.style.fontSize = "1rem";
divElement.style.padding = ".5rem .25rem";
if (usernameLink) {
firstDivElement = usernameLink.querySelector('div.img');
if (firstDivElement) {
firstDivElement.appendChild(divElement);
}
}
}
var titleVideosElement = htmlDoc.evaluate('//*[@id="list_videos_uploaded_videos"]/div[1]/h2', htmlDoc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
var videoNumber = 0;
if (titleVideosElement && titleVideosElement.singleNodeValue) {
var titleVideosText = titleVideosElement.singleNodeValue.textContent;
if (titleVideosText) {
var matches = titleVideosText.match(/\(([^)]+)\)/);
videoNumber = matches ? matches[1] : 0;
}
}
if(videoNumber || videoNumber === 0){
divElement = document.createElement('div');
divElement.classList.add("videos-tag");
divElement.textContent = videoNumber;
divElement.style.position = "absolute";
divElement.style.bottom = "0";
divElement.style.right = "0";
divElement.style.backgroundColor = "CadetBlue";
divElement.style.color = "white";
divElement.style.padding = "3px 5px";
divElement.style.borderTopLeftRadius = "3px";
}
if (usernameLink) {
firstDivElement = usernameLink.querySelector('div.img');
if (firstDivElement) {
firstDivElement.appendChild(divElement);
}
}
} // end httpGET reponse
});
} else {
console.log("User ID not found in href for:", usernameText);
}
} else {
console.log("Username link not found for:", usernameText);
}
}
});
return true; // Return true if thumbnails processed successfully
}
// Function to add event listener to buttons
function addButtonEventListener() {
console.log("Adding event listener to buttons");
document.addEventListener('click', function(event) {
var target = event.target;
if (target && target.matches('a[data-action="ajax"]')) {
// console.log("Button clicked");
const h2Element = document.querySelector('#list_members_my_conversations div.headline h2');
const initialText = h2Element.textContent.trim();
// console.log("Initial text: " + initialText);
const checkChangeInterval = setInterval(function() {
const h2Element = document.querySelector('#list_members_my_conversations div.headline h2');
const currentText = h2Element.textContent.trim();
// console.log("Current text: " + currentText);
if (currentText !== initialText) {
clearInterval(checkChangeInterval);
processThumbnails();
}
}, 100);
}
});
}
console.log("Calling functions to apply styles and process thumbnails");
setTimeout(function() {
processThumbnails();
var thumbnailsProcessed = processThumbnails();
if (!thumbnailsProcessed) {
console.log("Processing thumbnails failed. Retrying in 1000 milliseconds.");
setTimeout(function() {
processThumbnails(); // Retry processing thumbnails
}, 1000);
}
addButtonEventListener();
}, 1000);
setInterval(function() {
processThumbnails()
console.log("Updating status of users")
}, 3000);
const refreshIcon = document.createElement('span');
refreshIcon.innerHTML = '↻'; // Unicode for refresh icon
refreshIcon.title = 'Refresh Utilities Mod';
refreshIcon.style.cursor = 'pointer';
refreshIcon.style.marginLeft = '10px'; // Adjust margin as needed
refreshIcon.addEventListener('click', function() {
processThumbnails()
console.log('Refreshing...');
});
const h2Element = document.querySelector('#list_members_my_conversations .headline h2');
console.log("adding..?")
h2Element.appendChild(refreshIcon);
}else if(window.location.href.startsWith("https://www.camwhores.tv/my/messages/")) {
document.addEventListener('keydown', function(event) {
if (event.ctrlKey && event.key === 'Enter') {
event.preventDefault();
document.querySelector('#send_message_form input[type="submit"]').click();
}
});
// Function to fetch the URL and extract title, content, and joined date
function fetchAndExtract(url) {
GM_xmlhttpRequest({
method: "GET",
url: url,
onload: function(response) {
// Parse the response text into a HTML document
var parser = new DOMParser();
var htmlDoc = parser.parseFromString(response.responseText, "text/html");
var titleElement = htmlDoc.evaluate('//*[@id="list_videos_uploaded_videos"]/div[1]/h2', htmlDoc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
var contentElement = htmlDoc.evaluate('//*[@id="list_videos_uploaded_videos_items"]/div[1]/a/div[3]/div[1]/em', htmlDoc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
var joinedDateElement = htmlDoc.evaluate('/html/body/div[2]/div[2]/div/div[2]/div[2]/div/div/div[1]/div[3]/em', htmlDoc, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
var joinedDateText = joinedDateElement.singleNodeValue.textContent;
var confirmButton = document.querySelector('input[name="confirm"]');
var rejectButton = document.querySelector('input[name="reject"]');
if (titleElement && titleElement.singleNodeValue && contentElement && contentElement.singleNodeValue && joinedDateElement && joinedDateElement.singleNodeValue) {
var titleText = titleElement.singleNodeValue.textContent;
var extractedTitle = titleText.match(/\(([^)]+)\)/);
if (extractedTitle && extractedTitle.length > 1) {
// THERE ARE VIDEOS
confirmButton.style.backgroundImage = 'linear-gradient(rgb(0, 128, 0) 0%, rgb(0, 255, 0) 100%)';
confirmButton.style.color = '#fff';
var contentText = contentElement.singleNodeValue.textContent;
var generatedText = "<div id=gen_joined_text>Joined " + joinedDateText + "<br>" + extractedTitle[1] + " videos, last " + contentText + "</div>";
// Add generated text to the visible page
var messageDiv = document.evaluate('//*[@id="list_messages_my_conversation_messages_items"]/div/div[3]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
if (messageDiv && messageDiv.singleNodeValue) {
messageDiv.singleNodeValue.innerHTML += generatedText;
}
} else {
console.log('Text inside parentheses in title not found.');
}
} else {
// THERE ARE NO VIDEOS
console.log('Title, content, or joined date element not found. Probably no videos uploaded yet.');
rejectButton.style.backgroundImage = 'linear-gradient(rgb(255, 0, 0) 0%, rgb(128, 0, 0) 100%)';
rejectButton.style.color = '#fff';
messageDiv = document.evaluate('//*[@id="list_messages_my_conversation_messages_items"]/div/div[3]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
if (messageDiv && messageDiv.singleNodeValue) {
messageDiv.singleNodeValue.innerHTML += "<div id=gen_joined_text>Joined " + joinedDateText + "<br><b>No videos</b></div>";
}
}
}
});
}
var bottomDiv = document.querySelector("#send_message_form > div > div.bottom");
var responsesDiv = document.createElement("div");
responsesDiv.id = "responses";
responsesDiv.style.marginTop = "1rem";
bottomDiv.insertAdjacentElement('afterend', responsesDiv);
var textarea = document.querySelector("#send_message_message");
auto_replies.forEach(function(string) {
var button = document.createElement("button");
button.textContent = string;
button.classList.add("button");
button.addEventListener("click", function(event) {
event.preventDefault();
if (textarea.value.trim() !== '') {
textarea.value += "\n";
}
textarea.value += string;
document.querySelector('#send_message_form').submit();
});
button.style.padding = ".25rem .5rem"
responsesDiv.appendChild(button);
});
// Get the href attribute of the first message
var messageLink = document.evaluate('//*[@id="list_messages_my_conversation_messages_items"]/div/div[1]/a', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
// Check if the element exists
if (messageLink && messageLink.singleNodeValue) {
var messageUrl = messageLink.singleNodeValue.href;
fetchAndExtract(messageUrl);
} else {
console.log('Message link not found.');
}
// Remove class "bottom" from specified element
var bottomElement = document.evaluate('//*[@id="list_messages_my_conversation_messages_items"]/div/div[3]/form/div', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
if (bottomElement && bottomElement.singleNodeValue) {
bottomElement.singleNodeValue.style.marginTop = "1em";
bottomElement.singleNodeValue.style.marginBottom = "1em";
bottomElement.singleNodeValue.classList.remove("bottom");
} else {
console.log('Bottom element not found.');
setTimeout(function() {
var gen_joined_text = document.getElementById('gen_joined_text');
gen_joined_text.innerHTML = "<br>" + gen_joined_text.innerHTML;
// gen_joined_text.style.marginTop = "1em";
// gen_joined_text.classList.add("custom-margin-top");
}, 200); // Adjust the delay time as needed
}
}else if(window.location.href !== "https://www.camwhores.tv/my/videos/") {
// Set to store unique video elements
const processedVideos = new Set();
function addPrivateVideos() {
const private_videos = [];
const videoElements = document.querySelectorAll('.line-premium, .line-private');
videoElements.forEach(video => {
const parentDiv = video.parentElement;
if (
parentDiv.tagName === 'DIV' &&
parentDiv.parentElement.tagName === 'A' &&
!processedVideos.has(parentDiv.parentElement)
) {
private_videos.push(parentDiv.parentElement);
processedVideos.add(parentDiv.parentElement);
}
});
return private_videos;
}
async function checkVideoPage(url) {
try {
const response = await fetch(url);
// Check if the response status code indicates an error
if (!response.ok) {
console.error('Error loading video page:', response.status);
return true; // Return true if there's an error loading the page
}
const html = await response.text();
const parser = new DOMParser();
const doc = parser.parseFromString(html, 'text/html');
const hasElement1 = doc.querySelector('body > div.container > div.content > div.block-video > div.video-holder > div.player > div > div > span') !== null;
// basically the "This video is a private video uploaded by" message
const hasElement2 = doc.querySelector('body > div.container > div.content > div.block-album > div.album-holder > div.images > span.message') !== null;
return hasElement1 || hasElement2;
} catch (error) {
console.error('Error checking video page:', error);
return false;
}
}
async function applyStyleToPrivateVideos() {
const privateVideos = addPrivateVideos();
for (const video of privateVideos) {
const videoLink = video.getAttribute('href');
// console.log(videoLink);
const hasElement = await checkVideoPage(videoLink);
// console.log(videoLink + " " + hasElement);
if (!hasElement) {
video.style.outline = 'green solid 2px';
const linePrivate = video.querySelector('div > .line-private');
if (linePrivate) {
linePrivate.style.display = 'none';
}
const imgElement = video.querySelector('div > img');
if (imgElement) {
imgElement.style.opacity = '1';
}
}else{
// add ❌
const linePrivate = video.querySelector('div > .line-private > span');
if (linePrivate) {
linePrivate.textContent += ' ❌';
}
}
await new Promise(resolve => setTimeout(resolve, 25)); // avoid 503
}
}
applyStyleToPrivateVideos();
function findRelevantKeywords() {
const titles = document.querySelectorAll('.title');
titles.forEach(title => {
const originalText = title.innerHTML;
highlight_keywords.forEach(keyword => {
const regex = new RegExp(keyword, 'gi');
if (originalText.match(regex)) {
const highlightedText = originalText.replace(regex, '<span style="color: green;">$&</span>');
title.innerHTML = highlightedText;
}
});
});
}
findRelevantKeywords()
// Define a global array to store blacklisted image sources
const blacklist = [];
function moveToSecondThumbnail() {
$('.thumb.lazy-load').off('mouseover').off('mouseout');
document.querySelectorAll('.thumb.lazy-load').forEach(image => {
const src = image.getAttribute('src');
if (blacklist.includes(src)) {
return;
}
const newSrc = src.replace('1.jpg', '2.jpg');
// console.log("ye")
if (src && src.endsWith('1.jpg')) {
image.setAttribute('src', newSrc);
}
const images = ['3.jpg', '4.jpg', '5.jpg', '1.jpg', '2.jpg'];
let currentIndex = 0;
let intervalId;
const loopImages = () => {
if (src && src.endsWith('jpg')) {
const newSrcName = images[currentIndex];
const newSrc = src.substring(0, src.length - 5) + newSrcName;
console.log(newSrc)
image.setAttribute('src', newSrc);
currentIndex = (currentIndex + 1) % images.length;
}
};
const mouseoverHandler = () => {
clearInterval(intervalId);
intervalId = setInterval(loopImages, 500);
};
const mouseoutHandler = () => {
if (src && src.endsWith('1.jpg') && !(blacklist.includes(src))) {
image.setAttribute('src', newSrc);
}
clearInterval(intervalId);
};
// Create a new image element to check if the original image was loaded successfully
const testImage = new Image();
testImage.src = src;
testImage.onload = function() {
// Image loaded successfully
// console.log('Image loaded successfully:', src);
};
testImage.onerror = function() {
// Image failed to load (404 error)
// console.log('Image failed to load (404 error):', src);
blacklist.push(src);
image.setAttribute('src', src);
// Remove previous mouseover and mouseout event listeners
image.removeEventListener('mouseover', mouseoverHandler);
image.removeEventListener('mouseout', mouseoutHandler);
};
// Add mouseover and mouseout event listeners
image.addEventListener('mouseover', mouseoverHandler);
image.addEventListener('mouseout', mouseoutHandler);
});
// console.log(blacklist);
}
altThumbs && moveToSecondThumbnail();
setInterval(() => {
applyStyleToPrivateVideos();
findRelevantKeywords()
altThumbs && moveToSecondThumbnail();
}, 750); // Interval in milliseconds (every second)
if (window.location.href.startsWith("https://www.camwhores.tv/members/")) {
console.log("NOT messages, user PROFILE");
function updateH2Element() {
const h2Element = document.querySelector('#list_videos_uploaded_videos .headline h2');
if (h2Element) {
const text = h2Element.textContent;
const startIndex = text.indexOf('(');
const endIndex = text.indexOf(')', startIndex);
if (startIndex !== -1 && endIndex !== -1 && text.includes("Page")) {
const videos_total = parseInt(text.substring(startIndex + 1, endIndex));
const rounded_videos_total = Math.ceil(videos_total / 5);
const innerHTML = h2Element.innerHTML.trim();
if (!innerHTML.includes('/')) {
h2Element.innerHTML = innerHTML + '/' + rounded_videos_total;
}
}
}
}
function addButtonEventListener() {
console.log("Adding event listener to buttons");
document.addEventListener('click', function(event) {
var target = event.target;
if (target && target.matches('a[data-action="ajax"]')) {
// console.log("Button clicked");
const h2Element = document.querySelector('#list_videos_uploaded_videos .headline h2');
const initialText = h2Element.textContent.trim();
// console.log("Initial text: " + initialText);
const checkChangeInterval = setInterval(function() {
const h2Element = document.querySelector('#list_videos_uploaded_videos .headline h2');
const currentText = h2Element.textContent.trim();
// console.log("Current text: " + currentText);
if (currentText !== initialText) {
clearInterval(checkChangeInterval);
updateH2Element();
}
}, 100);
}
});
}
updateH2Element()
addButtonEventListener()
}
if(window.location.href.startsWith("https://www.camwhores.tv/videos/")){
const element1 = document.querySelector('body > div.container > div.content > div.block-video > div.video-holder > div.player > div > div > span');
const element2 = document.querySelector('body > div.container > div.content > div.block-album > div.album-holder > div.images > span.message');
if (element1 !== null || element2 !== null) {
const href = document.querySelector('#tab_video_info > div > div.block-user > div > a').href;
fetch(href)
.then(response => response.text())
.then(html => {
const htmlDoc = new DOMParser().parseFromString(html, 'text/html');
const infoMessage = htmlDoc.querySelector('.info-message');
if (infoMessage && infoMessage.textContent.includes("is in your friends list.")) {
const friends_string = "You're already friends! Reloading in 4 seconds";
(element1 || element2).textContent = friends_string;
setInterval(() => {
location.reload();
}, 4000);
} else {
console.log("Not friends!");
}
})
.catch(error => console.error('Error fetching:', error));
}
}
}
})();