Merge branch 'release/2.1.0'

This commit is contained in:
Aaron Yarborough 2020-04-13 13:58:24 +01:00
commit ddaa902df5
13 changed files with 1493 additions and 114 deletions

30
src/.eslintrc.json Normal file
View file

@ -0,0 +1,30 @@
{
"env": {
"browser": true,
"es6": true
},
"extends": [
"standard",
"plugin:@typescript-eslint/eslint-recommended",
"plugin:@typescript-eslint/recommended"
],
"globals": {
"Atomics": "readonly",
"SharedArrayBuffer": "readonly"
},
"parser": "@typescript-eslint/parser",
"parserOptions": {
"ecmaVersion": 2018,
"sourceType": "module"
},
"plugins": [
"@typescript-eslint"
],
"rules": {
"no-tabs": "off",
"space-before-function-paren": "off",
"semi": "off",
"eol-last": "off",
"indent": "off"
}
}

View file

@ -1,53 +1,25 @@
import { TwitterHider } from "./hider"; import { TwitterHider } from './hider';
import { Preferences } from "./interfaces/preferences"; import { Preferences } from './interfaces/preferences';
import { PreferencesRepository } from './preferences-repository';
class Background { class Background {
static async init() { static async init(): Promise<void> {
if (await this.checkIfFirstStart()) { if (await this.checkIfFirstStart()) {
await this.setup(); await this.setup();
} }
this.listen(); const preferences: Preferences = await PreferencesRepository.get();
}
static async listen() {
const preferences: Preferences = await this.getSavedPreferences();
const hider = new TwitterHider(preferences); const hider = new TwitterHider(preferences);
hider.init(); hider.init();
} }
static setup() { private static async setup(): Promise<void> {
const defaults = { await PreferencesRepository.set(PreferencesRepository.DefaultPreferences);
hideLikes: true,
hideRetweets: true,
hideReplies: true,
setup: true
};
return new Promise(resolve => {
globalThis.chrome.storage.local.set(defaults, () => resolve());
});
} }
static checkIfFirstStart(): Promise<boolean> { private static async checkIfFirstStart(): Promise<boolean> {
return new Promise(resolve => { return !(await PreferencesRepository.get());
globalThis.chrome.storage.local.get('setup', data => {
resolve(!data.setup);
});
});
}
static getSavedPreferences(): Promise<Preferences> {
return new Promise(resolve => {
globalThis.chrome.storage.local.get(['hideLikes', 'hideReplies', 'hideRetweets'], (data: Preferences) => {
resolve(data);
});
})
} }
} }
Background.init(); Background.init();

View file

@ -27,17 +27,24 @@ body[data-hidelikes]
body[data-hideretweets] body[data-hideretweets]
div.css-1dbjc4n.r-1ila09b.r-qklmqi.r-1adg3ll div.css-1dbjc4n.r-1ila09b.r-qklmqi.r-1adg3ll
div.css-1dbjc4n.r-1kfrmmb.r-1efd50x.r-5kkj8d.r-18u37iz.r-9qu9m4 div.css-1dbjc4n.r-1kfrmmb.r-1efd50x.r-5kkj8d.r-18u37iz.r-9qu9m4
a[href$="retweets"] a[href$="retweets"] {
span.css-901oao.css-16my406.r-1qd0xha.r-vw2c0b.r-ad9z0x.r-bcqeeo.r-d3hbe1.r-1wgg2b2.r-axxi2z.r-qvutc0
span.css-901oao.css-16my406.r-1qd0xha.r-ad9z0x.r-bcqeeo.r-qvutc0 {
display: none; display: none;
} }
body[data-hideretweets] body[data-hidelikes]
div.css-1dbjc4n.r-1ila09b.r-qklmqi.r-1adg3ll div.css-1dbjc4n.r-1ila09b.r-qklmqi.r-1adg3ll
div.css-1dbjc4n.r-1kfrmmb.r-1efd50x.r-5kkj8d.r-18u37iz.r-9qu9m4 div.css-1dbjc4n.r-1kfrmmb.r-1efd50x.r-5kkj8d.r-18u37iz.r-9qu9m4
a[href$="likes"] a[href$="likes"] {
span.css-901oao.css-16my406.r-1qd0xha.r-vw2c0b.r-ad9z0x.r-bcqeeo.r-d3hbe1.r-1wgg2b2.r-axxi2z.r-qvutc0 display: none;
}
/*
Profile (example: https://twitter.com/DeborahMeaden/status/1249679362863124480)
*/
body[data-hidefollowers]
div.css-1dbjc4n.r-18u37iz
a[href$="followers"]
span.css-901oao.css-16my406.r-1qd0xha.r-ad9z0x.r-bcqeeo.r-qvutc0 { span.css-901oao.css-16my406.r-1qd0xha.r-ad9z0x.r-bcqeeo.r-qvutc0 {
display: none; display: none;
} }

View file

@ -1,31 +1,41 @@
import { Preferences } from "./interfaces/preferences"; import { Preferences } from './interfaces/preferences';
export class TwitterHider { export class TwitterHider {
private preferences: Preferences; private preferences: Preferences;
private readonly hideLikesToggleAttr = 'data-hidelikes';
private readonly hideRetweetsToggleAttr = 'data-hideretweets';
private readonly HideRepliesToggleAttr = 'data-hidereplies';
private readonly HideFollowersToggleAttr = 'data-hidefollowers';
constructor(preferences: Preferences) { constructor(preferences: Preferences) {
this.preferences = preferences; this.preferences = preferences;
} }
public init() { public init(): void {
this.setLikesVisibility(this.preferences.hideLikes); this.setLikesVisibility(this.preferences.hideLikes);
this.setRetweetsVisibility(this.preferences.hideRetweets); this.setRetweetsVisibility(this.preferences.hideRetweets);
this.setRepliesVisibility(this.preferences.hideReplies); this.setRepliesVisibility(this.preferences.hideReplies);
this.setFollowersVisibility(this.preferences.hideFollowers);
} }
private setLikesVisibility(visiblity: boolean) { private setLikesVisibility(visiblity: boolean): void {
this.setAttributeOrRemoveIfNull(document.body, "data-hidelikes", this.trueStringOrNull(visiblity)); this.setAttributeOrRemoveIfNull(document.body, this.hideLikesToggleAttr, this.trueStringOrNull(visiblity));
} }
private setRetweetsVisibility(visiblity: boolean) { private setRetweetsVisibility(visiblity: boolean): void {
this.setAttributeOrRemoveIfNull(document.body, "data-hideretweets", this.trueStringOrNull(visiblity)); this.setAttributeOrRemoveIfNull(document.body, this.hideRetweetsToggleAttr, this.trueStringOrNull(visiblity));
} }
private setRepliesVisibility(visiblity: boolean) { private setRepliesVisibility(visiblity: boolean): void {
this.setAttributeOrRemoveIfNull(document.body, "data-hidereplies", this.trueStringOrNull(visiblity)); this.setAttributeOrRemoveIfNull(document.body, this.HideRepliesToggleAttr, this.trueStringOrNull(visiblity));
} }
private setAttributeOrRemoveIfNull(element: HTMLElement, attributeName: string, value: string) { private setFollowersVisibility(visibility: boolean): void {
this.setAttributeOrRemoveIfNull(document.body, this.HideFollowersToggleAttr, this.trueStringOrNull(visibility));
}
private setAttributeOrRemoveIfNull(element: HTMLElement, attributeName: string, value: string): void {
if (!value) { if (!value) {
element.removeAttribute(attributeName); element.removeAttribute(attributeName);
return; return;
@ -34,7 +44,7 @@ export class TwitterHider {
element.setAttribute(attributeName, value); element.setAttribute(attributeName, value);
} }
private trueStringOrNull(value: boolean) { private trueStringOrNull(value: boolean): string {
return value ? "true" : null; return value ? 'true' : null;
} }
} }

View file

@ -2,4 +2,5 @@ export interface Preferences {
hideLikes: boolean; hideLikes: boolean;
hideRetweets: boolean; hideRetweets: boolean;
hideReplies: boolean; hideReplies: boolean;
hideFollowers: boolean;
} }

View file

@ -1,7 +1,7 @@
{ {
"name": "Cuckoo for Twitter", "name": "Cuckoo for Twitter",
"description": "Hide Twitter like, retweet and reply counts.", "description": "Hide Twitter like, retweet and reply counts.",
"version": "2.0.0", "version": "2.1.0",
"manifest_version": 2, "manifest_version": 2,
"content_scripts": [ "content_scripts": [
{ {

1306
src/package-lock.json generated

File diff suppressed because it is too large Load diff

View file

@ -1,8 +1,16 @@
{ {
"devDependencies": { "devDependencies": {
"@typescript-eslint/eslint-plugin": "^2.27.0",
"@typescript-eslint/parser": "^2.27.0",
"cpx": "1.5.0",
"cssnano": "^4.1.10", "cssnano": "^4.1.10",
"typescript": "^3.8.3", "eslint": "^6.8.0",
"cpx": "1.5.0" "eslint-config-standard": "^14.1.1",
"eslint-plugin-import": "^2.20.2",
"eslint-plugin-node": "^11.1.0",
"eslint-plugin-promise": "^4.2.1",
"eslint-plugin-standard": "^4.0.1",
"typescript": "^3.8.3"
}, },
"scripts": { "scripts": {
"build": "cpx manifest.json ../build/ && parcel build content.ts hider.ts popup.html hider.css --no-source-maps -d ../build/" "build": "cpx manifest.json ../build/ && parcel build content.ts hider.ts popup.html hider.css --no-source-maps -d ../build/"

View file

@ -1,6 +1,6 @@
h1 { h1 {
font-size: 18px; font-size: 18px;
color: #AAB8C2; color: #fff;
margin-bottom: 15px; margin-bottom: 15px;
margin: 0; margin: 0;
padding: 0; padding: 0;
@ -34,6 +34,17 @@ label, #savedChanges span {
border: none; border: none;
color: #fff; color: #fff;
padding: 10px 5px; padding: 10px 5px;
cursor: pointer;
}
#savedChanges {
padding: 10px;
border-radius: 5px;
background-color: #599b47;
}
#savedChanges span {
color: #fff;
} }
#banner { #banner {

View file

@ -13,22 +13,27 @@
<div class="container"> <div class="container">
<div> <div>
<label for="hidelikes">Hide likes?</label>
<input type="checkbox" name="hidelikes" id="hideLikes"> <input type="checkbox" name="hidelikes" id="hideLikes">
<label for="hidelikes">Hide likes?</label>
</div> </div>
<div class="row"> <div class="row">
<label for="hidereplies">Hide replies?</label>
<input type="checkbox" name="hidereplies" id="hideReplies"> <input type="checkbox" name="hidereplies" id="hideReplies">
<label for="hidereplies">Hide replies?</label>
</div> </div>
<div class="row"> <div class="row">
<label for="hideretweets">Hide re-tweets?</label>
<input type="checkbox" name="hideretweets" id="hideRetweets"> <input type="checkbox" name="hideretweets" id="hideRetweets">
<label for="hideretweets">Hide re-tweets?</label>
</div>
<div class="row">
<input type="checkbox" name="hidefollowers" id="hideFollowers">
<label for="hideretweets">Hide follower counts?</label>
</div> </div>
<div class="row" id="savedChanges" style="display: none"> <div class="row" id="savedChanges" style="display: none">
<span style="color: green">Your changes have been saved. You must refresh your page before they take effect.</span> <span>Your changes have been saved. You must refresh your page before they take effect.</span>
</div> </div>
<div class="row"> <div class="row">
@ -36,6 +41,6 @@
</div> </div>
</div> </div>
</div> </div>
<script src="popup.js"></script> <script src="popup.ts"></script>
</body> </body>
</html> </html>

View file

@ -1,50 +0,0 @@
class Popup {
static async init() {
this.bind();
const preferences = await this.getPreferences();
this.hideLikesCheckbox.checked = preferences.hideLikes;
this.hideRepliesCheckbox.checked = preferences.hideReplies;
this.hideRetweetsCheckbox.checked = preferences.hideRetweets;
}
static bind() {
this.hideLikesCheckbox = document.getElementById("hideLikes");
this.hideRepliesCheckbox = document.getElementById("hideReplies");
this.hideRetweetsCheckbox = document.getElementById("hideRetweets");
this.saveButton = document.getElementById("saveButton");
this.saveChangesMessage = document.getElementById("savedChanges");
saveButton.addEventListener("click", async () => {
await this.saveChanges();
this.saveChangesMessage.style.display = "block";
});
}
static saveChanges() {
return new Promise(resolve => {
chrome.storage.local.set({
hideLikes: this.hideLikesCheckbox.checked,
hideRetweets: this.hideRetweetsCheckbox.checked,
hideReplies: this.hideRepliesCheckbox.checked
}, () => resolve());
});
}
static getPreferences() {
return new Promise(resolve => {
chrome.storage.local.get([
'hideLikes',
'hideRetweets',
'hideReplies'
], data => {
resolve(data);
});
});
}
}
window.onload = () => {
Popup.init();
};

56
src/popup.ts Normal file
View file

@ -0,0 +1,56 @@
import { PreferencesRepository } from './preferences-repository';
import { Preferences } from './interfaces/preferences';
class Popup {
private static hideLikesCheckbox: HTMLInputElement;
private static hideRepliesCheckbox: HTMLInputElement;
private static hideRetweetsCheckbox: HTMLInputElement;
private static hideFollowersCheckbox: HTMLInputElement;
private static saveButton: HTMLButtonElement;
private static saveChangesMessage: HTMLElement;
static async init(): Promise<void> {
const preferences: Preferences = await PreferencesRepository.get();
this.bind();
this.hideLikesCheckbox.checked = preferences.hideLikes;
this.hideRepliesCheckbox.checked = preferences.hideReplies;
this.hideRetweetsCheckbox.checked = preferences.hideRetweets;
this.hideFollowersCheckbox.checked = preferences.hideFollowers;
}
static bind(): void {
this.hideLikesCheckbox = document.getElementById('hideLikes') as HTMLInputElement;
this.hideRepliesCheckbox = document.getElementById('hideReplies') as HTMLInputElement;
this.hideRetweetsCheckbox = document.getElementById('hideRetweets') as HTMLInputElement;
this.hideFollowersCheckbox = document.getElementById('hideFollowers') as HTMLInputElement;
this.saveChangesMessage = document.getElementById('savedChanges');
this.saveButton = document.getElementById('saveButton') as HTMLButtonElement;
this.saveButton.addEventListener('click', async () => {
await this.saveChanges();
this.saveChangesMessage.style.display = 'block';
});
}
static async saveChanges(): Promise<void> {
return new Promise(resolve => {
const updatedPreferences: Preferences = {
hideLikes: this.hideLikesCheckbox.checked,
hideRetweets: this.hideRetweetsCheckbox.checked,
hideReplies: this.hideRepliesCheckbox.checked,
hideFollowers: this.hideFollowersCheckbox.checked
};
PreferencesRepository.set(updatedPreferences);
resolve();
});
}
}
window.onload = (): void => {
Popup.init();
};

View file

@ -0,0 +1,23 @@
import { Preferences } from './interfaces/preferences';
export class PreferencesRepository {
static readonly DefaultPreferences: Preferences = {
hideLikes: true,
hideRetweets: true,
hideReplies: true,
hideFollowers: true
};
static get(): Promise<Preferences> {
const propertyNames: string[] = Object.keys(this.DefaultPreferences);
return new Promise(resolve => {
globalThis.chrome.storage.local.get(propertyNames, (data: Preferences) => resolve(data))
});
}
static set(preferences: Preferences): Promise<void> {
return new Promise(resolve => {
globalThis.chrome.storage.local.set(preferences, () => resolve());
});
}
}