* migrate to astal * Reorganize project structure. * progress * Migrate Dashboard and Window Title modules. * Migrate clock and notification bar modules. * Remove unused code * Media menu * Rework network and volume modules * Finish custom modules. * Migrate battery bar module. * Update battery module and organize helpers. * Migrate workspace module. * Wrap up bar modules. * Checkpoint before I inevitbly blow something up. * Updates * Fix event propagation logic. * Type fixes * More type fixes * Fix padding for event boxes. * Migrate volume menu and refactor scroll event handlers. * network module WIP * Migrate network service. * Migrate bluetooth menu * Updates * Migrate notifications * Update scrolling behavior for custom modules. * Improve popup notifications and add timer functionality. * Migration notifications menu header/controls. * Migrate notifications menu and consolidate notifications menu code. * Migrate power menu. * Dashboard progress * Migrate dashboard * Migrate media menu. * Reduce media menu nesting. * Finish updating media menu bindings to navigate active player. * Migrate battery menu * Consolidate code * Migrate calendar menu * Fix workspace logic to update on client add/change/remove and consolidate code. * Migrate osd * Consolidate hyprland service connections. * Implement startup dropdown menu position allocation. * Migrate settings menu (WIP) * Settings dialo menu fixes * Finish Dashboard menu * Type updates * update submoldule for types * update github ci * ci * Submodule update * Ci updates * Remove type checking for now. * ci fix * Fix a bunch of stuff, losing track... need rest. Brb coffee * Validate dropdown menu before render. * Consolidate code and add auto-hide functionality. * Improve auto-hide behavior. * Consolidate audio menu code * Organize bluetooth code * Improve active player logic * Properly dismiss a notification on action button resolution. * Implement CLI command engine and migrate CLI commands. * Handle variable disposal * Bar component fixes and add hyprland startup rules. * Handle potentially null bindings network and bluetooth bindings. * Handle potentially null wired adapter. * Fix GPU stats * Handle poller for GPU * Fix gpu bar logic. * Clean up logic for stat bars. * Handle wifi and wired bar icon bindings. * Fix battery percentages * Fix switch behavior * Wifi staging fixes * Reduce redundant hyprland service calls. * Code cleanup * Document the option code and reduce redundant calls to optimize performance. * Remove outdated comment. * Add JSDocs * Add meson to build hyprpanel * Consistency updates * Organize commands * Fix images not showing up on notifications. * Remove todo * Move hyprpanel configuration to the ~/.config/hyprpanel directory and add utility commands. * Handle SRC directory for the bundled/built hyprpanel. * Add namespaces to all windows * Migrate systray * systray updates * Update meson to include ts, tsx and scss files. * Remove log from meson * Fix file choose path and make it float. * Added a command to check the dependency status * Update dep names. * Get scale directly from env * Add todo
89 lines
2.7 KiB
TypeScript
89 lines
2.7 KiB
TypeScript
// TODO: Convert to a real service
|
|
|
|
import { bind, GLib, Variable } from 'astal';
|
|
import { FunctionPoller } from 'src/lib/poller/FunctionPoller';
|
|
import { GenericResourceData } from 'src/lib/types/customModules/generic';
|
|
|
|
class Ram {
|
|
private updateFrequency = Variable(2000);
|
|
private shouldRound = false;
|
|
private ramPoller: FunctionPoller<GenericResourceData, []>;
|
|
|
|
public ram = Variable<GenericResourceData>({ total: 0, used: 0, percentage: 0, free: 0 });
|
|
|
|
constructor() {
|
|
this.calculateUsage = this.calculateUsage.bind(this);
|
|
this.ramPoller = new FunctionPoller<GenericResourceData, []>(
|
|
this.ram,
|
|
[],
|
|
bind(this.updateFrequency),
|
|
this.calculateUsage,
|
|
);
|
|
|
|
this.ramPoller.initialize('ram');
|
|
}
|
|
|
|
public calculateUsage(): GenericResourceData {
|
|
try {
|
|
const [success, meminfoBytes] = GLib.file_get_contents('/proc/meminfo');
|
|
|
|
if (!success || !meminfoBytes) {
|
|
throw new Error('Failed to read /proc/meminfo or file content is null.');
|
|
}
|
|
|
|
const meminfo = new TextDecoder('utf-8').decode(meminfoBytes);
|
|
|
|
const totalMatch = meminfo.match(/MemTotal:\s+(\d+)/);
|
|
const availableMatch = meminfo.match(/MemAvailable:\s+(\d+)/);
|
|
|
|
if (!totalMatch || !availableMatch) {
|
|
throw new Error('Failed to parse /proc/meminfo for memory values.');
|
|
}
|
|
|
|
const totalRamInBytes = parseInt(totalMatch[1], 10) * 1024;
|
|
const availableRamInBytes = parseInt(availableMatch[1], 10) * 1024;
|
|
|
|
let usedRam = totalRamInBytes - availableRamInBytes;
|
|
usedRam = isNaN(usedRam) || usedRam < 0 ? 0 : usedRam;
|
|
|
|
return {
|
|
percentage: this.divide([totalRamInBytes, usedRam]),
|
|
total: totalRamInBytes,
|
|
used: usedRam,
|
|
free: availableRamInBytes,
|
|
};
|
|
} catch (error) {
|
|
console.error('Error calculating RAM usage:', error);
|
|
return { total: 0, used: 0, percentage: 0, free: 0 };
|
|
}
|
|
}
|
|
|
|
public setShouldRound(round: boolean): void {
|
|
this.shouldRound = round;
|
|
}
|
|
|
|
private divide([total, used]: number[]): number {
|
|
const percentageTotal = (used / total) * 100;
|
|
|
|
if (this.shouldRound) {
|
|
return total > 0 ? Math.round(percentageTotal) : 0;
|
|
}
|
|
|
|
return total > 0 ? parseFloat(percentageTotal.toFixed(2)) : 0;
|
|
}
|
|
|
|
updateTimer(timerInMs: number): void {
|
|
this.updateFrequency.set(timerInMs);
|
|
}
|
|
|
|
public stopPoller(): void {
|
|
this.ramPoller.stop();
|
|
}
|
|
|
|
public startPoller(): void {
|
|
this.ramPoller.start();
|
|
}
|
|
}
|
|
|
|
export default Ram;
|