initial commit

This commit is contained in:
2026-02-08 11:04:35 +05:30
parent a1287b696a
commit e1be0429bd
24 changed files with 1041 additions and 2 deletions

38
lib/main.dart Normal file
View File

@@ -0,0 +1,38 @@
import 'package:flutter/material.dart';
import "package:prod/views/editors.dart";
import "package:prod/views/home.dart";
import "package:yaru/yaru.dart";
import "package:provider/provider.dart";
import "package:prod/models/globalModel.dart";
void main() {
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
@override
Widget build(context) {
return ChangeNotifierProvider<GlobalModel>(
lazy: false,
create: (context) => GlobalModel(),
builder: (context, child) {
return YaruTheme(
builder: (context, yaru, child) {
return MaterialApp(
theme: yaru.theme,
themeMode: .system,
darkTheme: yaru.darkTheme,
routes: {
"/": (context) => HomePage(),
"/editors": (context) => EditorEditor(),
},
initialRoute: "/",
);
},
);
},
);
}
}

20
lib/models/editor.dart Normal file
View File

@@ -0,0 +1,20 @@
import "package:flutter/material.dart";
import "package:process_run/which.dart";
class Editor {
final String name;
final String command;
final String commandTemplate;
// final Icon icon;
const Editor(this.name, this.command, this.commandTemplate);
bool validateCommand() {
final String? fullPath = whichSync(command);
if (fullPath == null) {
return false;
} else {
return true;
}
}
}

View File

@@ -0,0 +1,68 @@
import 'package:flutter/foundation.dart';
import 'package:prod/models/editor.dart';
import 'package:prod/models/project.dart';
class GlobalModel extends ChangeNotifier {
late List<Project> projects;
late List<bool> hoverShow;
late List<Editor> editors;
GlobalModel() {
projects = [];
editors = [];
hoverShow = List.filled(projects.length, false, growable: true);
}
void addPrj(Project prj) {
projects.add(prj);
hoverShow.add(false);
notifyListeners();
}
void delPrj(int index) {
projects.removeAt(index);
hoverShow.removeAt(index);
notifyListeners();
}
List<Project> get lsPrj {
return projects;
}
int get lenPrj {
return projects.length;
}
Project nthPrj(int index) {
return projects[index];
}
void setHoverShow(int index, bool state) {
hoverShow[index] = state;
notifyListeners();
}
bool getHoverShow(int index) {
return hoverShow[index];
}
// Editor List Management.
void addEdt(Editor edt) {
editors.add(edt);
notifyListeners();
}
void delEdt(int index) {
editors.removeAt(index);
notifyListeners();
}
int get lenEdt {
return editors.length;
}
Editor nthEdt(int index) {
return editors[index];
}
}

44
lib/models/project.dart Normal file
View File

@@ -0,0 +1,44 @@
import 'package:prod/models/editor.dart';
import "dart:io";
import "package:path/path.dart" as p;
class Project {
final String name;
final String language;
final File path;
final List<Editor> editors;
final bool isGit;
final bool enableTerminal;
Project(
this.name,
this.language,
this.path,
this.editors,
this.isGit,
this.enableTerminal,
);
factory Project.validated(
String name,
String lang,
String path,
List<Editor> editors,
bool enableTerminal,
) {
final File fpath = File(path);
if (fpath.existsSync()) {
print("Project not found!!!");
} else {
print("Project Exists on disk");
}
String gitPath = p.join(path, ".git", "HEAD");
final bool isGit = File(gitPath).existsSync();
return Project(name, lang, fpath, editors, isGit, enableTerminal);
}
// bool validatePath(){
// return File
// }
}

38
lib/views/editors.dart Normal file
View File

@@ -0,0 +1,38 @@
import "package:flutter/material.dart";
import "package:prod/models/editor.dart";
import "package:prod/models/globalModel.dart";
import "package:prod/widgets/editorCard.dart";
import "package:provider/provider.dart";
class EditorEditor extends StatelessWidget {
const EditorEditor({super.key});
@override
Widget build(BuildContext context) {
GlobalModel gm = Provider.of(context);
return Scaffold(
appBar: AppBar(title: Text("PROject Dashboard")),
body: gm.lenEdt > 0
? LayoutBuilder(
builder: (context, constraints) {
int cols = (constraints.maxWidth / 200).floor();
return GridView.builder(
gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
crossAxisCount: cols,
crossAxisSpacing: 10,
mainAxisSpacing: 10,
),
itemCount: gm.lenEdt,
itemBuilder: (context, index) => EditorCard(index),
);
},
)
: Center(
child: Text(
"Add editors by pressing the \"+\" button.",
style: TextStyle(fontSize: 20),
),
),
);
}
}

48
lib/views/home.dart Normal file
View File

@@ -0,0 +1,48 @@
import "package:prod/models/globalModel.dart";
import "package:prod/models/project.dart";
import "package:prod/widgets/prjFAB.dart";
import "package:prod/widgets/drawer.dart";
import "package:prod/widgets/projectCard.dart";
import "package:provider/provider.dart";
import "package:yaru/yaru.dart";
import "package:flutter/material.dart";
class HomePage extends StatelessWidget {
const HomePage({super.key});
@override
Widget build(BuildContext context) {
GlobalModel gm = Provider.of<GlobalModel>(context);
return SafeArea(
child: Scaffold(
appBar: AppBar(title: Text("PROject Dashboard")),
drawer: ProdDrawer(),
floatingActionButton: ProjFAB(),
body: Padding(
padding: const EdgeInsets.all(8.0),
child: gm.lenPrj > 0
? LayoutBuilder(
builder: (context, constraints) {
int cols = (constraints.maxWidth / 200).floor();
return GridView.builder(
gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
crossAxisCount: cols,
crossAxisSpacing: 10,
mainAxisSpacing: 10,
),
itemCount: gm.lenPrj,
itemBuilder: (context, index) => ProjectCard(index),
);
},
)
: Center(
child: Text(
"Add projects by pressing the \"+\" button.",
style: TextStyle(fontSize: 20),
),
),
),
),
);
}
}

24
lib/widgets/drawer.dart Normal file
View File

@@ -0,0 +1,24 @@
import 'package:flutter/material.dart';
import 'package:prod/widgets/simpleListTile.dart';
class ProdDrawer extends StatelessWidget {
const ProdDrawer({super.key});
@override
Widget build(BuildContext context) {
return Drawer(
child: ListView(
children: [
ListTile(
leading: Icon(Icons.arrow_back),
title: Text("Back"),
onTap: () => Navigator.pop(context),
),
Divider(),
SListTile(Icons.backup_table, "Editors", "/editors"),
SListTile(Icons.settings, "Settings", "/settings"),
],
),
);
}
}

View File

@@ -0,0 +1,31 @@
import "package:flutter/material.dart";
import "package:prod/models/editor.dart";
import "package:prod/models/globalModel.dart";
import "package:prod/models/project.dart";
import "package:provider/provider.dart";
import "package:yaru/yaru.dart";
import "package:process_run/shell.dart";
class EditorCard extends StatelessWidget {
const EditorCard(this.id, {super.key});
final int id;
@override
Widget build(BuildContext context) {
GlobalModel gm = Provider.of<GlobalModel>(context);
final Editor edt = gm.nthEdt(id);
return YaruBanner(
onTap: () {},
child: Center(
child: Column(
mainAxisAlignment: .start,
crossAxisAlignment: .start,
children: [
Text("${edt.name}", style: TextStyle(fontSize: 30)),
Text("${edt.commandTemplate}"),
],
),
),
);
}
}

24
lib/widgets/edtFAB.dart Normal file
View File

@@ -0,0 +1,24 @@
import "package:flutter/material.dart";
import "package:prod/models/editor.dart";
class EditorFAB extends StatelessWidget {
const EditorFAB({super.key});
@override
Widget build(BuildContext context) {
return FloatingActionButton(
onPressed: () {
// gm.add(
// Project.validated(
// "Kimi",
// "Rust",
// "/home/arrow/Gitted/cowin",
// [],
// true,
// ),
// );
},
child: Icon(Icons.add),
);
}
}

26
lib/widgets/prjFAB.dart Normal file
View File

@@ -0,0 +1,26 @@
import "package:flutter/material.dart";
import "package:prod/models/globalModel.dart";
import "package:prod/models/project.dart";
import "package:provider/provider.dart";
class ProjFAB extends StatelessWidget {
const ProjFAB({super.key});
@override
Widget build(BuildContext context) {
return FloatingActionButton(
onPressed: () {
Provider.of<GlobalModel>(context, listen: false).addPrj(
Project.validated(
"Kimi",
"Rust",
"/home/arrow/Gitted/cowin",
[],
true,
),
);
},
child: Icon(Icons.add),
);
}
}

View File

@@ -0,0 +1,58 @@
import "package:flutter/material.dart";
import "package:prod/models/globalModel.dart";
import "package:prod/models/project.dart";
import "package:provider/provider.dart";
import "package:yaru/yaru.dart";
import "package:process_run/shell.dart";
class ProjectCard extends StatelessWidget {
const ProjectCard(this.id, {super.key});
final int id;
@override
Widget build(BuildContext context) {
GlobalModel gm = Provider.of<GlobalModel>(context);
final Project prj = gm.nthPrj(id);
return YaruBanner(
padding: .only(
left: kYaruPagePadding,
top: kYaruPagePadding,
bottom: kYaruPagePadding,
right: kYaruPagePadding,
),
onHover: (st) => gm.setHoverShow(id, st),
onTap: () {
var shell = Shell();
shell.run("konsole -e nvim ${prj.path.path}");
},
child: Center(
child: Column(
mainAxisAlignment: .start,
crossAxisAlignment: .start,
children: [
Row(
mainAxisAlignment: .spaceBetween,
children: [
Text("${prj.name}", style: TextStyle(fontSize: 30)),
gm.getHoverShow(id)
? ElevatedButton(
child: Icon(Icons.close),
onPressed: () => print("pressed delete"),
)
: Container(),
],
),
Row(
spacing: 10,
children: [
Text("${prj.language}"),
prj.isGit ? Icon(Icons.commit) : Container(),
],
),
gm.getHoverShow(id) ? Text("${prj.path.path}") : Container(),
],
),
),
);
}
}

View File

@@ -0,0 +1,21 @@
//ignore_for_file: file_names
import "package:flutter/material.dart";
class SListTile extends StatelessWidget {
const SListTile(this.icon, this.title, this.route, {super.key});
final IconData icon;
final String title;
final String route;
@override
Widget build(BuildContext context) {
return ListTile(
leading: Icon(icon),
title: Text(title),
onTap: () {
Navigator.pushNamed(context, route);
},
);
}
}