Initial commit: Roster HTTP client for GNOME
Roster is a modern HTTP client application for GNOME, similar to Postman, built with GTK 4 and libadwaita. Features: - Send HTTP requests (GET, POST, PUT, DELETE) - Configure custom headers with add/remove functionality - Request body editor for POST/PUT/DELETE requests - View response headers and bodies in separate tabs - Track request history with JSON persistence - Load previous requests from history with confirmation dialog - Beautiful GNOME-native UI with libadwaita components - HTTPie backend for reliable HTTP communication Technical implementation: - Python 3 with GTK 4 and libadwaita 1 - Meson build system with Flatpak support - Custom widgets (HeaderRow, HistoryItem) with GObject signals - Background threading for non-blocking HTTP requests - AdwTabView for modern tabbed interface - History persistence to ~/.config/roster/history.json - Comprehensive error handling and user feedback
This commit is contained in:
commit
cbc1972797
27
.gitignore
vendored
Normal file
27
.gitignore
vendored
Normal file
@ -0,0 +1,27 @@
|
||||
# Build directories
|
||||
builddir/
|
||||
build-dir/
|
||||
_build/
|
||||
.flatpak-builder/
|
||||
|
||||
# Backup files
|
||||
*~
|
||||
*.bak
|
||||
|
||||
# Python
|
||||
__pycache__/
|
||||
*.pyc
|
||||
*.pyo
|
||||
*.pyd
|
||||
.Python
|
||||
*.so
|
||||
|
||||
# IDE
|
||||
.vscode/
|
||||
.idea/
|
||||
*.swp
|
||||
*.swo
|
||||
|
||||
# Flatpak
|
||||
.flatpak/
|
||||
repo/
|
||||
128
CLAUDE.md
Normal file
128
CLAUDE.md
Normal file
@ -0,0 +1,128 @@
|
||||
# CLAUDE.md
|
||||
|
||||
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
||||
|
||||
## Project Overview
|
||||
|
||||
Roster is a GNOME application written in Python using GTK 4 and libadwaita. The project follows GNOME application conventions and uses the Meson build system.
|
||||
|
||||
- **Application ID**: `cz.vesp.roster`
|
||||
- **Build System**: Meson
|
||||
- **Runtime**: GNOME Platform (org.gnome.Platform)
|
||||
- **UI Framework**: GTK 4 + libadwaita (Adw)
|
||||
- **Language**: Python 3
|
||||
- **License**: GPL-3.0-or-later
|
||||
|
||||
## Build Commands
|
||||
|
||||
### Native Build (requires HTTPie installed on host)
|
||||
|
||||
```bash
|
||||
# Configure the build (from project root)
|
||||
meson setup builddir
|
||||
|
||||
# Build the project
|
||||
meson compile -C builddir
|
||||
|
||||
# Install locally
|
||||
meson install -C builddir
|
||||
|
||||
# Run tests
|
||||
meson test -C builddir
|
||||
|
||||
# Clean build directory
|
||||
rm -rf builddir
|
||||
```
|
||||
|
||||
### Running the Application
|
||||
|
||||
**IMPORTANT**: This application requires HTTPie to make HTTP requests.
|
||||
|
||||
- **Flatpak (GNOME Builder)**: HTTPie is automatically included as a Flatpak module
|
||||
- **Native**: Install HTTPie on your system with `pip install httpie` or `sudo dnf install httpie`
|
||||
|
||||
## Development with Flatpak
|
||||
|
||||
The project includes a Flatpak manifest (`cz.vesp.roster.json`) for building and running the application in a sandboxed environment:
|
||||
|
||||
```bash
|
||||
# Build with GNOME Builder (recommended for GNOME apps)
|
||||
# Or use flatpak-builder directly:
|
||||
flatpak-builder --user --install --force-clean build-dir cz.vesp.roster.json
|
||||
```
|
||||
|
||||
## Architecture
|
||||
|
||||
### Application Structure
|
||||
|
||||
The application follows the standard GNOME/Adwaita application pattern:
|
||||
|
||||
- **RosterApplication** (main.py): Singleton application class inheriting from `Adw.Application`
|
||||
- Manages application lifecycle and actions (quit, about, preferences)
|
||||
- Sets up keyboard shortcuts (e.g., Ctrl+Q for quit)
|
||||
- Creates and presents the main window
|
||||
|
||||
- **RosterWindow** (window.py): Main application window inheriting from `Adw.ApplicationWindow`
|
||||
- Defined as a GTK template class using `@Gtk.Template` decorator
|
||||
- UI layout loaded from `window.ui` via GResources
|
||||
|
||||
### Resource Management
|
||||
|
||||
UI files are packaged into a GResource bundle:
|
||||
- Defined in `src/roster.gresource.xml`
|
||||
- Resource base path: `/cz/vesp/roster`
|
||||
- UI files are preprocessed with `xml-stripblanks`
|
||||
|
||||
### File Organization
|
||||
|
||||
```
|
||||
src/
|
||||
__init__.py # Package marker
|
||||
main.py # Application class and entry point
|
||||
window.py # Main window class
|
||||
window.ui # Main window UI definition (GTK Builder XML)
|
||||
shortcuts-dialog.ui # Keyboard shortcuts dialog
|
||||
roster.in # Launcher script template (configured by Meson)
|
||||
roster.gresource.xml # GResource bundle definition
|
||||
|
||||
data/
|
||||
cz.vesp.roster.desktop.in # Desktop entry (i18n template)
|
||||
cz.vesp.roster.metainfo.xml.in # AppStream metadata (i18n template)
|
||||
cz.vesp.roster.gschema.xml # GSettings schema
|
||||
cz.vesp.roster.service.in # D-Bus service file (configured by Meson)
|
||||
icons/ # Application icons
|
||||
|
||||
po/
|
||||
# Translation files (gettext)
|
||||
```
|
||||
|
||||
### Key Patterns
|
||||
|
||||
1. **UI Templates**: Window classes use `@Gtk.Template` decorator with `resource_path` pointing to `.ui` files in GResources. Child widgets are accessed via `Gtk.Template.Child()`.
|
||||
|
||||
2. **Actions**: Application actions are created with the `create_action()` helper method, which sets up actions and optional keyboard shortcuts.
|
||||
|
||||
3. **Internationalization**: Uses gettext with domain "roster". Template files (`.in`) are processed through `i18n.merge_file()` during build.
|
||||
|
||||
4. **Configuration Data**: Meson substitutes variables in `.in` files using `configuration_data()` (e.g., paths, version).
|
||||
|
||||
5. **Module Installation**: Python modules are installed to `{datadir}/roster/roster/` and the launcher script is configured with the correct Python interpreter path.
|
||||
|
||||
## Adding New Python Modules
|
||||
|
||||
When adding new `.py` files to `src/`:
|
||||
1. Add the filename to `roster_sources` list in `src/meson.build`
|
||||
2. Import in `main.py` or other modules as needed
|
||||
|
||||
## Adding New UI Files
|
||||
|
||||
When adding new `.ui` files:
|
||||
1. Place in `src/` directory
|
||||
2. Add `<file>` entry to `src/roster.gresource.xml`
|
||||
3. Use `@Gtk.Template(resource_path='/cz/vesp/roster/filename.ui')` in the corresponding Python class
|
||||
|
||||
## GSettings Schema
|
||||
|
||||
Settings are defined in `data/cz.vesp.roster.gschema.xml`. After modifying:
|
||||
- Schema is validated during `meson test`
|
||||
- Compiled automatically during installation via `gnome.post_install()`
|
||||
675
COPYING
Normal file
675
COPYING
Normal file
@ -0,0 +1,675 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
the GNU General Public License is intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. We, the Free Software Foundation, use the
|
||||
GNU General Public License for most of our software; it applies also to
|
||||
any other work released this way by its authors. You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to prevent others from denying you
|
||||
these rights or asking you to surrender the rights. Therefore, you have
|
||||
certain responsibilities if you distribute copies of the software, or if
|
||||
you modify it: responsibilities to respect the freedom of others.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must pass on to the recipients the same
|
||||
freedoms that you received. You must make sure that they, too, receive
|
||||
or can get the source code. And you must show them these terms so they
|
||||
know their rights.
|
||||
|
||||
Developers that use the GNU GPL protect your rights with two steps:
|
||||
(1) assert copyright on the software, and (2) offer you this License
|
||||
giving you legal permission to copy, distribute and/or modify it.
|
||||
|
||||
For the developers' and authors' protection, the GPL clearly explains
|
||||
that there is no warranty for this free software. For both users' and
|
||||
authors' sake, the GPL requires that modified versions be marked as
|
||||
changed, so that their problems will not be attributed erroneously to
|
||||
authors of previous versions.
|
||||
|
||||
Some devices are designed to deny users access to install or run
|
||||
modified versions of the software inside them, although the manufacturer
|
||||
can do so. This is fundamentally incompatible with the aim of
|
||||
protecting users' freedom to change the software. The systematic
|
||||
pattern of such abuse occurs in the area of products for individuals to
|
||||
use, which is precisely where it is most unacceptable. Therefore, we
|
||||
have designed this version of the GPL to prohibit the practice for those
|
||||
products. If such problems arise substantially in other domains, we
|
||||
stand ready to extend this provision to those domains in future versions
|
||||
of the GPL, as needed to protect the freedom of users.
|
||||
|
||||
Finally, every program is threatened constantly by software patents.
|
||||
States should not allow patents to restrict development and use of
|
||||
software on general-purpose computers, but in those that do, we wish to
|
||||
avoid the special danger that patents applied to a free program could
|
||||
make it effectively proprietary. To prevent this, the GPL assures that
|
||||
patents cannot be used to render the program non-free.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Use with the GNU Affero General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU Affero General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the special requirements of the GNU Affero General Public License,
|
||||
section 13, concerning interaction through a network will apply to the
|
||||
combination as such.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program does terminal interaction, make it output a short
|
||||
notice like this when it starts in an interactive mode:
|
||||
|
||||
<program> Copyright (C) <year> <name of author>
|
||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, your program's commands
|
||||
might be different; for a GUI interface, you would use an "about box".
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU GPL, see
|
||||
<https://www.gnu.org/licenses/>.
|
||||
|
||||
The GNU General Public License does not permit incorporating your program
|
||||
into proprietary programs. If your program is a subroutine library, you
|
||||
may consider it more useful to permit linking proprietary applications with
|
||||
the library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License. But first, please read
|
||||
<https://www.gnu.org/philosophy/why-not-lgpl.html>.
|
||||
|
||||
38
README.md
Normal file
38
README.md
Normal file
@ -0,0 +1,38 @@
|
||||
# Roster
|
||||
|
||||
A modern HTTP client for GNOME, built with GTK 4 and libadwaita.
|
||||
|
||||
## Features
|
||||
|
||||
- Send HTTP requests (GET, POST, PUT, DELETE)
|
||||
- Configure custom headers and request bodies
|
||||
- View response headers and bodies
|
||||
- Track request history with persistence
|
||||
- Beautiful GNOME-native UI
|
||||
|
||||
## Dependencies
|
||||
|
||||
- GTK 4
|
||||
- libadwaita 1
|
||||
- Python 3
|
||||
- HTTPie (http command)
|
||||
|
||||
## Building
|
||||
|
||||
```bash
|
||||
meson setup builddir
|
||||
meson compile -C builddir
|
||||
sudo meson install -C builddir
|
||||
```
|
||||
|
||||
## Usage
|
||||
|
||||
Roster uses HTTPie as the backend for making HTTP requests. Ensure HTTPie is installed:
|
||||
|
||||
```bash
|
||||
pip install httpie
|
||||
# or on Fedora
|
||||
sudo dnf install httpie
|
||||
```
|
||||
|
||||
Then run Roster from your application menu or with the `roster` command.
|
||||
54
cz.vesp.roster.json
Normal file
54
cz.vesp.roster.json
Normal file
@ -0,0 +1,54 @@
|
||||
{
|
||||
"id" : "cz.vesp.roster",
|
||||
"runtime" : "org.gnome.Platform",
|
||||
"runtime-version" : "master",
|
||||
"sdk" : "org.gnome.Sdk",
|
||||
"command" : "roster",
|
||||
"finish-args" : [
|
||||
"--share=network",
|
||||
"--share=ipc",
|
||||
"--socket=fallback-x11",
|
||||
"--device=dri",
|
||||
"--socket=wayland",
|
||||
"--filesystem=xdg-config/roster:create"
|
||||
],
|
||||
"cleanup" : [
|
||||
"/include",
|
||||
"/lib/pkgconfig",
|
||||
"/man",
|
||||
"/share/doc",
|
||||
"/share/gtk-doc",
|
||||
"/share/man",
|
||||
"/share/pkgconfig",
|
||||
"*.la",
|
||||
"*.a"
|
||||
],
|
||||
"build-options" : {
|
||||
"build-args" : [
|
||||
"--share=network"
|
||||
]
|
||||
},
|
||||
"modules" : [
|
||||
{
|
||||
"name" : "httpie",
|
||||
"buildsystem" : "simple",
|
||||
"build-commands" : [
|
||||
"pip3 install --prefix=${FLATPAK_DEST} httpie"
|
||||
]
|
||||
},
|
||||
{
|
||||
"name" : "roster",
|
||||
"builddir" : true,
|
||||
"buildsystem" : "meson",
|
||||
"sources" : [
|
||||
{
|
||||
"type" : "git",
|
||||
"url" : "file:///home/pavelb/GnomeBuilderProjects"
|
||||
}
|
||||
],
|
||||
"config-opts" : [
|
||||
"--libdir=lib"
|
||||
]
|
||||
}
|
||||
]
|
||||
}
|
||||
10
data/cz.vesp.roster.desktop.in
Normal file
10
data/cz.vesp.roster.desktop.in
Normal file
@ -0,0 +1,10 @@
|
||||
[Desktop Entry]
|
||||
Name=roster
|
||||
Exec=roster
|
||||
Icon=cz.vesp.roster
|
||||
Terminal=false
|
||||
Type=Application
|
||||
Categories=Utility;
|
||||
Keywords=GTK;
|
||||
StartupNotify=true
|
||||
DBusActivatable=true
|
||||
5
data/cz.vesp.roster.gschema.xml
Normal file
5
data/cz.vesp.roster.gschema.xml
Normal file
@ -0,0 +1,5 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<schemalist gettext-domain="roster">
|
||||
<schema id="cz.vesp.roster" path="/cz/vesp/roster/">
|
||||
</schema>
|
||||
</schemalist>
|
||||
73
data/cz.vesp.roster.metainfo.xml.in
Normal file
73
data/cz.vesp.roster.metainfo.xml.in
Normal file
@ -0,0 +1,73 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<component type="desktop-application">
|
||||
<id>cz.vesp.roster</id>
|
||||
<metadata_license>CC0-1.0</metadata_license>
|
||||
<project_license>GPL-3.0-or-later</project_license>
|
||||
|
||||
<name>Roster</name>
|
||||
<summary>Keep the summary shorter, between 10 and 35 characters</summary>
|
||||
<description>
|
||||
<p>No description</p>
|
||||
</description>
|
||||
|
||||
<developer id="tld.vendor">
|
||||
<name>Developer name</name>
|
||||
</developer>
|
||||
|
||||
<!-- Required: Should be a link to the upstream homepage for the component -->
|
||||
<url type="homepage">https://example.org/</url>
|
||||
<!-- Recommended: It is highly recommended for open-source projects to display the source code repository -->
|
||||
<url type="vcs-browser">https://example.org/repository</url>
|
||||
<!-- Should point to the software's bug tracking system, for users to report new bugs -->
|
||||
<url type="bugtracker">https://example.org/issues</url>
|
||||
<!-- Should link a FAQ page for this software, to answer some of the most-asked questions in detail -->
|
||||
<!-- URLs of this type should point to a webpage where users can submit or modify translations of the upstream project -->
|
||||
<url type="translate">https://example.org/translate</url>
|
||||
<url type="faq">https://example.org/faq</url>
|
||||
<!-- Should provide a web link to an online user's reference, a software manual or help page -->
|
||||
<url type="help">https://example.org/help</url>
|
||||
<!-- URLs of this type should point to a webpage showing information on how to donate to the described software project -->
|
||||
<url type="donation">https://example.org/donate</url>
|
||||
<!-- This could for example be an HTTPS URL to an online form or a page describing how to contact the developer -->
|
||||
<url type="contact">https://example.org/contact</url>
|
||||
<!-- URLs of this type should point to a webpage showing information on how to contribute to the described software project -->
|
||||
<url type="contribute">https://example.org/contribute</url>
|
||||
|
||||
<translation type="gettext">roster</translation>
|
||||
<!-- All graphical applications having a desktop file must have this tag in the MetaInfo.
|
||||
If this is present, appstreamcli compose will pull icons, keywords and categories from the desktop file. -->
|
||||
<launchable type="desktop-id">cz.vesp.roster.desktop</launchable>
|
||||
<!-- Use the OARS website (https://hughsie.github.io/oars/generate.html) to generate these and make sure to use oars-1.1 -->
|
||||
<content_rating type="oars-1.1" />
|
||||
|
||||
<!-- Applications should set a brand color in both light and dark variants like so -->
|
||||
<branding>
|
||||
<color type="primary" scheme_preference="light">#ff00ff</color>
|
||||
<color type="primary" scheme_preference="dark">#993d3d</color>
|
||||
</branding>
|
||||
|
||||
<screenshots>
|
||||
<screenshot type="default">
|
||||
<image>https://example.org/example1.png</image>
|
||||
<caption>A caption</caption>
|
||||
</screenshot>
|
||||
<screenshot>
|
||||
<image>https://example.org/example2.png</image>
|
||||
<caption>A caption</caption>
|
||||
</screenshot>
|
||||
</screenshots>
|
||||
|
||||
<releases>
|
||||
<release version="1.0.1" date="2024-01-18">
|
||||
<url type="details">https://example.org/changelog.html#version_1.0.1</url>
|
||||
<description translate="no">
|
||||
<p>Release description</p>
|
||||
<ul>
|
||||
<li>List of changes</li>
|
||||
<li>List of changes</li>
|
||||
</ul>
|
||||
</description>
|
||||
</release>
|
||||
</releases>
|
||||
|
||||
</component>
|
||||
3
data/cz.vesp.roster.service.in
Normal file
3
data/cz.vesp.roster.service.in
Normal file
@ -0,0 +1,3 @@
|
||||
[D-BUS Service]
|
||||
Name=cz.vesp.roster
|
||||
Exec=@bindir@/roster --gapplication-service
|
||||
130
data/icons/hicolor/scalable/apps/cz.vesp.roster.svg
Normal file
130
data/icons/hicolor/scalable/apps/cz.vesp.roster.svg
Normal file
@ -0,0 +1,130 @@
|
||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<svg
|
||||
xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||
xmlns:cc="http://creativecommons.org/ns#"
|
||||
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
||||
xmlns:svg="http://www.w3.org/2000/svg"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="33.750061mm"
|
||||
height="33.750061mm"
|
||||
viewBox="0 0 33.750061 33.750061"
|
||||
version="1.1"
|
||||
id="svg974">
|
||||
<defs
|
||||
id="defs968">
|
||||
<clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath18689">
|
||||
<rect
|
||||
clip-path="none"
|
||||
transform="rotate(45)"
|
||||
ry="32.000008"
|
||||
rx="32.000008"
|
||||
y="123.9986"
|
||||
x="486.03726"
|
||||
height="362.94299"
|
||||
width="362.94299"
|
||||
id="rect18691"
|
||||
style="display:inline;opacity:1;vector-effect:none;fill:#4a86cf;fill-opacity:1;stroke:none;stroke-width:26.0669;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal;enable-background:new" />
|
||||
</clipPath>
|
||||
<clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath18689-3">
|
||||
<rect
|
||||
clip-path="none"
|
||||
transform="rotate(45)"
|
||||
ry="32.000008"
|
||||
rx="32.000008"
|
||||
y="123.9986"
|
||||
x="486.03726"
|
||||
height="362.94299"
|
||||
width="362.94299"
|
||||
id="rect18691-6"
|
||||
style="display:inline;opacity:1;vector-effect:none;fill:#4a86cf;fill-opacity:1;stroke:none;stroke-width:26.0669;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal;enable-background:new" />
|
||||
</clipPath>
|
||||
</defs>
|
||||
<metadata
|
||||
id="metadata971">
|
||||
<rdf:RDF>
|
||||
<cc:Work
|
||||
rdf:about="">
|
||||
<dc:format>image/svg+xml</dc:format>
|
||||
<dc:type
|
||||
rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
|
||||
<dc:title></dc:title>
|
||||
</cc:Work>
|
||||
</rdf:RDF>
|
||||
</metadata>
|
||||
<g
|
||||
id="layer1"
|
||||
transform="translate(-61.819823,-103.94395)">
|
||||
<g
|
||||
transform="matrix(0.26367235,0,0,0.26367235,61.819823,-529.39703)"
|
||||
style="display:inline;stroke-width:0.25;enable-background:new"
|
||||
id="g1836">
|
||||
<title
|
||||
id="title1838">application-x-executable</title>
|
||||
<g
|
||||
transform="matrix(0.25,0,0,0.25,0,2295)"
|
||||
id="g18818"
|
||||
style="stroke-width:0.25">
|
||||
<g
|
||||
style="stroke-width:0.269963"
|
||||
transform="matrix(0.92605186,0,0,0.92605186,18.930729,50.876335)"
|
||||
id="g18590">
|
||||
<g
|
||||
style="stroke-width:0.269963"
|
||||
id="g18681"
|
||||
clip-path="url(#clipPath18689-3)">
|
||||
<rect
|
||||
style="opacity:1;vector-effect:none;fill:#3584e4;fill-opacity:1;stroke:none;stroke-width:8.22095;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal"
|
||||
id="rect18571"
|
||||
width="424"
|
||||
height="424"
|
||||
x="458.33722"
|
||||
y="90.641701"
|
||||
rx="10.092117"
|
||||
ry="10.092117"
|
||||
transform="matrix(0.60528171,0.60528171,-0.60528171,0.60528171,33.440632,99.073632)"
|
||||
clip-path="none" />
|
||||
<circle
|
||||
style="opacity:1;vector-effect:none;fill:#f66151;fill-opacity:1;stroke:none;stroke-width:7.03712;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal"
|
||||
id="path18706"
|
||||
cx="0"
|
||||
cy="0"
|
||||
r="0"
|
||||
transform="translate(0,-212)" />
|
||||
<circle
|
||||
style="opacity:1;vector-effect:none;fill:#f66151;fill-opacity:1;stroke:none;stroke-width:7.03712;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal"
|
||||
id="path18708"
|
||||
cx="0"
|
||||
cy="0"
|
||||
r="0"
|
||||
transform="translate(0,-212)" />
|
||||
<path
|
||||
style="display:inline;opacity:1;vector-effect:none;fill:#98c1f1;fill-opacity:1;stroke:none;stroke-width:7.03712;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal;enable-background:new"
|
||||
d="m 408.91993,561.9183 -9.8861,29.82892 a 172.97099,172.97099 0 0 0 -1.42693,-0.0713 172.97099,172.97099 0 0 0 -23.92891,1.85189 l -13.80082,-28.50125 a 203.29325,203.29325 0 0 0 -29.40085,7.97217 l 2.28619,31.40474 a 172.97099,172.97099 0 0 0 -22.73152,11.31923 l -23.71796,-21.103 a 203.29325,203.29325 0 0 0 -24.05918,18.6741 l 14.09863,28.07319 a 172.97099,172.97099 0 0 0 -16.63608,19.21074 l -30.05845,-10.44758 a 203.29325,203.29325 0 0 0 -15.01683,26.48807 l 23.73035,20.50738 a 172.97099,172.97099 0 0 0 -7.98456,24.14293 l -31.73044,1.84879 a 203.29325,203.29325 0 0 0 -3.77825,30.21664 l 29.82892,9.8861 a 172.97099,172.97099 0 0 0 -0.0713,1.42693 172.97099,172.97099 0 0 0 1.85188,23.92889 l -28.50125,13.80084 a 203.29325,203.29325 0 0 0 7.97215,29.40084 l 31.40475,-2.28619 a 172.97099,172.97099 0 0 0 11.31922,22.73152 l -21.10296,23.71797 a 203.29325,203.29325 0 0 0 18.67409,24.05918 l 28.07319,-14.09863 a 172.97099,172.97099 0 0 0 19.21074,16.63606 l -10.44758,30.05847 a 203.29325,203.29325 0 0 0 26.48806,15.01683 l 20.50739,-23.73036 a 172.97099,172.97099 0 0 0 24.14293,7.98457 l 1.8488,31.73043 a 203.29325,203.29325 0 0 0 30.21666,3.77826 l 9.8861,-29.82892 a 172.97099,172.97099 0 0 0 1.42693,0.0713 172.97099,172.97099 0 0 0 23.9289,-1.85188 l 13.80084,28.50125 a 203.29325,203.29325 0 0 0 29.40084,-7.97217 l -2.2862,-31.40474 a 172.97099,172.97099 0 0 0 22.73153,-11.31922 l 23.71796,21.10297 A 203.29325,203.29325 0 0 0 532.96,916.00016 l -14.09864,-28.07319 a 172.97099,172.97099 0 0 0 16.63607,-19.21073 l 30.05846,10.44757 a 203.29325,203.29325 0 0 0 15.01683,-26.48807 l -23.73036,-20.50738 a 172.97099,172.97099 0 0 0 7.98457,-24.14293 l 31.73044,-1.84879 a 203.29325,203.29325 0 0 0 3.77825,-30.21667 l -29.82892,-9.8861 a 172.97099,172.97099 0 0 0 0.0713,-1.42692 172.97099,172.97099 0 0 0 -1.85189,-23.9289 l 28.50124,-13.80084 a 203.29325,203.29325 0 0 0 -7.97215,-29.40084 l -31.40474,2.2862 a 172.97099,172.97099 0 0 0 -11.31923,-22.73153 l 21.10297,-23.71797 a 203.29325,203.29325 0 0 0 -18.67409,-24.05918 l -28.07319,14.09863 a 172.97099,172.97099 0 0 0 -19.21074,-16.63606 l 10.44757,-30.05847 A 203.29325,203.29325 0 0 0 485.6357,581.68117 l -20.50738,23.73035 a 172.97099,172.97099 0 0 0 -24.14293,-7.98455 l -1.84879,-31.73044 a 203.29325,203.29325 0 0 0 -30.21667,-3.77826 z M 397.6069,637.72208 A 126.92605,126.92605 0 0 1 524.5318,764.64699 126.92605,126.92605 0 0 1 397.6069,891.57189 126.92605,126.92605 0 0 1 270.682,764.64699 126.92605,126.92605 0 0 1 397.6069,637.72208 Z"
|
||||
id="path18717-4" />
|
||||
<path
|
||||
id="path18758"
|
||||
d="m 51.748325,401.28402 -9.8861,29.82892 c -0.475543,-0.0257 -0.951191,-0.0495 -1.42693,-0.0713 -8.00956,0.0625 -16.005106,0.6813 -23.92891,1.85189 L 2.7055639,404.39228 c -9.9858697,1.91835 -19.8137359,4.58322 -29.4008489,7.97217 l 2.28619,31.40474 c -7.844275,3.21103 -15.441918,6.9943 -22.73152,11.31923 l -23.71796,-21.103 c -8.475372,5.61437 -16.517661,11.85658 -24.05918,18.6741 l 14.09863,28.07319 c -6.008901,5.98701 -11.569263,12.40791 -16.636077,19.21074 l -30.058438,-10.44758 c -5.66072,8.44155 -10.68041,17.29574 -15.01683,26.48807 l 23.73035,20.50738 c -3.25027,7.84084 -5.919,15.91028 -7.98456,24.14293 l -31.73044,1.84879 c -2.01308,9.96359 -3.27604,20.06413 -3.77825,30.21664 l 29.82892,9.8861 c -0.0257,0.47554 -0.0495,0.95119 -0.0713,1.42693 0.0625,8.00955 0.68129,16.00509 1.85188,23.92889 l -28.50125,13.80084 c 1.91835,9.98587 4.58321,19.81373 7.97215,29.40084 l 31.40475,-2.28619 c 3.21102,7.84427 6.99429,15.44192 11.31922,22.73152 l -21.10296,23.71797 c 5.61437,8.47537 11.85658,16.51766 18.67409,24.05918 l 28.073175,-14.09863 c 5.987006,6.00889 12.407911,11.56925 19.21074,16.63606 l -10.44758,30.05847 c 8.441549,5.66072 17.295731,10.68041 26.48806,15.01683 l 20.50739,-23.73036 c 7.840839,3.25027 15.910282,5.91901 24.142929,7.98457 l 1.8488,31.73043 c 9.9635962,2.01309 20.064147,3.27605 30.216661,3.77826 l 9.8861,-29.82892 c 0.475543,0.0257 0.951191,0.0495 1.42693,0.0713 8.009557,-0.0625 16.005099,-0.68129 23.9289,-1.85188 l 13.80084,28.50125 c 9.985867,-1.91835 19.813731,-4.58322 29.400855,-7.97217 l -2.2862,-31.40474 c 7.84428,-3.21102 15.44192,-6.99429 22.73153,-11.31922 l 23.71796,21.10297 c 8.47538,-5.61437 16.51767,-11.85658 24.05919,-18.6741 l -14.09864,-28.07319 c 6.0089,-5.987 11.56926,-12.4079 16.63607,-19.21073 l 30.05846,10.44757 c 5.66072,-8.44155 10.68041,-17.29574 15.01683,-26.48807 l -23.73036,-20.50738 c 3.25027,-7.84084 5.91901,-15.91028 7.98457,-24.14293 l 31.73044,-1.84879 c 2.01308,-9.9636 3.27604,-20.06415 3.77825,-30.21667 l -29.82892,-9.8861 c 0.0257,-0.47554 0.0495,-0.95118 0.0713,-1.42692 -0.0625,-8.00956 -0.6813,-16.0051 -1.85189,-23.9289 l 28.50124,-13.80084 c -1.91835,-9.98587 -4.58321,-19.81373 -7.97215,-29.40084 l -31.40474,2.2862 c -3.21103,-7.84428 -6.9943,-15.44192 -11.31923,-22.73153 l 21.10297,-23.71797 c -5.61437,-8.47537 -11.85658,-16.51766 -18.67409,-24.05918 l -28.07319,14.09863 c -5.98701,-6.00889 -12.40791,-11.56925 -19.21074,-16.63606 l 10.44757,-30.05847 c -8.44155,-5.66072 -17.29572,-10.6804 -26.48805,-15.01682 l -20.50738,23.73035 c -7.84085,-3.25027 -15.910298,-5.91899 -24.142945,-7.98455 l -1.84879,-31.73044 c -9.9636,-2.01309 -20.064152,-3.27605 -30.21667,-3.77826 z"
|
||||
style="display:inline;opacity:1;vector-effect:none;fill:#1a5fb4;fill-opacity:1;stroke:none;stroke-width:7.03712;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal;enable-background:new" />
|
||||
</g>
|
||||
</g>
|
||||
<path
|
||||
style="display:inline;opacity:0.534;vector-effect:none;fill:#1a5fb4;fill-opacity:1;stroke:none;stroke-width:1.62918;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal;enable-background:new"
|
||||
clip-path="none"
|
||||
d="m 8.4765625,2676 c -1.1711695,2.8866 -0.5827763,6.3078 1.7656255,8.6562 l 48.101562,48.1016 c 3.133898,3.1339 8.178602,3.1339 11.3125,0 l 48.10156,-48.1016 c 2.3484,-2.3484 2.9368,-5.7696 1.76563,-8.6562 -0.39174,0.9655 -0.98013,1.8708 -1.76563,2.6562 l -48.10156,48.1016 c -3.133898,3.1339 -8.178602,3.1339 -11.3125,0 L 10.242188,2678.6562 C 9.4566904,2677.8708 8.8682972,2676.9655 8.4765625,2676 Z"
|
||||
transform="matrix(4,0,0,4,0,-10028)"
|
||||
id="rect18571-6" />
|
||||
</g>
|
||||
<rect
|
||||
y="2402"
|
||||
x="-1.5000001e-06"
|
||||
height="128"
|
||||
width="128"
|
||||
id="rect9125-7-2"
|
||||
style="display:inline;opacity:1;vector-effect:none;fill:none;fill-opacity:1;stroke:none;stroke-width:1.25;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none;paint-order:normal;enable-background:new" />
|
||||
</g>
|
||||
</g>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 12 KiB |
@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16"><g color="#000" fill="#2e3436"><path d="M7.188 2.281c-.094.056-.192.125-.29.19L5.566 3.803a1.684 1.684 0 11-2.17 2.17L2.332 7.037c.506-.069 1.017-.136 1.2.026.242.214.139 1.031.155 1.656.213.088.427.171.657.219.04.008.085-.007.125 0 .337-.525.683-1.288 1-1.344.322-.057.905.562 1.406.937a3.67 3.67 0 00.656-.468c-.195-.595-.594-1.369-.437-1.657.158-.29 1.019-.37 1.625-.531.028-.183.062-.371.062-.562 0-.075-.027-.146-.031-.22-.587-.217-1.435-.385-1.562-.687-.128-.302.34-1.021.593-1.593a3.722 3.722 0 00-.593-.532zm3.875 3.25c-.165.475-.305 1.086-.47 1.563-.43.047-.84.14-1.218.312-.38-.322-.787-.773-1.156-1.093a5.562 5.562 0 00-.688.468c.177.46.453 1.001.625 1.469-.298.309-.531.67-.719 1.063-.494 0-1.102-.084-1.593-.094a5.68 5.68 0 00-.219.812c.435.24 1.006.468 1.438.72-.006.093-.032.185-.032.28 0 .333.049.66.125.97-.382.304-.898.63-1.28.937.015.044.04.083.058.127l.613.613c.417-.1.868-.223 1.266-.303.248.343.532.626.875.875-.027.135-.068.283-.104.428.174-.063.34-.155.482-.297l1.432-1.432a1.994 1.994 0 01.533-3.918c.919 0 1.684.623 1.918 1.467l1.338-1.338c.06-.06.11-.124.156-.191-.035-.062-.06-.13-.1-.188.096-.152.205-.31.315-.47.017-.348-.1-.7-.37-.971l-.177-.176c-.28.192-.561.387-.83.555-.345-.233-.746-.383-1.156-.5-.077-.507-.107-1.132-.187-1.625a5.44 5.44 0 00-.875-.063zm-9.247.608c-.087.068-.173.138-.254.205l.014.035z" style="marker:none" overflow="visible"/><path d="M8.707.293a1 1 0 00-1.415 0l-6.999 7a1 1 0 000 1.413l7 7.001a1 1 0 001.415 0l7-7a1 1 0 000-1.413zm-.708 2.121l5.587 5.587L8 13.586 2.414 7.999z" style="line-height:normal;font-variant-ligatures:normal;font-variant-position:normal;font-variant-caps:normal;font-variant-numeric:normal;font-variant-alternates:normal;font-feature-settings:normal;text-indent:0;text-align:start;text-decoration-line:none;text-decoration-style:solid;text-decoration-color:#000;text-transform:none;text-orientation:mixed;shape-padding:0;isolation:auto;mix-blend-mode:normal;marker:none" font-weight="400" font-family="sans-serif" overflow="visible"/></g></svg>
|
||||
|
After Width: | Height: | Size: 2.0 KiB |
13
data/icons/meson.build
Normal file
13
data/icons/meson.build
Normal file
@ -0,0 +1,13 @@
|
||||
application_id = 'cz.vesp.roster'
|
||||
|
||||
scalable_dir = 'hicolor' / 'scalable' / 'apps'
|
||||
install_data(
|
||||
scalable_dir / ('@0@.svg').format(application_id),
|
||||
install_dir: get_option('datadir') / 'icons' / scalable_dir
|
||||
)
|
||||
|
||||
symbolic_dir = 'hicolor' / 'symbolic' / 'apps'
|
||||
install_data(
|
||||
symbolic_dir / ('@0@-symbolic.svg').format(application_id),
|
||||
install_dir: get_option('datadir') / 'icons' / symbolic_dir
|
||||
)
|
||||
46
data/meson.build
Normal file
46
data/meson.build
Normal file
@ -0,0 +1,46 @@
|
||||
desktop_file = i18n.merge_file(
|
||||
input: 'cz.vesp.roster.desktop.in',
|
||||
output: 'cz.vesp.roster.desktop',
|
||||
type: 'desktop',
|
||||
po_dir: '../po',
|
||||
install: true,
|
||||
install_dir: get_option('datadir') / 'applications'
|
||||
)
|
||||
|
||||
desktop_utils = find_program('desktop-file-validate', required: false)
|
||||
if desktop_utils.found()
|
||||
test('Validate desktop file', desktop_utils, args: [desktop_file])
|
||||
endif
|
||||
|
||||
appstream_file = i18n.merge_file(
|
||||
input: 'cz.vesp.roster.metainfo.xml.in',
|
||||
output: 'cz.vesp.roster.metainfo.xml',
|
||||
po_dir: '../po',
|
||||
install: true,
|
||||
install_dir: get_option('datadir') / 'metainfo'
|
||||
)
|
||||
|
||||
appstreamcli = find_program('appstreamcli', required: false, disabler: true)
|
||||
test('Validate appstream file', appstreamcli,
|
||||
args: ['validate', '--no-net', '--explain', appstream_file])
|
||||
|
||||
install_data('cz.vesp.roster.gschema.xml',
|
||||
install_dir: get_option('datadir') / 'glib-2.0' / 'schemas'
|
||||
)
|
||||
|
||||
compile_schemas = find_program('glib-compile-schemas', required: false, disabler: true)
|
||||
test('Validate schema file',
|
||||
compile_schemas,
|
||||
args: ['--strict', '--dry-run', meson.current_source_dir()])
|
||||
|
||||
|
||||
service_conf = configuration_data()
|
||||
service_conf.set('bindir', get_option('prefix') / get_option('bindir'))
|
||||
configure_file(
|
||||
input: 'cz.vesp.roster.service.in',
|
||||
output: 'cz.vesp.roster.service',
|
||||
configuration: service_conf,
|
||||
install_dir: get_option('datadir') / 'dbus-1' / 'services'
|
||||
)
|
||||
|
||||
subdir('icons')
|
||||
21
meson.build
Normal file
21
meson.build
Normal file
@ -0,0 +1,21 @@
|
||||
project('roster',
|
||||
version: '0.1.0',
|
||||
meson_version: '>= 1.0.0',
|
||||
default_options: [ 'warning_level=2', 'werror=false', ],
|
||||
)
|
||||
|
||||
i18n = import('i18n')
|
||||
gnome = import('gnome')
|
||||
|
||||
|
||||
|
||||
|
||||
subdir('data')
|
||||
subdir('src')
|
||||
subdir('po')
|
||||
|
||||
gnome.post_install(
|
||||
glib_compile_schemas: true,
|
||||
gtk_update_icon_cache: true,
|
||||
update_desktop_database: true,
|
||||
)
|
||||
1
po/LINGUAS
Normal file
1
po/LINGUAS
Normal file
@ -0,0 +1 @@
|
||||
# Please keep this file sorted alphabetically.
|
||||
8
po/POTFILES.in
Normal file
8
po/POTFILES.in
Normal file
@ -0,0 +1,8 @@
|
||||
# List of source files containing translatable strings.
|
||||
# Please keep this file sorted alphabetically.
|
||||
data/cz.vesp.roster.desktop.in
|
||||
data/cz.vesp.roster.metainfo.xml.in
|
||||
data/cz.vesp.roster.gschema.xml
|
||||
src/main.py
|
||||
src/window.py
|
||||
src/window.ui
|
||||
1
po/meson.build
Normal file
1
po/meson.build
Normal file
@ -0,0 +1 @@
|
||||
i18n.gettext('roster', preset: 'glib')
|
||||
0
src/__init__.py
Normal file
0
src/__init__.py
Normal file
74
src/history_manager.py
Normal file
74
src/history_manager.py
Normal file
@ -0,0 +1,74 @@
|
||||
# history_manager.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
import json
|
||||
import os
|
||||
from pathlib import Path
|
||||
from typing import List
|
||||
from .models import HistoryEntry
|
||||
|
||||
|
||||
class HistoryManager:
|
||||
"""Manages request history persistence to JSON file."""
|
||||
|
||||
def __init__(self):
|
||||
self.config_dir = Path.home() / '.config' / 'roster'
|
||||
self.history_file = self.config_dir / 'history.json'
|
||||
self._ensure_config_dir()
|
||||
|
||||
def _ensure_config_dir(self):
|
||||
"""Create config directory if it doesn't exist."""
|
||||
self.config_dir.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
def load_history(self) -> List[HistoryEntry]:
|
||||
"""Load history from JSON file."""
|
||||
if not self.history_file.exists():
|
||||
return []
|
||||
|
||||
try:
|
||||
with open(self.history_file, 'r') as f:
|
||||
data = json.load(f)
|
||||
|
||||
return [HistoryEntry.from_dict(entry) for entry in data.get('entries', [])]
|
||||
except Exception as e:
|
||||
print(f"Error loading history: {e}")
|
||||
return []
|
||||
|
||||
def save_history(self, entries: List[HistoryEntry]):
|
||||
"""Save history to JSON file."""
|
||||
try:
|
||||
data = {
|
||||
'version': 1,
|
||||
'entries': [entry.to_dict() for entry in entries]
|
||||
}
|
||||
|
||||
with open(self.history_file, 'w') as f:
|
||||
json.dump(data, f, indent=2)
|
||||
except Exception as e:
|
||||
print(f"Error saving history: {e}")
|
||||
|
||||
def add_entry(self, entry: HistoryEntry):
|
||||
"""Add new entry to history and save."""
|
||||
entries = self.load_history()
|
||||
entries.insert(0, entry) # Most recent first
|
||||
|
||||
# Limit history size to 100 entries
|
||||
entries = entries[:100]
|
||||
|
||||
self.save_history(entries)
|
||||
141
src/http_client.py
Normal file
141
src/http_client.py
Normal file
@ -0,0 +1,141 @@
|
||||
# http_client.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
import subprocess
|
||||
import time
|
||||
from typing import Tuple, Optional
|
||||
from .models import HttpRequest, HttpResponse
|
||||
|
||||
|
||||
class HttpClient:
|
||||
"""HTTP client using HTTPie as backend."""
|
||||
|
||||
def execute_request(self, request: HttpRequest) -> Tuple[Optional[HttpResponse], Optional[str]]:
|
||||
"""
|
||||
Execute HTTP request using HTTPie.
|
||||
|
||||
Args:
|
||||
request: HttpRequest object with method, url, headers, body
|
||||
|
||||
Returns:
|
||||
Tuple of (response, error_message)
|
||||
If successful: (HttpResponse, None)
|
||||
If failed: (None, error_message)
|
||||
"""
|
||||
# Build command
|
||||
cmd = ['http', '--print=HhBb', request.method, request.url]
|
||||
|
||||
# Add headers
|
||||
for key, value in request.headers.items():
|
||||
if key and value: # Skip empty headers
|
||||
cmd.append(f'{key}:{value}')
|
||||
|
||||
# Prepare stdin for body (if POST/PUT/DELETE and body exists)
|
||||
stdin_data = None
|
||||
if request.method in ['POST', 'PUT', 'DELETE'] and request.body.strip():
|
||||
stdin_data = request.body.encode('utf-8')
|
||||
else:
|
||||
# Only use --ignore-stdin when there's no body
|
||||
cmd.insert(2, '--ignore-stdin')
|
||||
|
||||
# Execute with timing
|
||||
start_time = time.time()
|
||||
try:
|
||||
result = subprocess.run(
|
||||
cmd,
|
||||
input=stdin_data,
|
||||
capture_output=True,
|
||||
timeout=30, # 30 second timeout
|
||||
text=False # We'll decode manually
|
||||
)
|
||||
end_time = time.time()
|
||||
response_time = (end_time - start_time) * 1000 # Convert to ms
|
||||
|
||||
# Parse output
|
||||
output = result.stdout.decode('utf-8', errors='replace')
|
||||
|
||||
if result.returncode != 0:
|
||||
error_msg = result.stderr.decode('utf-8', errors='replace')
|
||||
return None, error_msg
|
||||
|
||||
# Parse HTTPie output
|
||||
response = self._parse_httpie_output(output, response_time)
|
||||
return response, None
|
||||
|
||||
except subprocess.TimeoutExpired:
|
||||
return None, "Request timeout (30 seconds)"
|
||||
except FileNotFoundError:
|
||||
return None, "HTTPie not installed. Please install with: pip install httpie"
|
||||
except Exception as e:
|
||||
return None, f"Error: {str(e)}"
|
||||
|
||||
def _parse_httpie_output(self, output: str, response_time: float) -> HttpResponse:
|
||||
"""
|
||||
Parse HTTPie --print=HhBb output format.
|
||||
|
||||
Format:
|
||||
REQUEST HEADERS
|
||||
<blank line>
|
||||
REQUEST BODY
|
||||
<blank line>
|
||||
<blank line>
|
||||
RESPONSE HEADERS (starts with HTTP/1.1 200 OK)
|
||||
<blank line>
|
||||
RESPONSE BODY
|
||||
"""
|
||||
# Split output into lines to manually find response section
|
||||
lines = output.split('\n')
|
||||
|
||||
# Find where response headers start (line starting with HTTP/)
|
||||
response_header_start = 0
|
||||
for i, line in enumerate(lines):
|
||||
if line.startswith('HTTP/'):
|
||||
response_header_start = i
|
||||
break
|
||||
|
||||
# Parse status line
|
||||
status_line = lines[response_header_start]
|
||||
parts = status_line.split(' ', 2)
|
||||
|
||||
if len(parts) < 2:
|
||||
raise ValueError(f"Invalid status line: {status_line}")
|
||||
|
||||
status_code = int(parts[1])
|
||||
status_text = parts[2] if len(parts) > 2 else ''
|
||||
|
||||
# Find where response headers end (first empty line after response starts)
|
||||
response_body_start = response_header_start + 1
|
||||
for i in range(response_header_start + 1, len(lines)):
|
||||
if lines[i].strip() == '':
|
||||
response_body_start = i + 1
|
||||
break
|
||||
|
||||
# Extract response headers (from HTTP/ line to first blank line)
|
||||
response_headers = '\n'.join(lines[response_header_start:response_body_start - 1])
|
||||
|
||||
# Extract response body (everything after the blank line)
|
||||
response_body = '\n'.join(lines[response_body_start:])
|
||||
|
||||
return HttpResponse(
|
||||
status_code=status_code,
|
||||
status_text=status_text,
|
||||
headers=response_headers,
|
||||
body=response_body.strip(),
|
||||
response_time_ms=response_time
|
||||
)
|
||||
246
src/main-window.ui
Normal file
246
src/main-window.ui
Normal file
@ -0,0 +1,246 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<interface>
|
||||
<requires lib="gtk" version="4.0"/>
|
||||
<requires lib="Adw" version="1.0"/>
|
||||
|
||||
<template class="RosterWindow" parent="AdwApplicationWindow">
|
||||
<property name="title">Roster</property>
|
||||
<property name="default-width">1200</property>
|
||||
<property name="default-height">800</property>
|
||||
<property name="content">
|
||||
<object class="AdwToolbarView">
|
||||
|
||||
<!-- Header Bar -->
|
||||
<child type="top">
|
||||
<object class="AdwHeaderBar">
|
||||
<child type="end">
|
||||
<object class="GtkMenuButton">
|
||||
<property name="primary">True</property>
|
||||
<property name="icon-name">open-menu-symbolic</property>
|
||||
<property name="tooltip-text" translatable="yes">Main Menu</property>
|
||||
<property name="menu-model">primary_menu</property>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Main Content -->
|
||||
<property name="content">
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">vertical</property>
|
||||
|
||||
<!-- URL Input Section -->
|
||||
<child>
|
||||
<object class="AdwClamp">
|
||||
<property name="maximum-size">1000</property>
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">horizontal</property>
|
||||
<property name="spacing">12</property>
|
||||
<property name="margin-start">12</property>
|
||||
<property name="margin-end">12</property>
|
||||
<property name="margin-top">12</property>
|
||||
<property name="margin-bottom">12</property>
|
||||
|
||||
<!-- Method Dropdown -->
|
||||
<child>
|
||||
<object class="GtkDropDown" id="method_dropdown">
|
||||
<property name="enable-search">False</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- URL Entry -->
|
||||
<child>
|
||||
<object class="GtkEntry" id="url_entry">
|
||||
<property name="placeholder-text">Enter URL...</property>
|
||||
<property name="hexpand">True</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Send Button -->
|
||||
<child>
|
||||
<object class="GtkButton" id="send_button">
|
||||
<property name="label">Send</property>
|
||||
<signal name="clicked" handler="on_send_clicked"/>
|
||||
<style>
|
||||
<class name="suggested-action"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Split View: Request (left) | Response (right) -->
|
||||
<child>
|
||||
<object class="AdwOverlaySplitView" id="split_view">
|
||||
<property name="vexpand">True</property>
|
||||
<property name="sidebar-position">start</property>
|
||||
<property name="show-sidebar">True</property>
|
||||
<property name="min-sidebar-width">300</property>
|
||||
<property name="max-sidebar-width">600</property>
|
||||
|
||||
<!-- LEFT SIDEBAR: Request Configuration -->
|
||||
<property name="sidebar">
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">vertical</property>
|
||||
|
||||
<!-- Request Tabs Placeholder -->
|
||||
<child>
|
||||
<object class="GtkBox" id="request_tabs_container">
|
||||
<property name="orientation">vertical</property>
|
||||
<property name="vexpand">True</property>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</property>
|
||||
|
||||
<!-- RIGHT CONTENT: Response Display -->
|
||||
<property name="content">
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">vertical</property>
|
||||
|
||||
<!-- Response Tabs Placeholder -->
|
||||
<child>
|
||||
<object class="GtkBox" id="response_tabs_container">
|
||||
<property name="orientation">vertical</property>
|
||||
<property name="vexpand">True</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Status Bar -->
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">horizontal</property>
|
||||
<property name="spacing">12</property>
|
||||
<property name="margin-start">12</property>
|
||||
<property name="margin-end">12</property>
|
||||
<property name="margin-top">6</property>
|
||||
<property name="margin-bottom">6</property>
|
||||
|
||||
<child>
|
||||
<object class="GtkLabel" id="status_label">
|
||||
<property name="label">Ready</property>
|
||||
<style>
|
||||
<class name="heading"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkLabel" id="time_label">
|
||||
<property name="label"></property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Spacer -->
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="hexpand">True</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- History Toggle Button -->
|
||||
<child>
|
||||
<object class="GtkToggleButton" id="history_toggle_button">
|
||||
<property name="icon-name">view-list-symbolic</property>
|
||||
<property name="tooltip-text">Toggle History Panel</property>
|
||||
<signal name="toggled" handler="on_history_toggle_button_toggled"/>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- History Panel (Collapsible) -->
|
||||
<child>
|
||||
<object class="GtkRevealer" id="history_revealer">
|
||||
<property name="reveal-child">False</property>
|
||||
<property name="transition-type">slide-up</property>
|
||||
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">vertical</property>
|
||||
<property name="height-request">200</property>
|
||||
|
||||
<!-- History Header -->
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">horizontal</property>
|
||||
<property name="spacing">12</property>
|
||||
<property name="margin-start">12</property>
|
||||
<property name="margin-end">12</property>
|
||||
<property name="margin-top">6</property>
|
||||
|
||||
<child>
|
||||
<object class="GtkLabel">
|
||||
<property name="label">Request History</property>
|
||||
<property name="hexpand">True</property>
|
||||
<property name="xalign">0</property>
|
||||
<style>
|
||||
<class name="heading"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkButton" id="hide_history_button">
|
||||
<property name="icon-name">go-down-symbolic</property>
|
||||
<property name="tooltip-text">Hide history</property>
|
||||
<signal name="clicked" handler="on_toggle_history"/>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- History List -->
|
||||
<child>
|
||||
<object class="GtkScrolledWindow">
|
||||
<property name="vexpand">True</property>
|
||||
|
||||
<child>
|
||||
<object class="GtkListBox" id="history_listbox">
|
||||
<style>
|
||||
<class name="boxed-list"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</property>
|
||||
</object>
|
||||
</property>
|
||||
</template>
|
||||
|
||||
<menu id="primary_menu">
|
||||
<section>
|
||||
<item>
|
||||
<attribute name="label" translatable="yes">Toggle History</attribute>
|
||||
<attribute name="action">win.toggle-history</attribute>
|
||||
</item>
|
||||
</section>
|
||||
<section>
|
||||
<item>
|
||||
<attribute name="label" translatable="yes">_Preferences</attribute>
|
||||
<attribute name="action">app.preferences</attribute>
|
||||
</item>
|
||||
<item>
|
||||
<attribute name="label" translatable="yes">_Keyboard Shortcuts</attribute>
|
||||
<attribute name="action">app.shortcuts</attribute>
|
||||
</item>
|
||||
<item>
|
||||
<attribute name="label" translatable="yes">_About Roster</attribute>
|
||||
<attribute name="action">app.about</attribute>
|
||||
</item>
|
||||
</section>
|
||||
</menu>
|
||||
</interface>
|
||||
90
src/main.py
Normal file
90
src/main.py
Normal file
@ -0,0 +1,90 @@
|
||||
# main.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
import sys
|
||||
import gi
|
||||
|
||||
gi.require_version('Gtk', '4.0')
|
||||
gi.require_version('Adw', '1')
|
||||
|
||||
from gi.repository import Gtk, Gio, Adw
|
||||
from .window import RosterWindow
|
||||
|
||||
|
||||
class RosterApplication(Adw.Application):
|
||||
"""The main application singleton class."""
|
||||
|
||||
def __init__(self):
|
||||
super().__init__(application_id='cz.vesp.roster',
|
||||
flags=Gio.ApplicationFlags.DEFAULT_FLAGS,
|
||||
resource_base_path='/cz/vesp/roster')
|
||||
self.create_action('quit', lambda *_: self.quit(), ['<control>q'])
|
||||
self.create_action('about', self.on_about_action)
|
||||
self.create_action('preferences', self.on_preferences_action)
|
||||
|
||||
def do_activate(self):
|
||||
"""Called when the application is activated.
|
||||
|
||||
We raise the application's main window, creating it if
|
||||
necessary.
|
||||
"""
|
||||
win = self.props.active_window
|
||||
if not win:
|
||||
win = RosterWindow(application=self)
|
||||
win.present()
|
||||
|
||||
def on_about_action(self, *args):
|
||||
"""Callback for the app.about action."""
|
||||
about = Adw.AboutDialog(application_name='Roster',
|
||||
application_icon='cz.vesp.roster',
|
||||
developer_name='Pavel Baksy',
|
||||
version='0.1.0',
|
||||
developers=['Pavel Baksy'],
|
||||
copyright='© 2025 Pavel Baksy',
|
||||
comments='HTTP client for testing APIs')
|
||||
about.set_website('https://github.com/pavelb/roster')
|
||||
about.set_license_type(Gtk.License.GPL_3_0)
|
||||
# Translators: Replace "translator-credits" with your name/username, and optionally an email or URL.
|
||||
about.set_translator_credits(_('translator-credits'))
|
||||
about.present(self.props.active_window)
|
||||
|
||||
def on_preferences_action(self, widget, _):
|
||||
"""Callback for the app.preferences action."""
|
||||
print('app.preferences action activated')
|
||||
|
||||
def create_action(self, name, callback, shortcuts=None):
|
||||
"""Add an application action.
|
||||
|
||||
Args:
|
||||
name: the name of the action
|
||||
callback: the function to be called when the action is
|
||||
activated
|
||||
shortcuts: an optional list of accelerators
|
||||
"""
|
||||
action = Gio.SimpleAction.new(name, None)
|
||||
action.connect("activate", callback)
|
||||
self.add_action(action)
|
||||
if shortcuts:
|
||||
self.set_accels_for_action(f"app.{name}", shortcuts)
|
||||
|
||||
|
||||
def main(version):
|
||||
"""The application's entry point."""
|
||||
app = RosterApplication()
|
||||
return app.run(sys.argv)
|
||||
47
src/meson.build
Normal file
47
src/meson.build
Normal file
@ -0,0 +1,47 @@
|
||||
pkgdatadir = get_option('prefix') / get_option('datadir') / meson.project_name()
|
||||
moduledir = pkgdatadir / 'roster'
|
||||
gnome = import('gnome')
|
||||
|
||||
gnome.compile_resources('roster',
|
||||
'roster.gresource.xml',
|
||||
gresource_bundle: true,
|
||||
install: true,
|
||||
install_dir: pkgdatadir,
|
||||
)
|
||||
|
||||
python = import('python')
|
||||
|
||||
conf = configuration_data()
|
||||
conf.set('PYTHON', python.find_installation('python3').full_path())
|
||||
conf.set('VERSION', meson.project_version())
|
||||
conf.set('localedir', get_option('prefix') / get_option('localedir'))
|
||||
conf.set('pkgdatadir', pkgdatadir)
|
||||
|
||||
configure_file(
|
||||
input: 'roster.in',
|
||||
output: 'roster',
|
||||
configuration: conf,
|
||||
install: true,
|
||||
install_dir: get_option('bindir'),
|
||||
install_mode: 'rwxr-xr-x'
|
||||
)
|
||||
|
||||
roster_sources = [
|
||||
'__init__.py',
|
||||
'main.py',
|
||||
'window.py',
|
||||
'models.py',
|
||||
'http_client.py',
|
||||
'history_manager.py',
|
||||
]
|
||||
|
||||
install_data(roster_sources, install_dir: moduledir)
|
||||
|
||||
# Install widgets submodule
|
||||
widgets_sources = [
|
||||
'widgets/__init__.py',
|
||||
'widgets/header_row.py',
|
||||
'widgets/history_item.py',
|
||||
]
|
||||
|
||||
install_data(widgets_sources, install_dir: moduledir / 'widgets')
|
||||
86
src/models.py
Normal file
86
src/models.py
Normal file
@ -0,0 +1,86 @@
|
||||
# models.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from dataclasses import dataclass, asdict
|
||||
from typing import Dict, Optional
|
||||
|
||||
|
||||
@dataclass
|
||||
class HttpRequest:
|
||||
"""Represents an HTTP request."""
|
||||
method: str # "GET", "POST", "PUT", "DELETE"
|
||||
url: str
|
||||
headers: Dict[str, str] # Key-value pairs
|
||||
body: str # Raw text body
|
||||
|
||||
def to_dict(self):
|
||||
"""Convert to dictionary for JSON serialization."""
|
||||
return asdict(self)
|
||||
|
||||
@classmethod
|
||||
def from_dict(cls, data):
|
||||
"""Create instance from dictionary."""
|
||||
return cls(**data)
|
||||
|
||||
|
||||
@dataclass
|
||||
class HttpResponse:
|
||||
"""Represents an HTTP response."""
|
||||
status_code: int
|
||||
status_text: str # e.g., "OK"
|
||||
headers: str # Raw header text from HTTPie
|
||||
body: str # Raw body text
|
||||
response_time_ms: float
|
||||
|
||||
def to_dict(self):
|
||||
"""Convert to dictionary for JSON serialization."""
|
||||
return asdict(self)
|
||||
|
||||
@classmethod
|
||||
def from_dict(cls, data):
|
||||
"""Create instance from dictionary."""
|
||||
return cls(**data)
|
||||
|
||||
|
||||
@dataclass
|
||||
class HistoryEntry:
|
||||
"""Represents a request/response pair in history."""
|
||||
timestamp: str # ISO format datetime
|
||||
request: HttpRequest
|
||||
response: Optional[HttpResponse]
|
||||
error: Optional[str] # Error message if request failed
|
||||
|
||||
def to_dict(self):
|
||||
"""Convert to dictionary for JSON serialization."""
|
||||
return {
|
||||
'timestamp': self.timestamp,
|
||||
'request': self.request.to_dict(),
|
||||
'response': self.response.to_dict() if self.response else None,
|
||||
'error': self.error
|
||||
}
|
||||
|
||||
@classmethod
|
||||
def from_dict(cls, data):
|
||||
"""Create instance from dictionary."""
|
||||
return cls(
|
||||
timestamp=data['timestamp'],
|
||||
request=HttpRequest.from_dict(data['request']),
|
||||
response=HttpResponse.from_dict(data['response']) if data.get('response') else None,
|
||||
error=data.get('error')
|
||||
)
|
||||
9
src/roster.gresource.xml
Normal file
9
src/roster.gresource.xml
Normal file
@ -0,0 +1,9 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<gresources>
|
||||
<gresource prefix="/cz/vesp/roster">
|
||||
<file preprocess="xml-stripblanks">main-window.ui</file>
|
||||
<file preprocess="xml-stripblanks">shortcuts-dialog.ui</file>
|
||||
<file preprocess="xml-stripblanks">widgets/header-row.ui</file>
|
||||
<file preprocess="xml-stripblanks">widgets/history-item.ui</file>
|
||||
</gresource>
|
||||
</gresources>
|
||||
46
src/roster.in
Executable file
46
src/roster.in
Executable file
@ -0,0 +1,46 @@
|
||||
#!@PYTHON@
|
||||
|
||||
# roster.in
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
import os
|
||||
import sys
|
||||
import signal
|
||||
import locale
|
||||
import gettext
|
||||
|
||||
VERSION = '@VERSION@'
|
||||
pkgdatadir = '@pkgdatadir@'
|
||||
localedir = '@localedir@'
|
||||
|
||||
sys.path.insert(1, pkgdatadir)
|
||||
signal.signal(signal.SIGINT, signal.SIG_DFL)
|
||||
locale.bindtextdomain('roster', localedir)
|
||||
locale.textdomain('roster')
|
||||
gettext.install('roster', localedir)
|
||||
|
||||
if __name__ == '__main__':
|
||||
import gi
|
||||
|
||||
from gi.repository import Gio
|
||||
resource = Gio.Resource.load(os.path.join(pkgdatadir, 'roster.gresource'))
|
||||
resource._register()
|
||||
|
||||
from roster import main
|
||||
sys.exit(main.main(VERSION))
|
||||
22
src/shortcuts-dialog.ui
Normal file
22
src/shortcuts-dialog.ui
Normal file
@ -0,0 +1,22 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<interface>
|
||||
<object class="AdwShortcutsDialog" id="shortcuts_dialog">
|
||||
<child>
|
||||
<object class="AdwShortcutsSection">
|
||||
<property name="title" translatable="yes">Shortcuts</property>
|
||||
<child>
|
||||
<object class="AdwShortcutsItem">
|
||||
<property name="title" translatable="yes" context="shortcut window">Show Shortcuts</property>
|
||||
<property name="action-name">app.shortcuts</property>
|
||||
</object>
|
||||
</child>
|
||||
<child>
|
||||
<object class="AdwShortcutsItem">
|
||||
<property name="title" translatable="yes" context="shortcut window">Quit</property>
|
||||
<property name="action-name">app.quit</property>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</interface>
|
||||
23
src/widgets/__init__.py
Normal file
23
src/widgets/__init__.py
Normal file
@ -0,0 +1,23 @@
|
||||
# widgets/__init__.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from .header_row import HeaderRow
|
||||
from .history_item import HistoryItem
|
||||
|
||||
__all__ = ['HeaderRow', 'HistoryItem']
|
||||
37
src/widgets/header-row.ui
Normal file
37
src/widgets/header-row.ui
Normal file
@ -0,0 +1,37 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<interface>
|
||||
<requires lib="gtk" version="4.0"/>
|
||||
<template class="HeaderRow" parent="GtkBox">
|
||||
<property name="orientation">horizontal</property>
|
||||
<property name="spacing">6</property>
|
||||
<property name="margin-start">6</property>
|
||||
<property name="margin-end">6</property>
|
||||
<property name="margin-top">3</property>
|
||||
<property name="margin-bottom">3</property>
|
||||
|
||||
<child>
|
||||
<object class="GtkEntry" id="key_entry">
|
||||
<property name="placeholder-text">Header name</property>
|
||||
<property name="hexpand">True</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkEntry" id="value_entry">
|
||||
<property name="placeholder-text">Value</property>
|
||||
<property name="hexpand">True</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkButton" id="remove_button">
|
||||
<property name="icon-name">edit-delete-symbolic</property>
|
||||
<property name="tooltip-text">Remove header</property>
|
||||
<signal name="clicked" handler="on_remove_clicked" swapped="no"/>
|
||||
<style>
|
||||
<class name="flat"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
</template>
|
||||
</interface>
|
||||
53
src/widgets/header_row.py
Normal file
53
src/widgets/header_row.py
Normal file
@ -0,0 +1,53 @@
|
||||
# header_row.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from gi.repository import Gtk, GObject
|
||||
|
||||
|
||||
@Gtk.Template(resource_path='/cz/vesp/roster/widgets/header-row.ui')
|
||||
class HeaderRow(Gtk.Box):
|
||||
"""Widget for editing a single HTTP header key-value pair."""
|
||||
|
||||
__gtype_name__ = 'HeaderRow'
|
||||
|
||||
key_entry = Gtk.Template.Child()
|
||||
value_entry = Gtk.Template.Child()
|
||||
remove_button = Gtk.Template.Child()
|
||||
|
||||
# Signal emitted when remove button clicked
|
||||
__gsignals__ = {
|
||||
'remove-requested': (GObject.SIGNAL_RUN_FIRST, None, ())
|
||||
}
|
||||
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
@Gtk.Template.Callback()
|
||||
def on_remove_clicked(self, button):
|
||||
"""Handle remove button click."""
|
||||
self.emit('remove-requested')
|
||||
|
||||
def get_header(self):
|
||||
"""Return (key, value) tuple."""
|
||||
return (self.key_entry.get_text(), self.value_entry.get_text())
|
||||
|
||||
def set_header(self, key: str, value: str):
|
||||
"""Set header key and value."""
|
||||
self.key_entry.set_text(key)
|
||||
self.value_entry.set_text(value)
|
||||
183
src/widgets/history-item.ui
Normal file
183
src/widgets/history-item.ui
Normal file
@ -0,0 +1,183 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<interface>
|
||||
<requires lib="gtk" version="4.0"/>
|
||||
<requires lib="Adw" version="1.0"/>
|
||||
<template class="HistoryItem" parent="GtkBox">
|
||||
<property name="orientation">vertical</property>
|
||||
<property name="spacing">0</property>
|
||||
|
||||
<!-- Summary (always visible) -->
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">horizontal</property>
|
||||
<property name="spacing">12</property>
|
||||
<property name="margin-start">12</property>
|
||||
<property name="margin-end">12</property>
|
||||
<property name="margin-top">8</property>
|
||||
<property name="margin-bottom">8</property>
|
||||
|
||||
<!-- Expand/Collapse Icon -->
|
||||
<child>
|
||||
<object class="GtkImage" id="expand_icon">
|
||||
<property name="icon-name">go-next-symbolic</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Method Label -->
|
||||
<child>
|
||||
<object class="GtkLabel" id="method_label">
|
||||
<property name="label">GET</property>
|
||||
<property name="width-chars">6</property>
|
||||
<property name="xalign">0</property>
|
||||
<style>
|
||||
<class name="heading"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- URL Label -->
|
||||
<child>
|
||||
<object class="GtkLabel" id="url_label">
|
||||
<property name="label">URL</property>
|
||||
<property name="ellipsize">end</property>
|
||||
<property name="xalign">0</property>
|
||||
<property name="hexpand">True</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Status Label -->
|
||||
<child>
|
||||
<object class="GtkLabel" id="status_label">
|
||||
<property name="label">200 OK</property>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Timestamp Label -->
|
||||
<child>
|
||||
<object class="GtkLabel" id="timestamp_label">
|
||||
<property name="label">Timestamp</property>
|
||||
<style>
|
||||
<class name="dim-label"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Details (expandable) -->
|
||||
<child>
|
||||
<object class="GtkRevealer" id="details_revealer">
|
||||
<property name="reveal-child">False</property>
|
||||
<property name="transition-type">slide-down</property>
|
||||
|
||||
<child>
|
||||
<object class="GtkBox">
|
||||
<property name="orientation">vertical</property>
|
||||
<property name="spacing">6</property>
|
||||
<property name="margin-start">36</property>
|
||||
<property name="margin-end">12</property>
|
||||
<property name="margin-bottom">12</property>
|
||||
|
||||
<!-- Request Section -->
|
||||
<child>
|
||||
<object class="GtkLabel">
|
||||
<property name="label">Request:</property>
|
||||
<property name="xalign">0</property>
|
||||
<property name="margin-top">6</property>
|
||||
<style>
|
||||
<class name="heading"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkLabel" id="request_headers_label">
|
||||
<property name="label">Headers...</property>
|
||||
<property name="xalign">0</property>
|
||||
<property name="selectable">True</property>
|
||||
<property name="wrap">True</property>
|
||||
<property name="wrap-mode">word-char</property>
|
||||
<style>
|
||||
<class name="monospace"/>
|
||||
<class name="dim-label"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkLabel" id="request_body_label">
|
||||
<property name="label">Body...</property>
|
||||
<property name="xalign">0</property>
|
||||
<property name="selectable">True</property>
|
||||
<property name="wrap">True</property>
|
||||
<property name="wrap-mode">word-char</property>
|
||||
<style>
|
||||
<class name="monospace"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Response Section -->
|
||||
<child>
|
||||
<object class="GtkLabel">
|
||||
<property name="label">Response:</property>
|
||||
<property name="xalign">0</property>
|
||||
<property name="margin-top">6</property>
|
||||
<style>
|
||||
<class name="heading"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkLabel" id="response_headers_label">
|
||||
<property name="label">Headers...</property>
|
||||
<property name="xalign">0</property>
|
||||
<property name="selectable">True</property>
|
||||
<property name="wrap">True</property>
|
||||
<property name="wrap-mode">word-char</property>
|
||||
<style>
|
||||
<class name="monospace"/>
|
||||
<class name="dim-label"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<child>
|
||||
<object class="GtkLabel" id="response_body_label">
|
||||
<property name="label">Body...</property>
|
||||
<property name="xalign">0</property>
|
||||
<property name="selectable">True</property>
|
||||
<property name="wrap">True</property>
|
||||
<property name="wrap-mode">word-char</property>
|
||||
<style>
|
||||
<class name="monospace"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Load Button -->
|
||||
<child>
|
||||
<object class="GtkButton" id="load_button">
|
||||
<property name="label">Load Request</property>
|
||||
<property name="halign">start</property>
|
||||
<property name="margin-top">6</property>
|
||||
<signal name="clicked" handler="on_load_clicked" swapped="no"/>
|
||||
<style>
|
||||
<class name="suggested-action"/>
|
||||
</style>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
|
||||
<!-- Click gesture for toggling expansion -->
|
||||
<child>
|
||||
<object class="GtkGestureClick" id="click_gesture">
|
||||
<signal name="released" handler="on_clicked" swapped="no"/>
|
||||
</object>
|
||||
</child>
|
||||
</template>
|
||||
</interface>
|
||||
119
src/widgets/history_item.py
Normal file
119
src/widgets/history_item.py
Normal file
@ -0,0 +1,119 @@
|
||||
# history_item.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from gi.repository import Gtk, GObject
|
||||
from datetime import datetime
|
||||
|
||||
|
||||
@Gtk.Template(resource_path='/cz/vesp/roster/widgets/history-item.ui')
|
||||
class HistoryItem(Gtk.Box):
|
||||
"""Widget for displaying a history entry."""
|
||||
|
||||
__gtype_name__ = 'HistoryItem'
|
||||
|
||||
expand_icon = Gtk.Template.Child()
|
||||
details_revealer = Gtk.Template.Child()
|
||||
method_label = Gtk.Template.Child()
|
||||
url_label = Gtk.Template.Child()
|
||||
timestamp_label = Gtk.Template.Child()
|
||||
status_label = Gtk.Template.Child()
|
||||
request_headers_label = Gtk.Template.Child()
|
||||
request_body_label = Gtk.Template.Child()
|
||||
response_headers_label = Gtk.Template.Child()
|
||||
response_body_label = Gtk.Template.Child()
|
||||
load_button = Gtk.Template.Child()
|
||||
|
||||
__gsignals__ = {
|
||||
'load-requested': (GObject.SIGNAL_RUN_FIRST, None, ())
|
||||
}
|
||||
|
||||
def __init__(self, entry):
|
||||
super().__init__()
|
||||
self.entry = entry
|
||||
self.expanded = False
|
||||
self._populate_ui()
|
||||
|
||||
def _populate_ui(self):
|
||||
"""Populate UI with entry data."""
|
||||
# Summary
|
||||
self.method_label.set_text(self.entry.request.method)
|
||||
self.url_label.set_text(self.entry.request.url)
|
||||
|
||||
# Format timestamp
|
||||
try:
|
||||
dt = datetime.fromisoformat(self.entry.timestamp)
|
||||
timestamp_str = dt.strftime("%H:%M:%S")
|
||||
except:
|
||||
timestamp_str = self.entry.timestamp
|
||||
|
||||
self.timestamp_label.set_text(timestamp_str)
|
||||
|
||||
# Status
|
||||
if self.entry.response:
|
||||
status_text = f"{self.entry.response.status_code} {self.entry.response.status_text}"
|
||||
self.status_label.set_text(status_text)
|
||||
elif self.entry.error:
|
||||
self.status_label.set_text("Error")
|
||||
|
||||
# Details - Request headers
|
||||
headers_str = "\n".join([f"{k}: {v}" for k, v in self.entry.request.headers.items()])
|
||||
if not headers_str:
|
||||
headers_str = "(no headers)"
|
||||
self.request_headers_label.set_text(headers_str)
|
||||
|
||||
# Details - Request body
|
||||
body_str = self.entry.request.body if self.entry.request.body else "(empty)"
|
||||
# Truncate long bodies
|
||||
if len(body_str) > 200:
|
||||
body_str = body_str[:200] + "..."
|
||||
self.request_body_label.set_text(body_str)
|
||||
|
||||
# Details - Response
|
||||
if self.entry.response:
|
||||
self.response_headers_label.set_text(self.entry.response.headers)
|
||||
|
||||
response_body = self.entry.response.body if self.entry.response.body else "(empty)"
|
||||
# Truncate long bodies
|
||||
if len(response_body) > 200:
|
||||
response_body = response_body[:200] + "..."
|
||||
self.response_body_label.set_text(response_body)
|
||||
elif self.entry.error:
|
||||
self.response_headers_label.set_text("(error)")
|
||||
self.response_body_label.set_text(self.entry.error)
|
||||
|
||||
@Gtk.Template.Callback()
|
||||
def on_clicked(self, gesture, n_press, x, y):
|
||||
"""Toggle expansion when clicked."""
|
||||
self.toggle_expanded()
|
||||
|
||||
@Gtk.Template.Callback()
|
||||
def on_load_clicked(self, button):
|
||||
"""Emit load signal when load button clicked."""
|
||||
self.emit('load-requested')
|
||||
|
||||
def toggle_expanded(self):
|
||||
"""Toggle between collapsed and expanded view."""
|
||||
self.expanded = not self.expanded
|
||||
self.details_revealer.set_reveal_child(self.expanded)
|
||||
|
||||
# Update icon
|
||||
if self.expanded:
|
||||
self.expand_icon.set_from_icon_name('go-down-symbolic')
|
||||
else:
|
||||
self.expand_icon.set_from_icon_name('go-next-symbolic')
|
||||
419
src/window.py
Normal file
419
src/window.py
Normal file
@ -0,0 +1,419 @@
|
||||
# window.py
|
||||
#
|
||||
# Copyright 2025 Pavel Baksy
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
#
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from gi.repository import Adw, Gtk, GLib, Gio
|
||||
from .models import HttpRequest, HttpResponse, HistoryEntry
|
||||
from .http_client import HttpClient
|
||||
from .history_manager import HistoryManager
|
||||
from .widgets.header_row import HeaderRow
|
||||
from .widgets.history_item import HistoryItem
|
||||
from datetime import datetime
|
||||
import threading
|
||||
|
||||
|
||||
@Gtk.Template(resource_path='/cz/vesp/roster/main-window.ui')
|
||||
class RosterWindow(Adw.ApplicationWindow):
|
||||
__gtype_name__ = 'RosterWindow'
|
||||
|
||||
# Top bar widgets
|
||||
method_dropdown = Gtk.Template.Child()
|
||||
url_entry = Gtk.Template.Child()
|
||||
send_button = Gtk.Template.Child()
|
||||
|
||||
# Containers for tabs
|
||||
request_tabs_container = Gtk.Template.Child()
|
||||
response_tabs_container = Gtk.Template.Child()
|
||||
|
||||
# Status bar
|
||||
status_label = Gtk.Template.Child()
|
||||
time_label = Gtk.Template.Child()
|
||||
history_toggle_button = Gtk.Template.Child()
|
||||
|
||||
# History
|
||||
history_revealer = Gtk.Template.Child()
|
||||
history_listbox = Gtk.Template.Child()
|
||||
hide_history_button = Gtk.Template.Child()
|
||||
|
||||
def __init__(self, **kwargs):
|
||||
super().__init__(**kwargs)
|
||||
|
||||
self.http_client = HttpClient()
|
||||
self.history_manager = HistoryManager()
|
||||
|
||||
# Create window actions
|
||||
self._create_actions()
|
||||
|
||||
# Setup UI
|
||||
self._setup_method_dropdown()
|
||||
self._setup_request_tabs()
|
||||
self._setup_response_tabs()
|
||||
self._load_history()
|
||||
|
||||
# Add initial header row
|
||||
self._add_header_row()
|
||||
|
||||
def _create_actions(self):
|
||||
"""Create window-level actions."""
|
||||
action = Gio.SimpleAction.new("toggle-history", None)
|
||||
action.connect("activate", self.on_toggle_history)
|
||||
self.add_action(action)
|
||||
|
||||
def _setup_method_dropdown(self):
|
||||
"""Populate HTTP method dropdown."""
|
||||
methods = Gtk.StringList()
|
||||
methods.append("GET")
|
||||
methods.append("POST")
|
||||
methods.append("PUT")
|
||||
methods.append("DELETE")
|
||||
self.method_dropdown.set_model(methods)
|
||||
self.method_dropdown.set_selected(0) # Default to GET
|
||||
|
||||
def _setup_request_tabs(self):
|
||||
"""Create request tabs programmatically."""
|
||||
# Create tab view
|
||||
self.request_tab_view = Adw.TabView()
|
||||
|
||||
# Create tab bar to show tabs
|
||||
request_tab_bar = Adw.TabBar()
|
||||
request_tab_bar.set_view(self.request_tab_view)
|
||||
request_tab_bar.set_autohide(False)
|
||||
|
||||
# Add tab bar and view to container
|
||||
self.request_tabs_container.append(request_tab_bar)
|
||||
self.request_tabs_container.append(self.request_tab_view)
|
||||
|
||||
# Headers tab
|
||||
headers_scroll = Gtk.ScrolledWindow()
|
||||
self.headers_listbox = Gtk.ListBox()
|
||||
self.headers_listbox.add_css_class("boxed-list")
|
||||
headers_scroll.set_child(self.headers_listbox)
|
||||
|
||||
# Add header button container
|
||||
headers_box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=6)
|
||||
headers_box.set_margin_start(12)
|
||||
headers_box.set_margin_end(12)
|
||||
headers_box.set_margin_top(12)
|
||||
headers_box.set_margin_bottom(12)
|
||||
headers_box.append(headers_scroll)
|
||||
|
||||
self.add_header_button = Gtk.Button(label="Add Header")
|
||||
self.add_header_button.connect("clicked", self.on_add_header_clicked)
|
||||
headers_box.append(self.add_header_button)
|
||||
|
||||
headers_page = self.request_tab_view.append(headers_box)
|
||||
headers_page.set_title("Headers")
|
||||
|
||||
# Body tab
|
||||
body_scroll = Gtk.ScrolledWindow()
|
||||
body_scroll.set_vexpand(True)
|
||||
self.body_textview = Gtk.TextView()
|
||||
self.body_textview.set_monospace(True)
|
||||
self.body_textview.set_left_margin(12)
|
||||
self.body_textview.set_right_margin(12)
|
||||
self.body_textview.set_top_margin(12)
|
||||
self.body_textview.set_bottom_margin(12)
|
||||
body_scroll.set_child(self.body_textview)
|
||||
|
||||
body_page = self.request_tab_view.append(body_scroll)
|
||||
body_page.set_title("Body")
|
||||
|
||||
def _setup_response_tabs(self):
|
||||
"""Create response tabs programmatically."""
|
||||
# Create tab view
|
||||
self.response_tab_view = Adw.TabView()
|
||||
|
||||
# Create tab bar to show tabs
|
||||
response_tab_bar = Adw.TabBar()
|
||||
response_tab_bar.set_view(self.response_tab_view)
|
||||
response_tab_bar.set_autohide(False)
|
||||
|
||||
# Add tab bar and view to container
|
||||
self.response_tabs_container.append(response_tab_bar)
|
||||
self.response_tabs_container.append(self.response_tab_view)
|
||||
|
||||
# Response Headers tab
|
||||
headers_scroll = Gtk.ScrolledWindow()
|
||||
headers_scroll.set_vexpand(True)
|
||||
self.response_headers_textview = Gtk.TextView()
|
||||
self.response_headers_textview.set_editable(False)
|
||||
self.response_headers_textview.set_monospace(True)
|
||||
self.response_headers_textview.set_left_margin(12)
|
||||
self.response_headers_textview.set_right_margin(12)
|
||||
self.response_headers_textview.set_top_margin(12)
|
||||
self.response_headers_textview.set_bottom_margin(12)
|
||||
headers_scroll.set_child(self.response_headers_textview)
|
||||
|
||||
headers_page = self.response_tab_view.append(headers_scroll)
|
||||
headers_page.set_title("Headers")
|
||||
|
||||
# Response Body tab
|
||||
body_scroll = Gtk.ScrolledWindow()
|
||||
body_scroll.set_vexpand(True)
|
||||
self.response_body_textview = Gtk.TextView()
|
||||
self.response_body_textview.set_editable(False)
|
||||
self.response_body_textview.set_monospace(True)
|
||||
self.response_body_textview.set_left_margin(12)
|
||||
self.response_body_textview.set_right_margin(12)
|
||||
self.response_body_textview.set_top_margin(12)
|
||||
self.response_body_textview.set_bottom_margin(12)
|
||||
body_scroll.set_child(self.response_body_textview)
|
||||
|
||||
body_page = self.response_tab_view.append(body_scroll)
|
||||
body_page.set_title("Body")
|
||||
|
||||
@Gtk.Template.Callback()
|
||||
def on_send_clicked(self, button):
|
||||
"""Handle Send button click."""
|
||||
# Validate URL
|
||||
url = self.url_entry.get_text().strip()
|
||||
if not url:
|
||||
self._show_toast("Please enter a URL")
|
||||
return
|
||||
|
||||
# Build request from UI
|
||||
request = self._build_request_from_ui()
|
||||
|
||||
# Disable send button during request
|
||||
self.send_button.set_sensitive(False)
|
||||
self.send_button.set_label("Sending...")
|
||||
self.status_label.set_text("Sending...")
|
||||
self.time_label.set_text("")
|
||||
|
||||
# Execute in thread to avoid blocking UI
|
||||
thread = threading.Thread(target=self._execute_request_thread, args=(request,))
|
||||
thread.daemon = True
|
||||
thread.start()
|
||||
|
||||
def _execute_request_thread(self, request):
|
||||
"""Execute request in background thread."""
|
||||
response, error = self.http_client.execute_request(request)
|
||||
|
||||
# Update UI in main thread
|
||||
GLib.idle_add(self._handle_response, request, response, error)
|
||||
|
||||
def _handle_response(self, request, response, error):
|
||||
"""Handle response in main thread."""
|
||||
# Re-enable send button
|
||||
self.send_button.set_sensitive(True)
|
||||
self.send_button.set_label("Send")
|
||||
|
||||
# Create history entry
|
||||
entry = HistoryEntry(
|
||||
timestamp=datetime.now().isoformat(),
|
||||
request=request,
|
||||
response=response,
|
||||
error=error
|
||||
)
|
||||
|
||||
# Save to history
|
||||
self.history_manager.add_entry(entry)
|
||||
|
||||
# Update UI
|
||||
if response:
|
||||
self._display_response(response)
|
||||
else:
|
||||
self._display_error(error)
|
||||
|
||||
# Refresh history list
|
||||
self._load_history()
|
||||
|
||||
def _build_request_from_ui(self):
|
||||
"""Build HttpRequest from UI inputs."""
|
||||
method = self.method_dropdown.get_selected_item().get_string()
|
||||
url = self.url_entry.get_text().strip()
|
||||
|
||||
# Collect headers
|
||||
headers = {}
|
||||
child = self.headers_listbox.get_first_child()
|
||||
while child is not None:
|
||||
# In GTK4, ListBox wraps children in ListBoxRow, so we need to get the actual child
|
||||
if isinstance(child, Gtk.ListBoxRow):
|
||||
header_row = child.get_child()
|
||||
if isinstance(header_row, HeaderRow):
|
||||
key, value = header_row.get_header()
|
||||
if key and value:
|
||||
headers[key] = value
|
||||
child = child.get_next_sibling()
|
||||
|
||||
# Get body
|
||||
buffer = self.body_textview.get_buffer()
|
||||
body = buffer.get_text(buffer.get_start_iter(), buffer.get_end_iter(), False)
|
||||
|
||||
return HttpRequest(method=method, url=url, headers=headers, body=body)
|
||||
|
||||
def _display_response(self, response):
|
||||
"""Display response in UI."""
|
||||
# Update status
|
||||
status_text = f"{response.status_code} {response.status_text}"
|
||||
self.status_label.set_text(status_text)
|
||||
|
||||
# Update time
|
||||
time_text = f"{response.response_time_ms:.0f} ms"
|
||||
self.time_label.set_text(time_text)
|
||||
|
||||
# Update response headers
|
||||
buffer = self.response_headers_textview.get_buffer()
|
||||
buffer.set_text(response.headers)
|
||||
|
||||
# Update response body
|
||||
buffer = self.response_body_textview.get_buffer()
|
||||
buffer.set_text(response.body)
|
||||
|
||||
def _display_error(self, error):
|
||||
"""Display error in UI."""
|
||||
self.status_label.set_text("Error")
|
||||
self.time_label.set_text("")
|
||||
|
||||
# Clear response headers
|
||||
buffer = self.response_headers_textview.get_buffer()
|
||||
buffer.set_text("")
|
||||
|
||||
# Display error in body
|
||||
buffer = self.response_body_textview.get_buffer()
|
||||
buffer.set_text(error)
|
||||
|
||||
# Show toast
|
||||
self._show_toast(f"Request failed: {error}")
|
||||
|
||||
def on_add_header_clicked(self, button):
|
||||
"""Add new header row."""
|
||||
self._add_header_row()
|
||||
|
||||
def _add_header_row(self, key='', value=''):
|
||||
"""Add a header row to the list."""
|
||||
row = HeaderRow()
|
||||
row.set_header(key, value)
|
||||
row.connect('remove-requested', self._on_header_remove)
|
||||
self.headers_listbox.append(row)
|
||||
|
||||
def _on_header_remove(self, header_row):
|
||||
"""Handle header row removal."""
|
||||
# In GTK4, we need to remove the parent ListBoxRow, not the HeaderRow itself
|
||||
parent = header_row.get_parent()
|
||||
if parent:
|
||||
self.headers_listbox.remove(parent)
|
||||
|
||||
def _load_history(self):
|
||||
"""Load history from file and populate list."""
|
||||
# Clear existing history items
|
||||
while True:
|
||||
child = self.history_listbox.get_first_child()
|
||||
if child is None:
|
||||
break
|
||||
self.history_listbox.remove(child)
|
||||
|
||||
# Load and display history
|
||||
entries = self.history_manager.load_history()
|
||||
for entry in entries:
|
||||
item = HistoryItem(entry)
|
||||
item.connect('load-requested', self._on_history_load_requested, entry)
|
||||
self.history_listbox.append(item)
|
||||
|
||||
def _on_history_load_requested(self, widget, entry):
|
||||
"""Handle load request from history item."""
|
||||
# Show warning dialog
|
||||
dialog = Adw.AlertDialog()
|
||||
dialog.set_heading("Load Request?")
|
||||
dialog.set_body("This will replace your current request. Any unsaved changes will be lost.")
|
||||
dialog.add_response("cancel", "Cancel")
|
||||
dialog.add_response("load", "Load")
|
||||
dialog.set_response_appearance("load", Adw.ResponseAppearance.SUGGESTED)
|
||||
dialog.set_default_response("cancel")
|
||||
dialog.set_close_response("cancel")
|
||||
|
||||
dialog.connect("response", self._on_load_dialog_response, entry)
|
||||
dialog.present(self)
|
||||
|
||||
def _on_load_dialog_response(self, dialog, response, entry):
|
||||
"""Handle load dialog response."""
|
||||
if response == "load":
|
||||
self._load_request_from_entry(entry)
|
||||
|
||||
def _load_request_from_entry(self, entry):
|
||||
"""Load request from history entry into UI."""
|
||||
request = entry.request
|
||||
|
||||
# Set method
|
||||
methods = ["GET", "POST", "PUT", "DELETE"]
|
||||
if request.method in methods:
|
||||
self.method_dropdown.set_selected(methods.index(request.method))
|
||||
|
||||
# Set URL
|
||||
self.url_entry.set_text(request.url)
|
||||
|
||||
# Clear existing headers
|
||||
while True:
|
||||
child = self.headers_listbox.get_first_child()
|
||||
if child is None:
|
||||
break
|
||||
self.headers_listbox.remove(child)
|
||||
|
||||
# Set headers
|
||||
for key, value in request.headers.items():
|
||||
self._add_header_row(key, value)
|
||||
|
||||
# Add one empty row if no headers
|
||||
if not request.headers:
|
||||
self._add_header_row()
|
||||
|
||||
# Set body
|
||||
buffer = self.body_textview.get_buffer()
|
||||
buffer.set_text(request.body)
|
||||
|
||||
# Switch to headers tab
|
||||
self.request_tab_view.set_selected_page(self.request_tab_view.get_nth_page(0))
|
||||
|
||||
self._show_toast("Request loaded from history")
|
||||
|
||||
@Gtk.Template.Callback()
|
||||
def on_toggle_history(self, *args):
|
||||
"""Toggle history panel visibility (from menu)."""
|
||||
current = self.history_revealer.get_reveal_child()
|
||||
self.history_revealer.set_reveal_child(not current)
|
||||
|
||||
# Update toggle button state
|
||||
self.history_toggle_button.set_active(not current)
|
||||
|
||||
# Update hide button icon
|
||||
if not current:
|
||||
self.hide_history_button.set_icon_name("go-down-symbolic")
|
||||
else:
|
||||
self.hide_history_button.set_icon_name("go-up-symbolic")
|
||||
|
||||
@Gtk.Template.Callback()
|
||||
def on_history_toggle_button_toggled(self, button):
|
||||
"""Toggle history panel from bottom button."""
|
||||
active = button.get_active()
|
||||
self.history_revealer.set_reveal_child(active)
|
||||
|
||||
# Update hide button icon
|
||||
if active:
|
||||
self.hide_history_button.set_icon_name("go-down-symbolic")
|
||||
else:
|
||||
self.hide_history_button.set_icon_name("go-up-symbolic")
|
||||
|
||||
def _show_toast(self, message):
|
||||
"""Show a toast notification."""
|
||||
toast = Adw.Toast()
|
||||
toast.set_title(message)
|
||||
toast.set_timeout(3)
|
||||
|
||||
# Get the toast overlay (we need to add one)
|
||||
# For now, just print to console
|
||||
print(f"Toast: {message}")
|
||||
50
src/window.ui
Normal file
50
src/window.ui
Normal file
@ -0,0 +1,50 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<interface>
|
||||
<requires lib="gtk" version="4.0"/>
|
||||
<requires lib="Adw" version="1.0"/>
|
||||
<template class="RosterWindow" parent="AdwApplicationWindow">
|
||||
<property name="title" translatable="yes">Roster</property>
|
||||
<property name="default-width">800</property>
|
||||
<property name="default-height">600</property>
|
||||
<property name="content">
|
||||
<object class="AdwToolbarView">
|
||||
<child type="top">
|
||||
<object class="AdwHeaderBar">
|
||||
<child type="end">
|
||||
<object class="GtkMenuButton">
|
||||
<property name="primary">True</property>
|
||||
<property name="icon-name">open-menu-symbolic</property>
|
||||
<property name="tooltip-text" translatable="yes">Main Menu</property>
|
||||
<property name="menu-model">primary_menu</property>
|
||||
</object>
|
||||
</child>
|
||||
</object>
|
||||
</child>
|
||||
<property name="content">
|
||||
<object class="GtkLabel" id="label">
|
||||
<property name="label" translatable="yes">Hello, World!</property>
|
||||
<style>
|
||||
<class name="title-1"/>
|
||||
</style>
|
||||
</object>
|
||||
</property>
|
||||
</object>
|
||||
</property>
|
||||
</template>
|
||||
<menu id="primary_menu">
|
||||
<section>
|
||||
<item>
|
||||
<attribute name="label" translatable="yes">_Preferences</attribute>
|
||||
<attribute name="action">app.preferences</attribute>
|
||||
</item>
|
||||
<item>
|
||||
<attribute name="label" translatable="yes">_Keyboard Shortcuts</attribute>
|
||||
<attribute name="action">app.shortcuts</attribute>
|
||||
</item>
|
||||
<item>
|
||||
<attribute name="label" translatable="yes">_About Roster</attribute>
|
||||
<attribute name="action">app.about</attribute>
|
||||
</item>
|
||||
</section>
|
||||
</menu>
|
||||
</interface>
|
||||
Loading…
x
Reference in New Issue
Block a user