initial commit

Signed-off-by: Christine Dodrill <me@christine.website>
This commit is contained in:
Cadey Ratio 2021-09-13 08:52:24 -04:00
commit bb312e83cd
6 changed files with 345 additions and 0 deletions

12
LICENSE Normal file
View File

@ -0,0 +1,12 @@
YOLO LICENSE
Version 1, July 10 2015
THIS SOFTWARE LICENSE IS PROVIDED "ALL CAPS" SO THAT YOU KNOW IT IS SUPER
SERIOUS AND YOU DON'T MESS AROUND WITH COPYRIGHT LAW BECAUSE YOU WILL GET IN
TROUBLE HERE ARE SOME OTHER BUZZWORDS COMMONLY IN THESE THINGS WARRANTIES
LIABILITY CONTRACT TORT LIABLE CLAIMS RESTRICTION MERCHANTABILITY SUBJECT TO
THE FOLLOWING CONDITIONS:
1. #yolo
2. #swag
3. #blazeit

91
convenience.js Normal file
View File

@ -0,0 +1,91 @@
/*
Copyright (c) 2011-2012, Giovanni Campagna <scampa.giovanni@gmail.com>
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
* Neither the name of the GNOME nor the
names of its contributors may be used to endorse or promote products
derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
const Gettext = imports.gettext;
const Gio = imports.gi.Gio;
const Config = imports.misc.config;
const ExtensionUtils = imports.misc.extensionUtils;
/**
* initTranslations:
* @domain: (optional): the gettext domain to use
*
* Initialize Gettext to load translations from extensionsdir/locale.
* If @domain is not provided, it will be taken from metadata['gettext-domain']
*/
function initTranslations(domain) {
let extension = ExtensionUtils.getCurrentExtension();
domain = domain || extension.metadata['gettext-domain'];
// check if this extension was built with "make zip-file", and thus
// has the locale files in a subfolder
// otherwise assume that extension has been installed in the
// same prefix as gnome-shell
let localeDir = extension.dir.get_child('locale');
if (localeDir.query_exists(null))
Gettext.bindtextdomain(domain, localeDir.get_path());
else
Gettext.bindtextdomain(domain, Config.LOCALEDIR);
}
/**
* getSettings:
* @schema: (optional): the GSettings schema id
*
* Builds and return a GSettings schema for @schema, using schema files
* in extensionsdir/schemas. If @schema is not provided, it is taken from
* metadata['settings-schema'].
*/
function getSettings(schema) {
let extension = ExtensionUtils.getCurrentExtension();
schema = schema || extension.metadata['settings-schema'];
const GioSSS = Gio.SettingsSchemaSource;
// check if this extension was built with "make zip-file", and thus
// has the schema files in a subfolder
// otherwise assume that extension has been installed in the
// same prefix as gnome-shell (and therefore schemas are available
// in the standard folders)
let schemaDir = extension.dir.get_child('schemas');
let schemaSource;
if (schemaDir.query_exists(null))
schemaSource = GioSSS.new_from_directory(schemaDir.get_path(),
GioSSS.get_default(),
false);
else
schemaSource = GioSSS.get_default();
let schemaObj = schemaSource.lookup(schema, true);
if (!schemaObj)
throw new Error('Schema ' + schema + ' could not be found for extension '
+ extension.metadata.uuid + '. Please check your installation.');
return new Gio.Settings({settings_schema: schemaObj});
}

49
extension.js Normal file
View File

@ -0,0 +1,49 @@
const {St, Clutter, Soup} = imports.gi;
const Main = imports.ui.main;
const Self = imports.misc.extensionUtils.getCurrentExtension();
const Timer = Self.imports.timer;
let panelButton;
function getFront() {
let soupSyncSession = new Soup.SessionSync();
let message = Soup.Message.new('GET', "https://home.cetacean.club/front");
let responseCode = soupSyncSession.send_message(message);
let res;
if(responseCode == 200) {
return message['response-body'].data;
}
}
function updateButton() {
let panelButtonText = new St.Label({
text : getFront(),
y_align: Clutter.ActorAlign.CENTER,
});
panelButton.set_child(panelButtonText);
}
function init() {
panelButton = new St.Bin({
style_class : "panel-button",
});
}
function enable() {
updateButton();
Main.panel._rightBox.insert_child_at_index(panelButton, 0);
let timer = new Timer.AFTimer();
timer.registerCallback(updateButton);
timer.setMinutes(5);
timer.start();
}
function disable() {
Main.panel._rightBox.remove_child(panelButton);
let timer = new Timer.AFTimer();
timer.cleanup();
}

10
metadata.json Normal file
View File

@ -0,0 +1,10 @@
{
"name" : "Who Is Front?",
"description" : "Shows who is front in Within",
"shell-version" : [
"40"
],
"url" : "",
"uuid" : "whoisfront@within.website",
"version" : 0.1
}

38
settings.js Normal file
View File

@ -0,0 +1,38 @@
const Gio = imports.gi.Gio;
const Self = imports.misc.extensionUtils.getCurrentExtension();
const Convenience = Self.imports.convenience;
var Settings = class {
/**
* Settings object.
*
* @param [schema]
* @private
*/
constructor(schema) {
this._settings = Convenience.getSettings(schema);
}
observe(key, callback) {
return this._settings.connect('changed::' + key, callback);
}
disconnect(handler) {
return this._settings.disconnect(handler);
}
set(key, type, value) {
if (this._settings['set_' + type](key, value)) {
Gio.Settings.sync(); // wait for write
} else {
throw "Could not set " + key + " (type: " + type + ") with the value " + value;
}
}
get(key, type) {
return this._settings['get_' + type](key);
}
};

145
timer.js Normal file
View File

@ -0,0 +1,145 @@
const Lang = imports.lang;
const GLib = imports.gi.GLib;
const Self = imports.misc.extensionUtils.getCurrentExtension();
const Prefs = Self.imports.settings;
let _afTimerInstance = null;
// Singleton implementation of _AFTimer
var AFTimer = function () {
if (!_afTimerInstance) {
_afTimerInstance = new _AFTimer();
}
return _afTimerInstance;
};
/**
* Timer for the auto fetch feature.
*
* @type {Lang}
*/
var _AFTimer = class {
constructor() {
this._timeout = null;
this._timoutEndCallback = null;
this._minutes = 30;
this._settings = new Prefs.Settings();
}
isActive() {
return this._settings.get('auto-fetch', 'boolean');
}
remainingMinutes() {
let minutesElapsed = this._minutesElapsed();
let remainder = minutesElapsed % this._minutes;
return Math.max(this._minutes - remainder, 0);
}
registerCallback(callback) {
this._timoutEndCallback = callback;
}
/**
* Sets the minutes of the timer.
*
* @param minutes
*/
setMinutes(minutes) {
this._minutes = minutes;
}
/**
* Start the timer.
*
* @return void
*/
start() {
this.cleanup();
let last = this._settings.get('timer-last-trigger', 'int64');
if (last === 0) {
this.reset();
}
let millisRemaining = this.remainingMinutes() * 60 * 1000;
// set new wallpaper if the interval was surpassed and set the timestamp to when it should have been updated
if (this._surpassedInterval()) {
if (this._timoutEndCallback) {
this._timoutEndCallback();
}
let millisOverdue = (this._minutes * 60 * 1000) - millisRemaining;
this._settings.set('timer-last-trigger', 'int64', Date.now() - millisOverdue);
}
// actual timer function
this._timeout = GLib.timeout_add(GLib.PRIORITY_DEFAULT, millisRemaining, () => {
if (this._timoutEndCallback) {
this._timoutEndCallback();
}
this.reset(); // reset timer
this.start(); // restart timer
});
}
/**
* Stop the timer.
*
* @return void
*/
stop() {
this._settings.set('timer-last-trigger', 'int64', 0);
this.cleanup();
}
/**
* Cleanup the timeout callback if it exists.
*
* @return void
*/
cleanup() {
if (this._timeout) { // only remove if a timeout is active
GLib.source_remove(this._timeout);
this._timeout = null;
}
}
/**
* Reset the timer.
*
* @return void
*/
reset() {
this._settings.set('timer-last-trigger', 'int64', new Date().getTime());
this.cleanup();
}
_minutesElapsed() {
let now = Date.now();
let last = this._settings.get('timer-last-trigger', 'int64');
if (last === 0) {
return 0;
}
let elapsed = Math.max(now - last, 0);
return Math.floor(elapsed / (60 * 1000));
}
_surpassedInterval() {
let now = Date.now();
let last = this._settings.get('timer-last-trigger', 'int64');
let diff = now - last;
let intervalLength = this._minutes * 60 * 1000;
return diff > intervalLength;
}
};