Compare commits
271 Commits
Author | SHA1 | Date | |
---|---|---|---|
b98408dd32 | |||
36054434a5 | |||
d9e41c27ef | |||
8e8f8e4706 | |||
edb1a6f9e1 | |||
![]() |
33723db314 | ||
4bc440e906 | |||
![]() |
27b248d851 | ||
e3c2e024bb | |||
![]() |
b69610d7f3 | ||
92a768d6f1 | |||
![]() |
5ecd7d754d | ||
![]() |
e3161f64f6 | ||
![]() |
5b0b091ab6 | ||
![]() |
4fdf28fb79 | ||
49213a1cf0 | |||
e30db231f6 | |||
6609c4867f | |||
![]() |
84c5e8bca6 | ||
1a25749315 | |||
![]() |
3852f5995d | ||
9093b7a16e | |||
afead1c2c0 | |||
1b2572f381 | |||
![]() |
27ca6b3f73 | ||
![]() |
f157648d1e | ||
![]() |
d4baaeccfc | ||
![]() |
d412b66c6e | ||
![]() |
74ae511f92 | ||
![]() |
a3fbafe4ae | ||
![]() |
8a45fbe9ed | ||
![]() |
01c7929ee3 | ||
![]() |
2261af17d7 | ||
![]() |
41f34c3992 | ||
![]() |
36b1cd2dde | ||
![]() |
80374b3ec6 | ||
![]() |
bc00d1e45f | ||
![]() |
6ddb507247 | ||
![]() |
b0f3b8f3d0 | ||
![]() |
06e7191056 | ||
![]() |
997131ba64 | ||
![]() |
ef9f412e5f | ||
![]() |
7b69d41c60 | ||
![]() |
552947f507 | ||
![]() |
f69f2df63c | ||
![]() |
aa270067f6 | ||
![]() |
28977313da | ||
![]() |
cfe35d7795 | ||
![]() |
4abbe332db | ||
![]() |
3104abd4ad | ||
![]() |
7714f7b503 | ||
![]() |
9d06f10a9b | ||
![]() |
8c27f7683b | ||
![]() |
bf1a491789 | ||
![]() |
701cb540e5 | ||
![]() |
0f18a0349a | ||
![]() |
205a152350 | ||
![]() |
dae1a56691 | ||
![]() |
39181f42cf | ||
![]() |
3d706ad1a7 | ||
![]() |
a31b77e5bc | ||
![]() |
b417643415 | ||
![]() |
dac2b4ff16 | ||
![]() |
2d12b0c747 | ||
![]() |
5fe26edaa6 | ||
![]() |
a736cfd548 | ||
![]() |
5af4214daa | ||
![]() |
98ef7b329a | ||
![]() |
b0e1962fa7 | ||
![]() |
83684e8882 | ||
![]() |
1ed7d077a9 | ||
![]() |
ba4b36618f | ||
![]() |
af647a4bad | ||
![]() |
b719530ef0 | ||
![]() |
530d61d67e | ||
![]() |
3d76bcb48c | ||
![]() |
70ce2f1f3c | ||
![]() |
21a190a1aa | ||
![]() |
a88d222709 | ||
![]() |
b2defceae5 | ||
![]() |
8d7565ffc4 | ||
![]() |
9960b25dff | ||
![]() |
411c116deb | ||
![]() |
64e007f1fd | ||
![]() |
c62fc365db | ||
![]() |
e7e630bd5d | ||
![]() |
7d77175fbb | ||
![]() |
f033553972 | ||
![]() |
ae1ed044ba | ||
![]() |
35d601c7a7 | ||
![]() |
f89aeec374 | ||
![]() |
42c3b4b42e | ||
![]() |
322fc31396 | ||
![]() |
6e4d5cfddd | ||
![]() |
d81485a8bf | ||
![]() |
a9260d956c | ||
![]() |
025bb560bc | ||
![]() |
cdd1488e13 | ||
![]() |
b8b2644969 | ||
![]() |
09b5c0a5d0 | ||
![]() |
1ed66f3aa2 | ||
![]() |
3b4d5d1c3f | ||
![]() |
f98fd5a7ad | ||
![]() |
b961109623 | ||
![]() |
8f0d1023a4 | ||
![]() |
79773ba95c | ||
![]() |
ff3f02fb8c | ||
![]() |
0af2b20b3a | ||
![]() |
b026fe357e | ||
![]() |
ade36c013b | ||
![]() |
54a6a7f590 | ||
![]() |
20728e4a0f | ||
![]() |
afa456f6b3 | ||
![]() |
3fbfb77430 | ||
![]() |
051bcb62c6 | ||
![]() |
d2c57eed6d | ||
![]() |
f2f38c0e0d | ||
![]() |
4046bdbc6a | ||
![]() |
c83cc206fd | ||
![]() |
41ac14fbd3 | ||
![]() |
e4c1cad1a7 | ||
![]() |
863c1ac27a | ||
![]() |
c85eba617c | ||
![]() |
dcb7b7c309 | ||
![]() |
3cee186663 | ||
![]() |
06afe507f6 | ||
![]() |
4268d4d770 | ||
![]() |
d61dae5ab4 | ||
![]() |
b5f39c1d53 | ||
![]() |
767a61b86e | ||
![]() |
0d3c0a2233 | ||
![]() |
0431103a1d | ||
![]() |
633e4e642c | ||
![]() |
77dc82a381 | ||
![]() |
44a63d4aaa | ||
![]() |
5b3297f916 | ||
![]() |
41413d4712 | ||
![]() |
fd2e5ce71e | ||
![]() |
e9ddd7d3b8 | ||
![]() |
da6795f002 | ||
![]() |
f497bb4b44 | ||
![]() |
4aea85649f | ||
![]() |
c56672dd04 | ||
![]() |
c2dc1cd65f | ||
![]() |
555783af4c | ||
![]() |
b9ffae25e3 | ||
![]() |
0f89936e01 | ||
![]() |
6ecccb3fe1 | ||
![]() |
2e003b56ea | ||
![]() |
a705418b6f | ||
![]() |
9767eb9d7f | ||
![]() |
450b15944f | ||
![]() |
1fa7cb2afa | ||
![]() |
d2415ff5f9 | ||
![]() |
cdbe8d73f6 | ||
![]() |
38ae86f315 | ||
![]() |
100201d23c | ||
![]() |
db7a7f38f7 | ||
![]() |
677ad5c54b | ||
![]() |
b8e3cd5d64 | ||
![]() |
dead1197bd | ||
![]() |
a2455daa6c | ||
![]() |
ddff1fc455 | ||
![]() |
82f2bd6231 | ||
![]() |
cc7c13e17f | ||
![]() |
04a64e1312 | ||
![]() |
bfecd9ebc6 | ||
![]() |
6fb8ceb501 | ||
![]() |
8124d101ad | ||
![]() |
de8602847e | ||
![]() |
9d1ca1986d | ||
![]() |
a0312ff604 | ||
![]() |
6f04513faa | ||
![]() |
a747f55423 | ||
![]() |
d27e6569a0 | ||
![]() |
135ed0bf8b | ||
![]() |
67ac27c054 | ||
![]() |
4e123d747a | ||
![]() |
9df470acfc | ||
![]() |
af8edc72f4 | ||
![]() |
2b2ae66b8d | ||
![]() |
0a95c5dff3 | ||
![]() |
56e1d80cf4 | ||
![]() |
65432fd317 | ||
![]() |
2ec21f5eba | ||
![]() |
92e82c203f | ||
![]() |
272bc5fe89 | ||
![]() |
e8d5efd3b3 | ||
![]() |
8b1a0a5b8e | ||
![]() |
293b05fec6 | ||
![]() |
37ba495354 | ||
![]() |
a2e689a0dc | ||
![]() |
2ea119fb31 | ||
![]() |
162f59d220 | ||
![]() |
880c5e722e | ||
![]() |
82efa28703 | ||
![]() |
c091a90cea | ||
![]() |
e4c014c1a8 | ||
![]() |
fe4704d0f6 | ||
![]() |
9ea797b00b | ||
![]() |
683f202191 | ||
![]() |
1cf6e2743b | ||
![]() |
2375aefebb | ||
![]() |
1e04f66a11 | ||
![]() |
f8bd483139 | ||
![]() |
3548ce84c1 | ||
![]() |
90b68f9987 | ||
![]() |
44e895ee08 | ||
![]() |
8032547fd0 | ||
![]() |
a34dc35f0f | ||
![]() |
68a5f0dd4b | ||
![]() |
654f8d0761 | ||
![]() |
942ff8bf4a | ||
![]() |
e2c4392c6d | ||
![]() |
7e771b1cfc | ||
![]() |
dd799cf5c0 | ||
![]() |
b8be28619a | ||
![]() |
ca012a1c80 | ||
![]() |
50de42ec9f | ||
![]() |
38c9dd50ff | ||
![]() |
27e4e3444a | ||
![]() |
a8935da416 | ||
![]() |
beb56c527b | ||
![]() |
5e3caedaf6 | ||
![]() |
0be17716f5 | ||
![]() |
dda5fbc330 | ||
![]() |
ffc16c6814 | ||
![]() |
705371d7ee | ||
![]() |
7c7810836e | ||
![]() |
af9c039dcb | ||
![]() |
7325835c3f | ||
![]() |
e7aa72a7ab | ||
![]() |
35c48de638 | ||
![]() |
ae50ed980f | ||
![]() |
1d6f0445f7 | ||
![]() |
799ef63b81 | ||
![]() |
6b22a0c03a | ||
![]() |
2d54638f49 | ||
![]() |
8adec835f6 | ||
![]() |
a34654a9a9 | ||
![]() |
b7c8108ad5 | ||
![]() |
c34c3c3654 | ||
![]() |
88988dd3e1 | ||
![]() |
351b98189f | ||
![]() |
e6a049c214 | ||
![]() |
51d59cc90b | ||
![]() |
4bff6efe97 | ||
![]() |
c514dc7a24 | ||
![]() |
24754edb6a | ||
![]() |
79fbb76524 | ||
![]() |
bbe61e4443 | ||
![]() |
b85aede966 | ||
![]() |
c66ca6e44f | ||
![]() |
a192024fef | ||
![]() |
918df9098a | ||
![]() |
c9271bfa36 | ||
![]() |
fa162c818b | ||
![]() |
ff976e3ec2 | ||
![]() |
17c5651305 | ||
![]() |
7c7b049ba1 | ||
![]() |
0823fe8b3d | ||
![]() |
e787fd0fb5 | ||
![]() |
650497bff4 | ||
![]() |
2f25bd30b0 | ||
![]() |
d58157a450 | ||
![]() |
9d049bd041 | ||
![]() |
988222b711 | ||
![]() |
f42bd3036f | ||
![]() |
297239fb5c | ||
![]() |
dbd46d44aa | ||
![]() |
254f4d6e92 |
4
.github/FUNDING.yml
vendored
Normal file
4
.github/FUNDING.yml
vendored
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
patreon: swordfish90
|
||||||
|
custom: ['https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=flscogna%40gmail.com&item_name=Support+CRT¤cy_code=EUR&source=url']
|
98
.github/workflows/appimage.yml
vendored
Normal file
98
.github/workflows/appimage.yml
vendored
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
name: "ci"
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags: "**"
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
defaults:
|
||||||
|
run:
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
appimage:
|
||||||
|
runs-on: ubuntu-18.04
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
submodules: true
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
sudo add-apt-repository -y ppa:beineri/opt-qt-5.15.2-bionic
|
||||||
|
sudo apt-get update -qq
|
||||||
|
sudo apt-get install -y \
|
||||||
|
build-essential make wget libgl1-mesa-dev \
|
||||||
|
qt515declarative qt515graphicaleffects \
|
||||||
|
qt515quickcontrols qt515quickcontrols2
|
||||||
|
|
||||||
|
- name: Download QT appimage builder
|
||||||
|
run: |
|
||||||
|
wget -c -O linuxdeployqt.AppImage \
|
||||||
|
https://github.com/probonopd/linuxdeployqt/releases/download/continuous/linuxdeployqt-continuous-x86_64.AppImage
|
||||||
|
chmod a+x linuxdeployqt.AppImage
|
||||||
|
|
||||||
|
- name: Build project
|
||||||
|
run: |
|
||||||
|
source /opt/qt*/bin/qt*-env.sh && \
|
||||||
|
qmake -v && \
|
||||||
|
qmake CONFIG+=release PREFIX=/usr && \
|
||||||
|
make -j$(nproc)
|
||||||
|
|
||||||
|
- name: Install to appdir
|
||||||
|
run: |
|
||||||
|
source /opt/qt*/bin/qt*-env.sh && \
|
||||||
|
make INSTALL_ROOT=appdir -j$(nproc) install
|
||||||
|
|
||||||
|
- name: Extract version number
|
||||||
|
run: |
|
||||||
|
# Extract version for linuxdeployqt to name the file. Use the tag as
|
||||||
|
# release name but remove prefix.
|
||||||
|
echo "VERSION=$(echo '${{ github.ref }}' | sed 's;.*/;;')" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Build appimage directory
|
||||||
|
run: |
|
||||||
|
mkdir -p \
|
||||||
|
appdir/usr/bin \
|
||||||
|
appdir/usr/lib \
|
||||||
|
appdir/usr/share/applications \
|
||||||
|
appdir/usr/share/metainfo \
|
||||||
|
appdir/usr/share/icons/hicolor/128x128/apps
|
||||||
|
|
||||||
|
cp cool-retro-term appdir/usr/bin/
|
||||||
|
cp cool-retro-term.desktop appdir/usr/share/applications/
|
||||||
|
cp packaging/appdata/cool-retro-term.appdata.xml appdir/usr/share/metainfo/
|
||||||
|
cp app/icons/128x128/cool-retro-term.png appdir/usr/share/icons/hicolor/128x128/apps/
|
||||||
|
cp -r ./app/qml appdir/usr/
|
||||||
|
# Workaround for https://github.com/probonopd/linuxdeployqt/issues/78
|
||||||
|
cp -r ./qmltermwidget/QMLTermWidget appdir/usr/qml/
|
||||||
|
|
||||||
|
find appdir | sort
|
||||||
|
|
||||||
|
- name: Build appimage
|
||||||
|
run: |
|
||||||
|
source /opt/qt*/bin/qt*-env.sh && \
|
||||||
|
./linuxdeployqt.AppImage appdir/usr/share/applications/cool-retro-term.desktop \
|
||||||
|
-verbose=1 -appimage \
|
||||||
|
-qmldir=./app/qml/ \
|
||||||
|
-qmldir=./qmltermwidget/
|
||||||
|
env:
|
||||||
|
# Unset environment variables
|
||||||
|
QTDIR:
|
||||||
|
QT_PLUGIN_PATH:
|
||||||
|
LD_LIBRARY_PATH:
|
||||||
|
|
||||||
|
- name: Upload release
|
||||||
|
uses: softprops/action-gh-release@v1
|
||||||
|
with:
|
||||||
|
body: appimage release
|
||||||
|
files: ./**/Cool_Retro_Term-*-x86_64.AppImage
|
||||||
|
|
||||||
|
- name: Clean up
|
||||||
|
if: always()
|
||||||
|
run: |
|
||||||
|
find appdir -executable -type f -exec ldd {} \; | grep " => /usr" | cut -d " " -f 2-3 | sort | uniq
|
||||||
|
make clean
|
||||||
|
rm -rf appdir
|
4
.gitignore
vendored
4
.gitignore
vendored
@ -18,6 +18,7 @@
|
|||||||
*.pro.user.*
|
*.pro.user.*
|
||||||
*.moc
|
*.moc
|
||||||
moc_*.cpp
|
moc_*.cpp
|
||||||
|
moc_*.h
|
||||||
qrc_*.cpp
|
qrc_*.cpp
|
||||||
ui_*.h
|
ui_*.h
|
||||||
Makefile*
|
Makefile*
|
||||||
@ -47,3 +48,6 @@ cool-retro-term
|
|||||||
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
*.app
|
*.app
|
||||||
|
|
||||||
|
# IntelliJ
|
||||||
|
.idea
|
||||||
|
1
.gitmodules
vendored
1
.gitmodules
vendored
@ -1,3 +1,4 @@
|
|||||||
[submodule "qmltermwidget"]
|
[submodule "qmltermwidget"]
|
||||||
path = qmltermwidget
|
path = qmltermwidget
|
||||||
url = https://github.com/Swordfish90/qmltermwidget
|
url = https://github.com/Swordfish90/qmltermwidget
|
||||||
|
branch = unstable
|
||||||
|
37
.travis.yml
Normal file
37
.travis.yml
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
sudo: required
|
||||||
|
dist: trusty
|
||||||
|
language: c++
|
||||||
|
|
||||||
|
install:
|
||||||
|
- sudo add-apt-repository -y ppa:beineri/opt-qt58-trusty
|
||||||
|
- sudo apt-get update -qq
|
||||||
|
- sudo apt-get -y install build-essential qt58declarative qt58graphicaleffects qt58quickcontrols libgl1-mesa-dev
|
||||||
|
- source /opt/qt*/bin/qt*-env.sh
|
||||||
|
|
||||||
|
script:
|
||||||
|
- qmake CONFIG+=release PREFIX=/usr
|
||||||
|
- make -j$(nproc)
|
||||||
|
- mkdir -p appdir/usr/share/metainfo appdir/usr/bin
|
||||||
|
- cp packaging/appdata/cool-retro-term.appdata.xml appdir/usr/share/metainfo/
|
||||||
|
- cp cool-retro-term appdir/usr/bin/
|
||||||
|
- cp ./cool-retro-term.desktop appdir/
|
||||||
|
- cp ./app/icons/128x128/cool-retro-term.png appdir/
|
||||||
|
- cp -r ./app/qml appdir/usr/
|
||||||
|
- cp -r ./qmltermwidget/QMLTermWidget appdir/usr/qml/ # Workaround for https://github.com/probonopd/linuxdeployqt/issues/78
|
||||||
|
- wget -c https://github.com/probonopd/linuxdeployqt/releases/download/continuous/linuxdeployqt-continuous-x86_64.AppImage
|
||||||
|
- chmod a+x linuxdeployqt-*.AppImage
|
||||||
|
- unset QTDIR; unset QT_PLUGIN_PATH ; unset LD_LIBRARY_PATH
|
||||||
|
- export VERSION=$(git rev-parse --short HEAD) # linuxdeployqt uses this for naming the file
|
||||||
|
- ./linuxdeployqt-*.AppImage appdir/usr/bin/cool-retro-term -qmldir=./app/qml/ -qmldir=./qmltermwidget/ # -verbose=3 2>&1 | grep "path:" -C 3
|
||||||
|
- ./linuxdeployqt-*.AppImage appdir/usr/bin/cool-retro-term -qmldir=./app/qml/ -qmldir=./qmltermwidget/ -verbose=2 -appimage
|
||||||
|
|
||||||
|
after_success:
|
||||||
|
- find appdir -executable -type f -exec ldd {} \; | grep " => /usr" | cut -d " " -f 2-3 | sort | uniq
|
||||||
|
- # curl --upload-file Cool*.AppImage https://transfer.sh/Cool_Retro_Term-git.$(git rev-parse --short HEAD)-x86_64.AppImage
|
||||||
|
- wget -c https://github.com/probonopd/uploadtool/raw/master/upload.sh
|
||||||
|
- bash upload.sh Cool*.AppImage*
|
||||||
|
|
||||||
|
branches:
|
||||||
|
except:
|
||||||
|
- # Do not build tags that we create when we upload to GitHub Releases
|
||||||
|
- /^(?i:continuous)/
|
167
README.md
167
README.md
@ -1,165 +1,30 @@
|
|||||||
# cool-retro-term
|
# cool-retro-term
|
||||||
|
|
||||||
|
|> Default Amber|C:\ IBM DOS|$ Default Green|
|
||||||
|
|---|---|---|
|
||||||
|
||||
|
||||||
|
|
||||||
## Description
|
## Description
|
||||||
cool-retro-term is a terminal emulator which mimics the look and feel of the old cathode tube screens.
|
cool-retro-term is a terminal emulator which mimics the look and feel of the old cathode tube screens.
|
||||||
It has been designed to be eye-candy, customizable, and reasonably lightweight.
|
It has been designed to be eye-candy, customizable, and reasonably lightweight.
|
||||||
|
|
||||||
It uses the QML port of qtermwidget (Konsole) developed by me: https://github.com/Swordfish90/qmltermwidget .
|
It uses the QML port of qtermwidget (Konsole): https://github.com/Swordfish90/qmltermwidget.
|
||||||
|
|
||||||
This terminal emulator works under Linux and OSX and requires Qt 5.2 or higher.
|
This terminal emulator works under Linux and macOS and requires Qt5. It's suggested that you stick to the latest LTS version.
|
||||||
|
|
||||||
|
Settings such as colors, fonts, and effects can be accessed via context menu.
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||

|

|
||||||

|

|
||||||

|

|
||||||
|
|
||||||
## Get cool-retro-term
|
## Install
|
||||||
You can either build cool-retro-term yourself (see below) or walk the easy way and install one of these packages:
|
|
||||||
|
|
||||||
Users of Fedora and openSUSE can grab a package from [Open Build Service](http://software.opensuse.org/package/cool-retro-term).
|
If you want to get a hold of the latest version, just go to the Releases page and grab the latest AppImage (Linux) or dmg (macOS).
|
||||||
|
|
||||||
Arch users can install this [package](https://aur.archlinux.org/packages/cool-retro-term-git/) directly via the [AUR](https://aur.archlinux.org):
|
Alternatively, most distributions such as Ubuntu, Fedora or Arch already package cool-retro-term in their official repositories.
|
||||||
|
|
||||||
yaourt -S aur/cool-retro-term-git
|
## Building
|
||||||
|
|
||||||
or use:
|
Check out the wiki and follow the instructions on how to build it on [Linux](https://github.com/Swordfish90/cool-retro-term/wiki/Build-Instructions-(Linux)) and [macOS](https://github.com/Swordfish90/cool-retro-term/wiki/Build-Instructions-(macOS)).
|
||||||
|
|
||||||
pacman -S cool-retro-term
|
|
||||||
|
|
||||||
to install precompiled from community repository.
|
|
||||||
|
|
||||||
Gentoo users can now install the first release "1.0" from a 3rd-party repository preferably via layman:
|
|
||||||
|
|
||||||
USE="git" emerge app-portage/layman
|
|
||||||
wget https://www.gerczei.eu/files/gerczei.xml -O /etc/layman/overlays/gerczei.xml
|
|
||||||
layman -f -a qt -a gerczei # those who've added the repo before 27/08/17 should remove, update and add it again as its source has changed
|
|
||||||
ACCEPT_KEYWORDS="~*" emerge =x11-terms/cool-retro-term-1.0.0-r1::gerczei
|
|
||||||
|
|
||||||
The live ebuild (version 9999-r1) tracking the bleeding-edge WIP codebase also remains available.
|
|
||||||
|
|
||||||
A word of warning: USE flags and keywords are to be added to portage's configuration files and every emerge operation should be executed with '-p' (short option for --pretend) appended to the command line first as per best practice!
|
|
||||||
|
|
||||||
Ubuntu users of 14.04 LTS (Trusty) up to 15.10 (Wily) can use [this PPA](https://launchpad.net/~bugs-launchpad-net-falkensweb)
|
|
||||||
|
|
||||||
OSX users can grab the latest dmg from the release page: https://github.com/Swordfish90/cool-retro-term/releases
|
|
||||||
|
|
||||||
## Build instructions (Linux)
|
|
||||||
|
|
||||||
## Dependencies
|
|
||||||
Make sure to install these first.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Ubuntu 14.04**
|
|
||||||
|
|
||||||
sudo apt-get install build-essential qmlscene qt5-qmake qt5-default qtdeclarative5-dev qtdeclarative5-controls-plugin qtdeclarative5-qtquick2-plugin libqt5qml-graphicaleffects qtdeclarative5-dialogs-plugin qtdeclarative5-localstorage-plugin qtdeclarative5-window-plugin
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Ubuntu 16.10**
|
|
||||||
|
|
||||||
sudo apt-get install build-essential qmlscene qt5-qmake qt5-default qtdeclarative5-dev qml-module-qtquick-controls qtdeclarative5-qtquick2-plugin libqt5qml-graphicaleffects qml-module-qtquick-dialogs qtdeclarative5-localstorage-plugin qtdeclarative5-window-plugin
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Ubuntu 17.04**
|
|
||||||
|
|
||||||
sudo apt-get install build-essential libqt5qml-graphicaleffects qml-module-qt-labs-folderlistmodel qml-module-qt-labs-settings qml-module-qtquick-controls qml-module-qtquick-dialogs qmlscene qt5-default qt5-qmake qtdeclarative5-dev qtdeclarative5-localstorage-plugin qtdeclarative5-qtquick2-plugin qtdeclarative5-window-plugin
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Debian Jessie**
|
|
||||||
|
|
||||||
sudo apt-get install build-essential qmlscene qt5-qmake qt5-default qtdeclarative5-dev qml-module-qtquick-controls qml-module-qtgraphicaleffects qml-module-qtquick-dialogs qml-module-qtquick-localstorage qml-module-qtquick-window2
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Fedora**
|
|
||||||
This command should install the known fedora dependencies:
|
|
||||||
|
|
||||||
sudo yum -y install qt5-qtbase qt5-qtbase-devel qt5-qtdeclarative qt5-qtdeclarative-devel qt5-qtgraphicaleffects qt5-qtquickcontrols redhat-rpm-config
|
|
||||||
|
|
||||||
or:
|
|
||||||
|
|
||||||
sudo dnf -y install qt5-qtbase qt5-qtbase-devel qt5-qtdeclarative qt5-qtdeclarative-devel qt5-qtgraphicaleffects qt5-qtquickcontrols redhat-rpm-config
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Arch Linux**
|
|
||||||
|
|
||||||
sudo pacman -S qt5-base qt5-declarative qt5-quickcontrols qt5-graphicaleffects
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**openSUSE**
|
|
||||||
|
|
||||||
Add repository with latest Qt 5 (this is only needed on openSUSE 13.1, Factory already has it):
|
|
||||||
|
|
||||||
sudo zypper ar http://download.opensuse.org/repositories/KDE:/Qt5/openSUSE_13.1/ KDE:Qt5
|
|
||||||
|
|
||||||
Install dependencies:
|
|
||||||
|
|
||||||
sudo zypper install libqt5-qtbase-devel libqt5-qtdeclarative-devel libqt5-qtquickcontrols libqt5-qtgraphicaleffects
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Anyone else**
|
|
||||||
|
|
||||||
Install Qt directly from here http://qt-project.org/downloads . Once done export them in you path (replace "_/opt/Qt5.3.1/5.3/gcc_64/bin_" with your correct folder):
|
|
||||||
|
|
||||||
export PATH=/opt/Qt5.3.1/5.3/gcc_64/bin/:$PATH
|
|
||||||
---
|
|
||||||
|
|
||||||
### Compile
|
|
||||||
Once you installed all dependencies (Qt is installed and in your path) you need to compile and run the application:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Get it from GitHub
|
|
||||||
git clone --recursive https://github.com/Swordfish90/cool-retro-term.git
|
|
||||||
|
|
||||||
# Build it
|
|
||||||
cd cool-retro-term
|
|
||||||
|
|
||||||
# Compile (Fedora and OpenSUSE user should use qmake-qt5 instead of qmake)
|
|
||||||
qmake && make
|
|
||||||
|
|
||||||
# Have fun!
|
|
||||||
./cool-retro-term
|
|
||||||
```
|
|
||||||
|
|
||||||
## Build instructions (OSX)
|
|
||||||
|
|
||||||
1. Install [Xcode](https://developer.apple.com/xcode/) and agree to the licence agreement
|
|
||||||
2. Enter the following commands into the terminal:
|
|
||||||
|
|
||||||
**Brew**
|
|
||||||
|
|
||||||
```sh
|
|
||||||
brew install qt5
|
|
||||||
git clone --recursive https://github.com/Swordfish90/cool-retro-term.git
|
|
||||||
export CPPFLAGS="-I/usr/local/opt/qt5/include"
|
|
||||||
export LDFLAGS="-L/usr/local/opt/qt5/lib"
|
|
||||||
export PATH=/usr/local/opt/qt5/bin:$PATH
|
|
||||||
cd cool-retro-term
|
|
||||||
qmake && make
|
|
||||||
mkdir cool-retro-term.app/Contents/PlugIns
|
|
||||||
cp -r qmltermwidget/QMLTermWidget cool-retro-term.app/Contents/PlugIns
|
|
||||||
open cool-retro-term.app
|
|
||||||
```
|
|
||||||
|
|
||||||
**MacPorts**
|
|
||||||
|
|
||||||
```sh
|
|
||||||
sudo port install qt5
|
|
||||||
git clone --recursive https://github.com/Swordfish90/cool-retro-term.git
|
|
||||||
cd cool-retro-term
|
|
||||||
/opt/local/libexec/qt5/bin/qmake && make
|
|
||||||
mkdir cool-retro-term.app/Contents/PlugIns
|
|
||||||
cp -r qmltermwidget/QMLTermWidget cool-retro-term.app/Contents/PlugIns
|
|
||||||
open cool-retro-term.app
|
|
||||||
```
|
|
||||||
|
|
||||||
## Donations
|
|
||||||
I made this project in my spare time because I love what I'm doing. If you are enjoying it and you want to buy me a beer click [here](https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=flscogna%40gmail%2ecom&lc=IT&item_name=Filippo%20Scognamiglio¤cy_code=EUR&bn=PP%2dDonationsBF%3abtn_donate_LG%2egif%3aNonHosted) .
|
|
||||||
|
|
||||||
You can also add "bounties" on your favourite issues. More information on the [Bountysource](https://www.bountysource.com/teams/crt/issues) page.
|
|
||||||
|
10
app/app.pro
10
app/app.pro
@ -1,20 +1,22 @@
|
|||||||
QT += qml quick widgets sql
|
QT += qml quick widgets sql quickcontrols2
|
||||||
TARGET = cool-retro-term
|
TARGET = cool-retro-term
|
||||||
|
|
||||||
DESTDIR = $$OUT_PWD/../
|
DESTDIR = $$OUT_PWD/../
|
||||||
|
|
||||||
HEADERS += \
|
HEADERS += \
|
||||||
fileio.h
|
fileio.h \
|
||||||
|
monospacefontmanager.h
|
||||||
|
|
||||||
SOURCES = main.cpp \
|
SOURCES = main.cpp \
|
||||||
fileio.cpp
|
fileio.cpp \
|
||||||
|
monospacefontmanager.cpp
|
||||||
|
|
||||||
macx:ICON = icons/crt.icns
|
macx:ICON = icons/crt.icns
|
||||||
|
|
||||||
RESOURCES += qml/resources.qrc
|
RESOURCES += qml/resources.qrc
|
||||||
|
|
||||||
#########################################
|
#########################################
|
||||||
## INTALLS
|
## INSTALLS
|
||||||
#########################################
|
#########################################
|
||||||
|
|
||||||
target.path += /usr/bin/
|
target.path += /usr/bin/
|
||||||
|
61
app/main.cpp
61
app/main.cpp
@ -6,11 +6,16 @@
|
|||||||
|
|
||||||
#include <QtWidgets/QApplication>
|
#include <QtWidgets/QApplication>
|
||||||
#include <QIcon>
|
#include <QIcon>
|
||||||
|
#include <QQuickStyle>
|
||||||
|
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include <QFontDatabase>
|
||||||
|
#include <QLoggingCategory>
|
||||||
|
|
||||||
#include <fileio.h>
|
#include <fileio.h>
|
||||||
|
#include <monospacefontmanager.h>
|
||||||
|
|
||||||
QString getNamedArgument(QStringList args, QString name, QString defaultName)
|
QString getNamedArgument(QStringList args, QString name, QString defaultName)
|
||||||
{
|
{
|
||||||
@ -30,14 +35,47 @@ int main(int argc, char *argv[])
|
|||||||
// This disables QT appmenu under Ubuntu, which is not working with QML apps.
|
// This disables QT appmenu under Ubuntu, which is not working with QML apps.
|
||||||
setenv("QT_QPA_PLATFORMTHEME", "", 1);
|
setenv("QT_QPA_PLATFORMTHEME", "", 1);
|
||||||
|
|
||||||
|
// Disable Connections slot warnings
|
||||||
|
QLoggingCategory::setFilterRules("qt.qml.connections.warning=false");
|
||||||
|
|
||||||
|
#if defined (Q_OS_LINUX)
|
||||||
|
setenv("QSG_RENDER_LOOP", "threaded", 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
#if defined(Q_OS_MAC)
|
#if defined(Q_OS_MAC)
|
||||||
// This allows UTF-8 characters usage in OSX.
|
// This allows UTF-8 characters usage in OSX.
|
||||||
setenv("LC_CTYPE", "UTF-8", 1);
|
setenv("LC_CTYPE", "UTF-8", 1);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
if (argc>1 && (!strcmp(argv[1],"-h") || !strcmp(argv[1],"--help"))) {
|
||||||
|
QTextStream cout(stdout, QIODevice::WriteOnly);
|
||||||
|
cout << "Usage: " << argv[0] << " [--default-settings] [--workdir <dir>] [--program <prog>] [-p|--profile <prof>] [--fullscreen] [-h|--help]" << '\n';
|
||||||
|
cout << " --default-settings Run cool-retro-term with the default settings" << '\n';
|
||||||
|
cout << " --workdir <dir> Change working directory to 'dir'" << '\n';
|
||||||
|
cout << " -e <cmd> Command to execute. This option will catch all following arguments, so use it as the last option." << '\n';
|
||||||
|
cout << " -T <title> Set window title to 'title'." << '\n';
|
||||||
|
cout << " --fullscreen Run cool-retro-term in fullscreen." << '\n';
|
||||||
|
cout << " -p|--profile <prof> Run cool-retro-term with the given profile." << '\n';
|
||||||
|
cout << " -h|--help Print this help." << '\n';
|
||||||
|
cout << " --verbose Print additional information such as profiles and settings." << '\n';
|
||||||
|
cout << " -v|--version Print the program name and version." << '\n';
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString appVersion("1.2.0");
|
||||||
|
|
||||||
|
if (argc>1 && (!strcmp(argv[1],"-v") || !strcmp(argv[1],"--version"))) {
|
||||||
|
QTextStream cout(stdout, QIODevice::WriteOnly);
|
||||||
|
cout << "cool-retro-term " << appVersion << '\n';
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
QApplication app(argc, argv);
|
QApplication app(argc, argv);
|
||||||
|
app.setAttribute(Qt::AA_MacDontSwapCtrlAndMeta, true);
|
||||||
|
|
||||||
QQmlApplicationEngine engine;
|
QQmlApplicationEngine engine;
|
||||||
FileIO fileIO;
|
FileIO fileIO;
|
||||||
|
MonospaceFontManager monospaceFontManager;
|
||||||
|
|
||||||
#if !defined(Q_OS_MAC)
|
#if !defined(Q_OS_MAC)
|
||||||
app.setWindowIcon(QIcon::fromTheme("cool-retro-term", QIcon(":../icons/32x32/cool-retro-term.png")));
|
app.setWindowIcon(QIcon::fromTheme("cool-retro-term", QIcon(":../icons/32x32/cool-retro-term.png")));
|
||||||
@ -45,26 +83,11 @@ int main(int argc, char *argv[])
|
|||||||
app.setWindowIcon(QIcon(":../icons/32x32/cool-retro-term.png"));
|
app.setWindowIcon(QIcon(":../icons/32x32/cool-retro-term.png"));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
app.setOrganizationName("cool-retro-term");
|
||||||
|
app.setOrganizationDomain("cool-retro-term");
|
||||||
|
|
||||||
// Manage command line arguments from the cpp side
|
// Manage command line arguments from the cpp side
|
||||||
QStringList args = app.arguments();
|
QStringList args = app.arguments();
|
||||||
if (args.contains("-h") || args.contains("--help")) {
|
|
||||||
// BUG: This usage help text goes to stderr, should go to stdout.
|
|
||||||
// BUG: First line of output is surrounded by double quotes.
|
|
||||||
qDebug() << "Usage: " + args.at(0) + " [--default-settings] [--workdir <dir>] [--program <prog>] [-p|--profile <prof>] [--fullscreen] [-h|--help]";
|
|
||||||
qDebug() << " --default-settings Run cool-retro-term with the default settings";
|
|
||||||
qDebug() << " --workdir <dir> Change working directory to 'dir'";
|
|
||||||
qDebug() << " -e <cmd> Command to execute. This option will catch all following arguments, so use it as the last option.";
|
|
||||||
qDebug() << " --fullscreen Run cool-retro-term in fullscreen.";
|
|
||||||
qDebug() << " -p|--profile <prof> Run cool-retro-term with the given profile.";
|
|
||||||
qDebug() << " -h|--help Print this help.";
|
|
||||||
qDebug() << " --verbose Print additional information such as profiles and settings.";
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args.contains("-v") || args.contains("--version")) {
|
|
||||||
qDebug() << "cool-retro-term 1.0.1";
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Manage default command
|
// Manage default command
|
||||||
QStringList cmdList;
|
QStringList cmdList;
|
||||||
@ -73,11 +96,13 @@ int main(int argc, char *argv[])
|
|||||||
}
|
}
|
||||||
QVariant command(cmdList.empty() ? QVariant() : cmdList[0]);
|
QVariant command(cmdList.empty() ? QVariant() : cmdList[0]);
|
||||||
QVariant commandArgs(cmdList.size() <= 1 ? QVariant() : QVariant(cmdList.mid(1)));
|
QVariant commandArgs(cmdList.size() <= 1 ? QVariant() : QVariant(cmdList.mid(1)));
|
||||||
|
engine.rootContext()->setContextProperty("appVersion", appVersion);
|
||||||
engine.rootContext()->setContextProperty("defaultCmd", command);
|
engine.rootContext()->setContextProperty("defaultCmd", command);
|
||||||
engine.rootContext()->setContextProperty("defaultCmdArgs", commandArgs);
|
engine.rootContext()->setContextProperty("defaultCmdArgs", commandArgs);
|
||||||
|
|
||||||
engine.rootContext()->setContextProperty("workdir", getNamedArgument(args, "--workdir", "$HOME"));
|
engine.rootContext()->setContextProperty("workdir", getNamedArgument(args, "--workdir", "$HOME"));
|
||||||
engine.rootContext()->setContextProperty("fileIO", &fileIO);
|
engine.rootContext()->setContextProperty("fileIO", &fileIO);
|
||||||
|
engine.rootContext()->setContextProperty("monospaceSystemFonts", monospaceFontManager.retrieveMonospaceFonts());
|
||||||
|
|
||||||
engine.rootContext()->setContextProperty("devicePixelRatio", app.devicePixelRatio());
|
engine.rootContext()->setContextProperty("devicePixelRatio", app.devicePixelRatio());
|
||||||
|
|
||||||
|
25
app/monospacefontmanager.cpp
Normal file
25
app/monospacefontmanager.cpp
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#include "monospacefontmanager.h"
|
||||||
|
|
||||||
|
#include <QDebug>
|
||||||
|
|
||||||
|
MonospaceFontManager::MonospaceFontManager(QObject *parent) : QObject(parent)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringList MonospaceFontManager::retrieveMonospaceFonts() {
|
||||||
|
QStringList result;
|
||||||
|
|
||||||
|
QFontDatabase fontDatabase;
|
||||||
|
QStringList fontFamilies = fontDatabase.families();
|
||||||
|
|
||||||
|
for (int i = 0; i < fontFamilies.size(); i++) {
|
||||||
|
QString fontFamily = fontFamilies[i];
|
||||||
|
QFont font(fontFamily);
|
||||||
|
if (fontDatabase.isFixedPitch(font.family())) {
|
||||||
|
result.append(fontFamily);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
15
app/monospacefontmanager.h
Normal file
15
app/monospacefontmanager.h
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
#ifndef MONOSPACEFONTMANAGER_H
|
||||||
|
#define MONOSPACEFONTMANAGER_H
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QFontDatabase>
|
||||||
|
|
||||||
|
class MonospaceFontManager : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit MonospaceFontManager(QObject *parent = nullptr);
|
||||||
|
Q_INVOKABLE QStringList retrieveMonospaceFonts();
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // MONOSPACEFONTMANAGER_H
|
@ -1,5 +1,24 @@
|
|||||||
|
/*******************************************************************************
|
||||||
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
|
*
|
||||||
|
* This file is part of cool-retro-term.
|
||||||
|
*
|
||||||
|
* cool-retro-term 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 <http://www.gnu.org/licenses/>.
|
||||||
|
*******************************************************************************/
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.2
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.1
|
||||||
import QtQuick.Window 2.0
|
import QtQuick.Window 2.0
|
||||||
|
|
||||||
@ -16,9 +35,12 @@ Window{
|
|||||||
anchors.margins: 15
|
anchors.margins: 15
|
||||||
spacing: 15
|
spacing: 15
|
||||||
Text {
|
Text {
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
Layout.alignment: Qt.AlignHCenter
|
||||||
text: "cool-retro-term"
|
text: "cool-retro-term"
|
||||||
font {bold: true; pointSize: 18}
|
font {
|
||||||
|
bold: true
|
||||||
|
pointSize: 18
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Loader {
|
Loader {
|
||||||
id: mainContent
|
id: mainContent
|
||||||
@ -41,7 +63,7 @@ Window{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
Component.onCompleted: mainContent.state = "Default";
|
Component.onCompleted: mainContent.state = "Default"
|
||||||
}
|
}
|
||||||
Item {
|
Item {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
@ -50,13 +72,14 @@ Window{
|
|||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
text: qsTr("License")
|
text: qsTr("License")
|
||||||
onClicked: {
|
onClicked: {
|
||||||
mainContent.state == "Default" ? mainContent.state = "License" : mainContent.state = "Default"
|
mainContent.state == "Default" ? mainContent.state
|
||||||
|
= "License" : mainContent.state = "Default"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Button {
|
Button {
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
text: qsTr("Close")
|
text: qsTr("Close")
|
||||||
onClicked: dialogwindow.close();
|
onClicked: dialogwindow.close()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -69,41 +92,42 @@ Window{
|
|||||||
Image {
|
Image {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
Layout.alignment: Qt.AlignHCenter
|
||||||
fillMode: Image.PreserveAspectFit
|
fillMode: Image.PreserveAspectFit
|
||||||
source: "images/crt256.png"
|
source: "images/crt256.png"
|
||||||
smooth: true
|
smooth: true
|
||||||
}
|
}
|
||||||
Text {
|
Text {
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
Layout.alignment: Qt.AlignCenter
|
||||||
horizontalAlignment: Text.AlignHCenter
|
horizontalAlignment: Text.AlignHCenter
|
||||||
text: appSettings.version + "\n" +
|
text: appSettings.version + "\n" + qsTr(
|
||||||
qsTr("Author: ") + "Filippo Scognamiglio\n" +
|
"Author: ") + "Filippo Scognamiglio\n" + qsTr(
|
||||||
qsTr("Email: ") + "flscogna@gmail.com\n" +
|
"Email: ") + "flscogna@gmail.com\n" + qsTr(
|
||||||
qsTr("Source: ") + "https://github.com/Swordfish90/cool-retro-term\n"
|
"Source: ") + "https://github.com/Swordfish90/cool-retro-term\n"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Component {
|
Component {
|
||||||
id: licenseComponent
|
id: licenseComponent
|
||||||
TextArea{
|
ScrollView {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
|
clip: true
|
||||||
|
TextArea {
|
||||||
readOnly: true
|
readOnly: true
|
||||||
text: "Copyright (c) 2013 Filippo Scognamiglio <flscogna@gmail.com>\n\n" +
|
wrapMode: TextEdit.Wrap
|
||||||
"https://github.com/Swordfish90/cool-retro-term\n\n" +
|
text: "Copyright (c) 2013-2021 Filippo Scognamiglio <flscogna@gmail.com>\n\n"
|
||||||
|
+ "https://github.com/Swordfish90/cool-retro-term\n\n" +
|
||||||
"cool-retro-term is free software: you can redistribute it and/or modify " +
|
"cool-retro-term is free software: you can redistribute it and/or modify "
|
||||||
"it under the terms of the GNU General Public License as published by " +
|
+ "it under the terms of the GNU General Public License as published by "
|
||||||
"the Free Software Foundation, either version 3 of the License, or " +
|
+ "the Free Software Foundation, either version 3 of the License, or "
|
||||||
"(at your option) any later version.\n\n" +
|
+ "(at your option) any later version.\n\n" +
|
||||||
|
"This program is distributed in the hope that it will be useful, "
|
||||||
"This program is distributed in the hope that it will be useful, " +
|
+ "but WITHOUT ANY WARRANTY; without even the implied warranty of "
|
||||||
"but WITHOUT ANY WARRANTY; without even the implied warranty of " +
|
+ "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the "
|
||||||
"MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the " +
|
+ "GNU General Public License for more details.\n\n" +
|
||||||
"GNU General Public License for more details.\n\n" +
|
"You should have received a copy of the GNU General Public License "
|
||||||
|
+ "along with this program. If not, see <http://www.gnu.org/licenses/>."
|
||||||
"You should have received a copy of the GNU General Public License " +
|
}
|
||||||
"along with this program. If not, see <http://www.gnu.org/licenses/>."
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,82 +17,110 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.0
|
import QtQuick.Controls 2.0
|
||||||
|
|
||||||
import "utils.js" as Utils
|
import "utils.js" as Utils
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
property string version: "1.0.1"
|
readonly property string version: appVersion
|
||||||
|
readonly property int profileVersion: 2
|
||||||
|
|
||||||
// STATIC CONSTANTS ////////////////////////////////////////////////////////
|
// STATIC CONSTANTS ////////////////////////////////////////////////////////
|
||||||
|
readonly property real screenCurvatureSize: 0.4
|
||||||
readonly property real minimumFontScaling: 0.25
|
readonly property real minimumFontScaling: 0.25
|
||||||
readonly property real maximumFontScaling: 2.50
|
readonly property real maximumFontScaling: 2.50
|
||||||
|
|
||||||
// GENERAL SETTINGS ///////////////////////////////////////////////////////
|
readonly property real minBurnInFadeTime: 160
|
||||||
|
readonly property real maxBurnInFadeTime: 1600
|
||||||
|
|
||||||
|
property bool isMacOS: Qt.platform.os === "osx"
|
||||||
|
|
||||||
|
// GENERAL SETTINGS ///////////////////////////////////////////////////////
|
||||||
property int x: 100
|
property int x: 100
|
||||||
property int y: 100
|
property int y: 100
|
||||||
property int width: 1024
|
property int width: 1024
|
||||||
property int height: 768
|
property int height: 768
|
||||||
|
|
||||||
property bool fullscreen: false
|
property bool fullscreen: false
|
||||||
property bool showMenubar: true
|
property bool showMenubar: false
|
||||||
|
|
||||||
property real windowOpacity: 1.0
|
property string wintitle: "cool-retro-term"
|
||||||
property real ambientLight: 0.2
|
|
||||||
property real contrast: 0.85
|
|
||||||
property real brightness: 0.5
|
|
||||||
|
|
||||||
property bool showTerminalSize: true
|
property bool showTerminalSize: true
|
||||||
property real windowScaling: 1.0
|
property real windowScaling: 1.0
|
||||||
|
|
||||||
property real fps: 24
|
property real fps: 20
|
||||||
property bool verbose: false
|
property bool verbose: false
|
||||||
|
|
||||||
onWindowScalingChanged: handleFontChanged();
|
property real bloomQuality: 0.5
|
||||||
|
property real burnInQuality: 0.5
|
||||||
|
|
||||||
|
property bool blinkingCursor: false
|
||||||
|
|
||||||
|
onWindowScalingChanged: handleFontChanged()
|
||||||
|
|
||||||
// PROFILE SETTINGS ///////////////////////////////////////////////////////
|
// PROFILE SETTINGS ///////////////////////////////////////////////////////
|
||||||
|
property real windowOpacity: 1.0
|
||||||
|
property real ambientLight: 0.2
|
||||||
|
property real contrast: 0.80
|
||||||
|
property real brightness: 0.5
|
||||||
|
|
||||||
property bool useCustomCommand: false
|
property bool useCustomCommand: false
|
||||||
property string customCommand: ""
|
property string customCommand: ""
|
||||||
|
|
||||||
property string _backgroundColor: "#000000"
|
property string _backgroundColor: "#000000"
|
||||||
property string _fontColor: "#ff8100"
|
property string _fontColor: "#ff8100"
|
||||||
property string saturatedColor: Utils.mix(Utils.strToColor("#FFFFFF"), Utils.strToColor(_fontColor), saturationColor * 0.5)
|
property string _frameColor: "#ffffff"
|
||||||
property color fontColor: Utils.mix(Utils.strToColor(saturatedColor), Utils.strToColor(_backgroundColor), 0.7 + (contrast * 0.3))
|
property string saturatedColor: Utils.mix(Utils.strToColor("#FFFFFF"),
|
||||||
property color backgroundColor: Utils.mix(Utils.strToColor(_backgroundColor), Utils.strToColor(saturatedColor), 0.7 + (contrast * 0.3))
|
Utils.strToColor(_fontColor),
|
||||||
|
saturationColor * 0.5)
|
||||||
|
property color fontColor: Utils.mix(Utils.strToColor(saturatedColor),
|
||||||
|
Utils.strToColor(_backgroundColor),
|
||||||
|
0.7 + (contrast * 0.3))
|
||||||
|
property color backgroundColor: Utils.mix(Utils.strToColor(
|
||||||
|
_backgroundColor),
|
||||||
|
Utils.strToColor(saturatedColor),
|
||||||
|
0.7 + (contrast * 0.3))
|
||||||
|
property color frameColor: Utils.strToColor(_frameColor)
|
||||||
|
|
||||||
property real staticNoise: 0.1
|
property real staticNoise: 0.12
|
||||||
property real screenCurvature: 0.1
|
property real screenCurvature: 0.3
|
||||||
property real glowingLine: 0.2
|
property real glowingLine: 0.2
|
||||||
property real burnIn: 0.40
|
property real burnIn: 0.25
|
||||||
property real bloom: 0.65
|
property real bloom: 0.55
|
||||||
|
|
||||||
property real bloomQuality: 0.5
|
property real chromaColor: 0.25
|
||||||
property real burnInQuality: 0.5
|
property real saturationColor: 0.25
|
||||||
|
property real frameGloss: 0
|
||||||
|
|
||||||
property real chromaColor: 0.0
|
property real jitter: 0.2
|
||||||
property real saturationColor: 0.0
|
|
||||||
|
|
||||||
property real jitter: 0.18
|
|
||||||
|
|
||||||
property real horizontalSync: 0.08
|
property real horizontalSync: 0.08
|
||||||
property real flickering: 0.1
|
property real flickering: 0.1
|
||||||
|
|
||||||
property real rbgShift: 0.0
|
property real rbgShift: 0.0
|
||||||
|
|
||||||
|
property real _margin: 0.5
|
||||||
|
property real _frameMargin: 0.5
|
||||||
|
|
||||||
|
property real margin: Utils.lint(1.0, 20.0, _margin)
|
||||||
|
property real frameMargin: Utils.lint(1.0, 50.0, _frameMargin)
|
||||||
|
|
||||||
|
property real totalMargin: frameMargin + margin
|
||||||
|
|
||||||
readonly property int no_rasterization: 0
|
readonly property int no_rasterization: 0
|
||||||
readonly property int scanline_rasterization: 1
|
readonly property int scanline_rasterization: 1
|
||||||
readonly property int pixel_rasterization: 2
|
readonly property int pixel_rasterization: 2
|
||||||
|
readonly property int subpixel_rasterization: 3
|
||||||
|
|
||||||
property int rasterization: no_rasterization
|
property int rasterization: no_rasterization
|
||||||
|
|
||||||
// FONTS //////////////////////////////////////////////////////////////////
|
// FONTS //////////////////////////////////////////////////////////////////
|
||||||
|
readonly property real baseFontScaling: 0.75
|
||||||
property real fontScaling: 1.0
|
property real fontScaling: 1.0
|
||||||
|
property real totalFontScaling: baseFontScaling * fontScaling
|
||||||
|
|
||||||
property real fontWidth: 1.0
|
property real fontWidth: 1.0
|
||||||
|
|
||||||
property bool lowResolutionFont: false
|
property bool lowResolutionFont: false
|
||||||
@ -100,205 +128,201 @@ QtObject{
|
|||||||
property var fontNames: ["TERMINUS_SCALED", "COMMODORE_PET", "COMMODORE_PET"]
|
property var fontNames: ["TERMINUS_SCALED", "COMMODORE_PET", "COMMODORE_PET"]
|
||||||
property var fontlist: fontManager.item.fontlist
|
property var fontlist: fontManager.item.fontlist
|
||||||
|
|
||||||
signal terminalFontChanged(string fontSource, int pixelSize, int lineSpacing, real screenScaling, real fontWidth)
|
signal terminalFontChanged(string fontFamily, int pixelSize, int lineSpacing, real screenScaling, real fontWidth)
|
||||||
|
|
||||||
signal initializedSettings()
|
signal initializedSettings
|
||||||
|
|
||||||
property Loader fontManager: Loader {
|
property Loader fontManager: Loader {
|
||||||
states: [
|
states: [
|
||||||
State { when: rasterization == no_rasterization
|
State {
|
||||||
PropertyChanges {target: fontManager; source: "Fonts.qml" } },
|
when: rasterization == no_rasterization
|
||||||
State { when: rasterization == scanline_rasterization
|
PropertyChanges {
|
||||||
PropertyChanges {target: fontManager; source: "FontScanlines.qml" } },
|
target: fontManager
|
||||||
State { when: rasterization == pixel_rasterization;
|
source: "Fonts.qml"
|
||||||
PropertyChanges {target: fontManager; source: "FontPixels.qml" } }
|
}
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
when: rasterization == scanline_rasterization
|
||||||
|
PropertyChanges {
|
||||||
|
target: fontManager
|
||||||
|
source: "FontScanlines.qml"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
when: rasterization == pixel_rasterization
|
||||||
|
PropertyChanges {
|
||||||
|
target: fontManager
|
||||||
|
source: "FontPixels.qml"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
State {
|
||||||
|
when: rasterization == subpixel_rasterization
|
||||||
|
PropertyChanges {
|
||||||
|
target: fontManager
|
||||||
|
source: "FontPixels.qml"
|
||||||
|
}
|
||||||
|
}
|
||||||
]
|
]
|
||||||
|
|
||||||
onLoaded: handleFontChanged()
|
onLoaded: handleFontChanged()
|
||||||
}
|
}
|
||||||
|
|
||||||
onFontScalingChanged: handleFontChanged();
|
property FontLoader fontLoader: FontLoader {}
|
||||||
onFontWidthChanged: handleFontChanged();
|
|
||||||
|
onTotalFontScalingChanged: handleFontChanged()
|
||||||
|
onFontWidthChanged: handleFontChanged()
|
||||||
|
|
||||||
function getIndexByName(name) {
|
function getIndexByName(name) {
|
||||||
for (var i = 0; i < fontlist.count; i++) {
|
for (var i = 0; i < fontlist.count; i++) {
|
||||||
if (name === fontlist.get(i).name)
|
var requestedName = fontlist.get(i).name
|
||||||
return i;
|
if (name === requestedName)
|
||||||
|
return i
|
||||||
}
|
}
|
||||||
return 0; // If the font is not available default to 0.
|
return 0 // If the font is not available default to 0.
|
||||||
}
|
}
|
||||||
|
|
||||||
function incrementScaling() {
|
function incrementScaling() {
|
||||||
fontScaling = Math.min(fontScaling + 0.05, maximumFontScaling);
|
fontScaling = Math.min(fontScaling + 0.05, maximumFontScaling)
|
||||||
handleFontChanged();
|
handleFontChanged()
|
||||||
}
|
}
|
||||||
|
|
||||||
function decrementScaling() {
|
function decrementScaling() {
|
||||||
fontScaling = Math.max(fontScaling - 0.05, minimumFontScaling);
|
fontScaling = Math.max(fontScaling - 0.05, minimumFontScaling)
|
||||||
handleFontChanged();
|
handleFontChanged()
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleFontChanged() {
|
function handleFontChanged() {
|
||||||
if (!fontManager.item) return;
|
if (!fontManager.item)
|
||||||
|
return
|
||||||
|
|
||||||
var index = getIndexByName(fontNames[rasterization]);
|
var index = getIndexByName(fontNames[rasterization])
|
||||||
if (index === undefined) return;
|
if (index === undefined)
|
||||||
|
return
|
||||||
|
|
||||||
fontManager.item.selectedFontIndex = index;
|
fontManager.item.selectedFontIndex = index
|
||||||
fontManager.item.scaling = fontScaling * windowScaling;
|
fontManager.item.scaling = totalFontScaling
|
||||||
|
|
||||||
var fontSource = fontManager.item.source;
|
var fontSource = fontManager.item.source
|
||||||
var pixelSize = fontManager.item.pixelSize;
|
var pixelSize = fontManager.item.pixelSize
|
||||||
var lineSpacing = fontManager.item.lineSpacing;
|
var lineSpacing = fontManager.item.lineSpacing
|
||||||
var screenScaling = fontManager.item.screenScaling;
|
var screenScaling = fontManager.item.screenScaling
|
||||||
var fontWidth = fontManager.item.defaultFontWidth * appSettings.fontWidth;
|
var fontWidth = fontManager.item.defaultFontWidth * appSettings.fontWidth
|
||||||
|
var fontFamily = fontManager.item.family
|
||||||
|
var isSystemFont = fontManager.item.isSystemFont
|
||||||
|
|
||||||
lowResolutionFont = fontManager.item.lowResolutionFont;
|
lowResolutionFont = fontManager.item.lowResolutionFont
|
||||||
|
|
||||||
terminalFontChanged(fontSource, pixelSize, lineSpacing, screenScaling, fontWidth);
|
if (!isSystemFont) {
|
||||||
|
fontLoader.source = fontSource
|
||||||
|
fontFamily = fontLoader.name
|
||||||
}
|
}
|
||||||
|
|
||||||
// FRAMES /////////////////////////////////////////////////////////////////
|
terminalFontChanged(fontFamily, pixelSize, lineSpacing, screenScaling,
|
||||||
|
fontWidth)
|
||||||
property ListModel framesList: ListModel{
|
|
||||||
ListElement{
|
|
||||||
name: "NO_FRAME"
|
|
||||||
text: "No frame"
|
|
||||||
source: ""
|
|
||||||
reflections: false
|
|
||||||
}
|
}
|
||||||
ListElement{
|
|
||||||
name: "SIMPLE_WHITE_FRAME"
|
|
||||||
text: "Simple white frame"
|
|
||||||
source: "./frames/WhiteSimpleFrame.qml"
|
|
||||||
reflections: true
|
|
||||||
}
|
|
||||||
ListElement{
|
|
||||||
name: "ROUGH_BLACK_FRAME"
|
|
||||||
text: "Rough black frame"
|
|
||||||
source: "./frames/BlackRoughFrame.qml"
|
|
||||||
reflections: true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getFrameIndexByName(name) {
|
|
||||||
for (var i = 0; i < framesList.count; i++) {
|
|
||||||
if (name === framesList.get(i).name)
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
return 0; // If the frame is not available default to 0.
|
|
||||||
}
|
|
||||||
|
|
||||||
property string frameSource: "./frames/WhiteSimpleFrame.qml"
|
|
||||||
property string frameName: "SIMPLE_WHITE_FRAME"
|
|
||||||
|
|
||||||
property bool _frameReflections: false
|
|
||||||
property bool reflectionsAllowed: true
|
|
||||||
property bool frameReflections: _frameReflections && reflectionsAllowed
|
|
||||||
|
|
||||||
onFrameNameChanged: {
|
|
||||||
var index = getFrameIndexByName(frameName);
|
|
||||||
frameSource = framesList.get(index).source;
|
|
||||||
reflectionsAllowed = framesList.get(index).reflections;
|
|
||||||
}
|
|
||||||
|
|
||||||
// DB STORAGE /////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
property Storage storage: Storage {}
|
property Storage storage: Storage {}
|
||||||
|
|
||||||
function stringify(obj) {
|
function stringify(obj) {
|
||||||
var replacer = function (key, val) {
|
var replacer = function (key, val) {
|
||||||
return val.toFixed ? Number(val.toFixed(4)) : val;
|
return val.toFixed ? Number(val.toFixed(4)) : val
|
||||||
}
|
}
|
||||||
return JSON.stringify(obj, replacer, 2);
|
return JSON.stringify(obj, replacer, 2)
|
||||||
}
|
}
|
||||||
|
|
||||||
function composeSettingsString() {
|
function composeSettingsString() {
|
||||||
var settings = {
|
var settings = {
|
||||||
fps: fps,
|
"fps": fps,
|
||||||
x: x,
|
"x": x,
|
||||||
y: y,
|
"y": y,
|
||||||
width: width,
|
"width": width,
|
||||||
height: height,
|
"height": height,
|
||||||
windowScaling: windowScaling,
|
"windowScaling": windowScaling,
|
||||||
showTerminalSize: showTerminalSize,
|
"showTerminalSize": showTerminalSize,
|
||||||
fontScaling: fontScaling,
|
"fontScaling": fontScaling,
|
||||||
fontNames: fontNames,
|
"fontNames": fontNames,
|
||||||
frameReflections: _frameReflections,
|
"showMenubar": showMenubar,
|
||||||
showMenubar: showMenubar,
|
"bloomQuality": bloomQuality,
|
||||||
bloomQuality: bloomQuality,
|
"burnInQuality": burnInQuality,
|
||||||
burnInQuality: burnInQuality
|
"useCustomCommand": useCustomCommand,
|
||||||
|
"customCommand": customCommand
|
||||||
}
|
}
|
||||||
return stringify(settings);
|
return stringify(settings)
|
||||||
}
|
}
|
||||||
|
|
||||||
function composeProfileObject() {
|
function composeProfileObject() {
|
||||||
var settings = {
|
var settings = {
|
||||||
backgroundColor: _backgroundColor,
|
"backgroundColor": _backgroundColor,
|
||||||
fontColor: _fontColor,
|
"fontColor": _fontColor,
|
||||||
flickering: flickering,
|
"frameColor": _frameColor,
|
||||||
horizontalSync: horizontalSync,
|
"flickering": flickering,
|
||||||
staticNoise: staticNoise,
|
"horizontalSync": horizontalSync,
|
||||||
chromaColor: chromaColor,
|
"staticNoise": staticNoise,
|
||||||
saturationColor: saturationColor,
|
"chromaColor": chromaColor,
|
||||||
screenCurvature: screenCurvature,
|
"saturationColor": saturationColor,
|
||||||
glowingLine: glowingLine,
|
"frameGloss": frameGloss,
|
||||||
frameName: frameName,
|
"screenCurvature": screenCurvature,
|
||||||
burnIn: burnIn,
|
"glowingLine": glowingLine,
|
||||||
bloom: bloom,
|
"burnIn": burnIn,
|
||||||
rasterization: rasterization,
|
"bloom": bloom,
|
||||||
jitter: jitter,
|
"rasterization": rasterization,
|
||||||
rbgShift: rbgShift,
|
"jitter": jitter,
|
||||||
brightness: brightness,
|
"rbgShift": rbgShift,
|
||||||
contrast: contrast,
|
"brightness": brightness,
|
||||||
ambientLight: ambientLight,
|
"contrast": contrast,
|
||||||
windowOpacity: windowOpacity,
|
"ambientLight": ambientLight,
|
||||||
fontName: fontNames[rasterization],
|
"windowOpacity": windowOpacity,
|
||||||
fontWidth: fontWidth,
|
"fontName": fontNames[rasterization],
|
||||||
useCustomCommand: useCustomCommand,
|
"fontWidth": fontWidth,
|
||||||
customCommand: customCommand
|
"margin": _margin,
|
||||||
|
"blinkingCursor": blinkingCursor,
|
||||||
|
"frameMargin": _frameMargin,
|
||||||
}
|
}
|
||||||
return settings;
|
return settings
|
||||||
}
|
}
|
||||||
|
|
||||||
function composeProfileString() {
|
function composeProfileString() {
|
||||||
return stringify(composeProfileObject());
|
return stringify(composeProfileObject())
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadSettings() {
|
function loadSettings() {
|
||||||
var settingsString = storage.getSetting("_CURRENT_SETTINGS");
|
var settingsString = storage.getSetting("_CURRENT_SETTINGS")
|
||||||
var profileString = storage.getSetting("_CURRENT_PROFILE");
|
var profileString = storage.getSetting("_CURRENT_PROFILE")
|
||||||
|
|
||||||
if(!settingsString) return;
|
if (!settingsString)
|
||||||
if(!profileString) return;
|
return
|
||||||
|
if (!profileString)
|
||||||
|
return
|
||||||
|
|
||||||
loadSettingsString(settingsString);
|
loadSettingsString(settingsString)
|
||||||
loadProfileString(profileString);
|
loadProfileString(profileString)
|
||||||
|
|
||||||
if (verbose)
|
if (verbose)
|
||||||
console.log("Loading settings: " + settingsString + profileString);
|
console.log("Loading settings: " + settingsString + profileString)
|
||||||
}
|
}
|
||||||
|
|
||||||
function storeSettings() {
|
function storeSettings() {
|
||||||
var settingsString = composeSettingsString();
|
var settingsString = composeSettingsString()
|
||||||
var profileString = composeProfileString();
|
var profileString = composeProfileString()
|
||||||
|
|
||||||
storage.setSetting("_CURRENT_SETTINGS", settingsString);
|
storage.setSetting("_CURRENT_SETTINGS", settingsString)
|
||||||
storage.setSetting("_CURRENT_PROFILE", profileString);
|
storage.setSetting("_CURRENT_PROFILE", profileString)
|
||||||
|
|
||||||
if (verbose) {
|
if (verbose) {
|
||||||
console.log("Storing settings: " + settingsString);
|
console.log("Storing settings: " + settingsString)
|
||||||
console.log("Storing profile: " + profileString);
|
console.log("Storing profile: " + profileString)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadSettingsString(settingsString) {
|
function loadSettingsString(settingsString) {
|
||||||
var settings = JSON.parse(settingsString);
|
var settings = JSON.parse(settingsString)
|
||||||
|
|
||||||
showTerminalSize = settings.showTerminalSize !== undefined ? settings.showTerminalSize : showTerminalSize
|
showTerminalSize = settings.showTerminalSize
|
||||||
|
!== undefined ? settings.showTerminalSize : showTerminalSize
|
||||||
|
|
||||||
fps = settings.fps !== undefined ? settings.fps : fps
|
fps = settings.fps !== undefined ? settings.fps : fps
|
||||||
windowScaling = settings.windowScaling !== undefined ? settings.windowScaling : windowScaling
|
windowScaling = settings.windowScaling
|
||||||
|
!== undefined ? settings.windowScaling : windowScaling
|
||||||
|
|
||||||
x = settings.x !== undefined ? settings.x : x
|
x = settings.x !== undefined ? settings.x : x
|
||||||
y = settings.y !== undefined ? settings.y : y
|
y = settings.y !== undefined ? settings.y : y
|
||||||
@ -308,140 +332,396 @@ QtObject{
|
|||||||
fontNames = settings.fontNames !== undefined ? settings.fontNames : fontNames
|
fontNames = settings.fontNames !== undefined ? settings.fontNames : fontNames
|
||||||
fontScaling = settings.fontScaling !== undefined ? settings.fontScaling : fontScaling
|
fontScaling = settings.fontScaling !== undefined ? settings.fontScaling : fontScaling
|
||||||
|
|
||||||
_frameReflections = settings.frameReflections !== undefined ? settings.frameReflections : _frameReflections;
|
showMenubar = settings.showMenubar !== undefined ? settings.showMenubar : showMenubar
|
||||||
|
|
||||||
showMenubar = settings.showMenubar !== undefined ? settings.showMenubar : showMenubar;
|
bloomQuality = settings.bloomQuality !== undefined ? settings.bloomQuality : bloomQuality
|
||||||
|
burnInQuality = settings.burnInQuality
|
||||||
|
!== undefined ? settings.burnInQuality : burnInQuality
|
||||||
|
|
||||||
bloomQuality = settings.bloomQuality !== undefined ? settings.bloomQuality : bloomQuality;
|
useCustomCommand = settings.useCustomCommand
|
||||||
burnInQuality = settings.burnInQuality !== undefined ? settings.burnInQuality : burnInQuality;
|
!== undefined ? settings.useCustomCommand : useCustomCommand
|
||||||
|
customCommand = settings.customCommand
|
||||||
|
!== undefined ? settings.customCommand : customCommand
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadProfileString(profileString) {
|
function loadProfileString(profileString) {
|
||||||
var settings = JSON.parse(profileString);
|
var settings = JSON.parse(profileString)
|
||||||
|
|
||||||
_backgroundColor = settings.backgroundColor !== undefined ? settings.backgroundColor : _backgroundColor;
|
_backgroundColor = settings.backgroundColor
|
||||||
_fontColor = settings.fontColor !== undefined ? settings.fontColor : _fontColor;
|
!== undefined ? settings.backgroundColor : _backgroundColor
|
||||||
|
_fontColor = settings.fontColor !== undefined ? settings.fontColor : _fontColor
|
||||||
|
_frameColor = settings.frameColor !== undefined ? settings.frameColor : _frameColor
|
||||||
|
|
||||||
horizontalSync = settings.horizontalSync !== undefined ? settings.horizontalSync : horizontalSync
|
horizontalSync = settings.horizontalSync
|
||||||
flickering = settings.flickering !== undefined ? settings.flickering : flickering;
|
!== undefined ? settings.horizontalSync : horizontalSync
|
||||||
staticNoise = settings.staticNoise !== undefined ? settings.staticNoise : staticNoise;
|
flickering = settings.flickering !== undefined ? settings.flickering : flickering
|
||||||
chromaColor = settings.chromaColor !== undefined ? settings.chromaColor : chromaColor;
|
staticNoise = settings.staticNoise !== undefined ? settings.staticNoise : staticNoise
|
||||||
saturationColor = settings.saturationColor !== undefined ? settings.saturationColor : saturationColor;
|
chromaColor = settings.chromaColor !== undefined ? settings.chromaColor : chromaColor
|
||||||
screenCurvature = settings.screenCurvature !== undefined ? settings.screenCurvature : screenCurvature;
|
saturationColor = settings.saturationColor
|
||||||
glowingLine = settings.glowingLine !== undefined ? settings.glowingLine : glowingLine;
|
!== undefined ? settings.saturationColor : saturationColor
|
||||||
|
frameGloss = settings.frameGloss !== undefined ? settings.frameGloss : frameGloss
|
||||||
|
screenCurvature = settings.screenCurvature
|
||||||
|
!== undefined ? settings.screenCurvature : screenCurvature
|
||||||
|
glowingLine = settings.glowingLine !== undefined ? settings.glowingLine : glowingLine
|
||||||
|
|
||||||
burnIn = settings.burnIn !== undefined ? settings.burnIn : burnIn
|
burnIn = settings.burnIn !== undefined ? settings.burnIn : burnIn
|
||||||
bloom = settings.bloom !== undefined ? settings.bloom : bloom
|
bloom = settings.bloom !== undefined ? settings.bloom : bloom
|
||||||
|
|
||||||
frameName = settings.frameName !== undefined ? settings.frameName : frameName;
|
rasterization = settings.rasterization
|
||||||
|
!== undefined ? settings.rasterization : rasterization
|
||||||
|
|
||||||
rasterization = settings.rasterization !== undefined ? settings.rasterization : rasterization;
|
jitter = settings.jitter !== undefined ? settings.jitter : jitter
|
||||||
|
|
||||||
jitter = settings.jitter !== undefined ? settings.jitter : jitter;
|
rbgShift = settings.rbgShift !== undefined ? settings.rbgShift : rbgShift
|
||||||
|
|
||||||
rbgShift = settings.rbgShift !== undefined ? settings.rbgShift : rbgShift;
|
ambientLight = settings.ambientLight !== undefined ? settings.ambientLight : ambientLight
|
||||||
|
contrast = settings.contrast !== undefined ? settings.contrast : contrast
|
||||||
|
brightness = settings.brightness !== undefined ? settings.brightness : brightness
|
||||||
|
windowOpacity = settings.windowOpacity
|
||||||
|
!== undefined ? settings.windowOpacity : windowOpacity
|
||||||
|
|
||||||
ambientLight = settings.ambientLight !== undefined ? settings.ambientLight : ambientLight;
|
fontNames[rasterization] = settings.fontName
|
||||||
contrast = settings.contrast !== undefined ? settings.contrast : contrast;
|
!== undefined ? settings.fontName : fontNames[rasterization]
|
||||||
brightness = settings.brightness !== undefined ? settings.brightness : brightness;
|
fontWidth = settings.fontWidth !== undefined ? settings.fontWidth : fontWidth
|
||||||
windowOpacity = settings.windowOpacity !== undefined ? settings.windowOpacity : windowOpacity;
|
|
||||||
|
|
||||||
fontNames[rasterization] = settings.fontName !== undefined ? settings.fontName : fontNames[rasterization];
|
_margin = settings.margin !== undefined ? settings.margin : _margin
|
||||||
fontWidth = settings.fontWidth !== undefined ? settings.fontWidth : fontWidth;
|
_frameMargin = settings.frameMargin !== undefined ? settings.frameMargin : _frameMargin
|
||||||
|
|
||||||
useCustomCommand = settings.useCustomCommand !== undefined ? settings.useCustomCommand : useCustomCommand
|
blinkingCursor = settings.blinkingCursor !== undefined ? settings.blinkingCursor : blinkingCursor
|
||||||
customCommand = settings.customCommand !== undefined ? settings.customCommand : customCommand
|
|
||||||
|
|
||||||
handleFontChanged();
|
handleFontChanged()
|
||||||
}
|
}
|
||||||
|
|
||||||
function storeCustomProfiles() {
|
function storeCustomProfiles() {
|
||||||
storage.setSetting("_CUSTOM_PROFILES", composeCustomProfilesString());
|
storage.setSetting("_CUSTOM_PROFILES", composeCustomProfilesString())
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadCustomProfiles() {
|
function loadCustomProfiles() {
|
||||||
var customProfileString = storage.getSetting("_CUSTOM_PROFILES");
|
var customProfileString = storage.getSetting("_CUSTOM_PROFILES")
|
||||||
if(customProfileString === undefined) customProfileString = "[]";
|
if (customProfileString === undefined)
|
||||||
loadCustomProfilesString(customProfileString);
|
customProfileString = "[]"
|
||||||
|
loadCustomProfilesString(customProfileString)
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadCustomProfilesString(customProfilesString) {
|
function loadCustomProfilesString(customProfilesString) {
|
||||||
var customProfiles = JSON.parse(customProfilesString);
|
var customProfiles = JSON.parse(customProfilesString)
|
||||||
for (var i = 0; i < customProfiles.length; i++) {
|
for (var i = 0; i < customProfiles.length; i++) {
|
||||||
var profile = customProfiles[i];
|
var profile = customProfiles[i]
|
||||||
|
|
||||||
if (verbose)
|
if (verbose)
|
||||||
console.log("Loading custom profile: " + stringify(profile));
|
console.log("Loading custom profile: " + stringify(profile))
|
||||||
|
|
||||||
profilesList.append(profile);
|
profilesList.append(profile)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function composeCustomProfilesString() {
|
function composeCustomProfilesString() {
|
||||||
var customProfiles = []
|
var customProfiles = []
|
||||||
for (var i = 0; i < profilesList.count; i++) {
|
for (var i = 0; i < profilesList.count; i++) {
|
||||||
var profile = profilesList.get(i);
|
var profile = profilesList.get(i)
|
||||||
if(profile.builtin) continue;
|
if (profile.builtin)
|
||||||
customProfiles.push({text: profile.text, obj_string: profile.obj_string, builtin: false})
|
continue
|
||||||
|
customProfiles.push({
|
||||||
|
"text": profile.text,
|
||||||
|
"obj_string": profile.obj_string,
|
||||||
|
"builtin": false
|
||||||
|
})
|
||||||
}
|
}
|
||||||
return stringify(customProfiles);
|
return stringify(customProfiles)
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadProfile(index) {
|
function loadProfile(index) {
|
||||||
var profile = profilesList.get(index);
|
var profile = profilesList.get(index)
|
||||||
loadProfileString(profile.obj_string);
|
loadProfileString(profile.obj_string)
|
||||||
}
|
}
|
||||||
|
|
||||||
function appendCustomProfile(name, profileString) {
|
function appendCustomProfile(name, profileString) {
|
||||||
profilesList.append({text: name, obj_string: profileString, builtin: false});
|
profilesList.append({
|
||||||
|
"text": name,
|
||||||
|
"obj_string": profileString,
|
||||||
|
"builtin": false
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// PROFILES ///////////////////////////////////////////////////////////////
|
// PROFILES ///////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
property ListModel profilesList: ListModel {
|
property ListModel profilesList: ListModel {
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "Default Amber"
|
text: "Default Amber"
|
||||||
obj_string: '{"ambientLight":0.16,"backgroundColor":"#000000","bloom":0.65,"brightness":0.5,"flickering":0.1,"contrast":0.85,"fontName":"TERMINUS_SCALED","fontColor":"#ff8100","frameName":"SIMPLE_WHITE_FRAME","glowingLine":0.2,"horizontalSync":0.16,"jitter":0.18,"burnIn":0.4,"staticNoise":0.1,"rasterization":0,"screenCurvature":0.1,"windowOpacity":1,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0.2,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.5538,
|
||||||
|
"brightness": 0.5,
|
||||||
|
"burnIn": 0.2517,
|
||||||
|
"chromaColor": 0.2483,
|
||||||
|
"contrast": 0.7959,
|
||||||
|
"flickering": 0.1,
|
||||||
|
"fontColor": "#ff8100",
|
||||||
|
"fontName": "TERMINUS_SCALED",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.2,
|
||||||
|
"horizontalSync": 0.08,
|
||||||
|
"jitter": 0.1997,
|
||||||
|
"rasterization": 0,
|
||||||
|
"rbgShift": 0,
|
||||||
|
"saturationColor": 0.2483,
|
||||||
|
"screenCurvature": 0.3,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.1198,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.1
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "Default Green"
|
text: "Monochrome Green"
|
||||||
obj_string: '{"ambientLight":0.16,"backgroundColor":"#000000","bloom":0.4,"brightness":0.5,"flickering":0.1,"contrast":0.85,"fontName":"TERMINUS_SCALED","fontColor":"#0ccc68","frameName":"SIMPLE_WHITE_FRAME","glowingLine":0.2,"horizontalSync":0.16,"jitter":0.18,"burnIn":0.45,"staticNoise":0.1,"rasterization":0,"screenCurvature":0.1,"windowOpacity":1,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0.2,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.5538,
|
||||||
|
"brightness": 0.5,
|
||||||
|
"burnIn": 0.2517,
|
||||||
|
"chromaColor": 0.0,
|
||||||
|
"contrast": 0.7959,
|
||||||
|
"flickering": 0.1,
|
||||||
|
"fontColor": "#0ccc68",
|
||||||
|
"fontName": "TERMINUS_SCALED",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.2,
|
||||||
|
"horizontalSync": 0.08,
|
||||||
|
"jitter": 0.1997,
|
||||||
|
"rasterization": 0,
|
||||||
|
"rbgShift": 0,
|
||||||
|
"saturationColor": 0.0,
|
||||||
|
"screenCurvature": 0.3,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.1198,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.1
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "Default Scanlines"
|
text: "Green Scanlines"
|
||||||
obj_string: '{"ambientLight":0.16,"backgroundColor":"#000000","bloom":0.4,"brightness":0.5,"flickering":0.1,"contrast":0.85,"fontName":"COMMODORE_PET","fontColor":"#00ff5b","frameName":"SIMPLE_WHITE_FRAME","glowingLine":0.2,"horizontalSync":0.14,"jitter":0.11,"burnIn":0.4,"staticNoise":0.05,"rasterization":1,"screenCurvature":0.1,"windowOpacity":1,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.6,
|
||||||
|
"brightness": 0.5,
|
||||||
|
"burnIn": 0.3,
|
||||||
|
"chromaColor": 0.5,
|
||||||
|
"contrast": 0.6,
|
||||||
|
"flickering": 0.1,
|
||||||
|
"fontColor": "#7cff4f",
|
||||||
|
"fontName": "PRO_FONT_SCALED",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.2,
|
||||||
|
"horizontalSync": 0.151,
|
||||||
|
"jitter": 0.11,
|
||||||
|
"rasterization": 1,
|
||||||
|
"rbgShift": 0,
|
||||||
|
"saturationColor": 0.5,
|
||||||
|
"screenCurvature": 0.3,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.15,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.1
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "Default Pixelated"
|
text: "Default Pixelated"
|
||||||
obj_string: '{"ambientLight":0.16,"backgroundColor":"#000000","bloom":0,"brightness":0.5,"flickering":0.2,"contrast":0.85,"fontName":"COMMODORE_PET","fontColor":"#ffffff","frameName":"ROUGH_BLACK_FRAME","glowingLine":0.2,"horizontalSync":0.2,"jitter":0,"burnIn":0.45,"staticNoise":0.19,"rasterization":2,"screenCurvature":0.05,"windowOpacity":1,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.4045,
|
||||||
|
"brightness": 0.6041,
|
||||||
|
"burnIn": 0.1024,
|
||||||
|
"chromaColor": 0.7517,
|
||||||
|
"contrast": 0.7473,
|
||||||
|
"flickering": 0.1962,
|
||||||
|
"fontColor": "#ffffff",
|
||||||
|
"fontName": "COMMODORE_PET",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.2,
|
||||||
|
"horizontalSync": 0.151,
|
||||||
|
"jitter": 0,
|
||||||
|
"rasterization": 2,
|
||||||
|
"rbgShift": 0,
|
||||||
|
"saturationColor": 0,
|
||||||
|
"screenCurvature": 0,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.15,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.1
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "Apple ]["
|
text: "Apple ]["
|
||||||
obj_string: '{"ambientLight":0.16,"backgroundColor":"#000000","bloom":0.5,"brightness":0.5,"flickering":0.2,"contrast":0.85,"fontName":"APPLE_II","fontColor":"#2fff91","frameName":"SIMPLE_WHITE_FRAME","glowingLine":0.22,"horizontalSync":0.16,"jitter":0.1,"burnIn":0.65,"staticNoise":0.08,"rasterization":1,"screenCurvature":0.18,"windowOpacity":1,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0.3038,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.5,
|
||||||
|
"brightness": 0.5,
|
||||||
|
"burnIn": 0.5017,
|
||||||
|
"chromaColor": 0,
|
||||||
|
"contrast": 0.85,
|
||||||
|
"flickering": 0.2,
|
||||||
|
"fontColor": "#00d56d",
|
||||||
|
"fontName": "APPLE_II",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.22,
|
||||||
|
"horizontalSync": 0.16,
|
||||||
|
"jitter": 0.1,
|
||||||
|
"rasterization": 1,
|
||||||
|
"rbgShift": 0,
|
||||||
|
"saturationColor": 0,
|
||||||
|
"screenCurvature": 0.5,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.099,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.2
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "Vintage"
|
text: "Vintage"
|
||||||
obj_string: '{"ambientLight":0.5,"backgroundColor":"#000000","bloom":0.4,"brightness":0.5,"flickering":0.9,"contrast":0.80,"fontName":"COMMODORE_PET","fontColor":"#00ff3e","frameName":"ROUGH_BLACK_FRAME","glowingLine":0.3,"horizontalSync":0.42,"jitter":0.4,"burnIn":0.75,"staticNoise":0.2,"rasterization":1,"screenCurvature":0.1,"windowOpacity":1,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0.5,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.4983,
|
||||||
|
"brightness": 0.5014,
|
||||||
|
"burnIn": 0.4983,
|
||||||
|
"chromaColor": 0,
|
||||||
|
"contrast": 0.7473,
|
||||||
|
"flickering": 0.9,
|
||||||
|
"fontColor": "#00ff3e",
|
||||||
|
"fontName": "COMMODORE_PET",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.3,
|
||||||
|
"horizontalSync": 0.42,
|
||||||
|
"jitter": 0.4,
|
||||||
|
"rasterization": 1,
|
||||||
|
"rbgShift": 0.2969,
|
||||||
|
"saturationColor": 0,
|
||||||
|
"screenCurvature": 0.5,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.2969,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.5
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "IBM Dos"
|
text: "IBM DOS"
|
||||||
obj_string: '{"ambientLight":0.16,"backgroundColor":"#000000","bloom":0.4,"brightness":0.5,"flickering":0.07,"contrast":0.85,"fontName":"IBM_DOS","fontColor":"#ffffff","frameName":"SIMPLE_WHITE_FRAME","glowingLine":0.13,"horizontalSync":0,"jitter":0.16,"burnIn":0.3,"staticNoise":0.03,"rasterization":0,"screenCurvature":0.1,"windowOpacity":1,"chromaColor":1,"saturationColor":0,"rbgShift":0.35,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0.151,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.2969,
|
||||||
|
"brightness": 0.5,
|
||||||
|
"burnIn": 0.0469,
|
||||||
|
"chromaColor": 1,
|
||||||
|
"contrast": 0.85,
|
||||||
|
"flickering": 0.0955,
|
||||||
|
"fontColor": "#ffffff",
|
||||||
|
"fontName": "IBM_DOS",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.1545,
|
||||||
|
"horizontalSync": 0,
|
||||||
|
"jitter": 0.1545,
|
||||||
|
"rasterization": 0,
|
||||||
|
"rbgShift": 0.3524,
|
||||||
|
"saturationColor": 0,
|
||||||
|
"screenCurvature": 0.4,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.0503,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.2
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "IBM 3278"
|
text: "IBM 3278"
|
||||||
obj_string: '{"ambientLight":0.1,"backgroundColor":"#000000","bloom":0.15,"brightness":0.5,"flickering":0,"contrast":0.85,"fontName":"IBM_3278","fontColor":"#0ccc68","frameName":"SIMPLE_WHITE_FRAME","glowingLine":0,"horizontalSync":0,"jitter":0,"burnIn":0.6,"staticNoise":0,"rasterization":0,"screenCurvature":0.1,"windowOpacity":1,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0.1,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.2969,
|
||||||
|
"brightness": 0.5,
|
||||||
|
"burnIn": 0.6,
|
||||||
|
"chromaColor": 0,
|
||||||
|
"contrast": 0.85,
|
||||||
|
"flickering": 0,
|
||||||
|
"fontColor": "#0ccc68",
|
||||||
|
"fontName": "IBM_3278",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0,
|
||||||
|
"horizontalSync": 0,
|
||||||
|
"jitter": 0,
|
||||||
|
"rasterization": 0,
|
||||||
|
"rbgShift": 0,
|
||||||
|
"saturationColor": 0,
|
||||||
|
"screenCurvature": 0.2,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0,
|
||||||
|
"windowOpacity": 1,
|
||||||
|
"margin": 0.5,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0.1
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
text: "Transparent Green"
|
text: "Futuristic"
|
||||||
obj_string: '{"ambientLight":0.2,"backgroundColor":"#000000","bloom":0.45,"brightness":0.5,"flickering":0.20,"contrast":0.85,"fontName":"TERMINUS_SCALED","fontColor":"#0ccc68","frameName":"NO_FRAME","glowingLine":0.16,"horizontalSync":0.1,"jitter":0.20,"burnIn":0.25,"staticNoise":0.20,"rasterization":0,"screenCurvature":0.05,"windowOpacity":0.60,"chromaColor":0,"saturationColor":0,"rbgShift":0,"fontWidth":1.0,"useCustomCommand":false,"customCommand":""}'
|
obj_string: '{
|
||||||
|
"ambientLight": 0,
|
||||||
|
"backgroundColor": "#000000",
|
||||||
|
"bloom": 0.5017,
|
||||||
|
"brightness": 0.5014,
|
||||||
|
"burnIn": 0.0955,
|
||||||
|
"chromaColor": 1,
|
||||||
|
"contrast": 0.85,
|
||||||
|
"flickering": 0.2,
|
||||||
|
"fontColor": "#729fcf",
|
||||||
|
"fontName": "TERMINUS",
|
||||||
|
"fontWidth": 1,
|
||||||
|
"glowingLine": 0.1476,
|
||||||
|
"horizontalSync": 0,
|
||||||
|
"jitter": 0.099,
|
||||||
|
"rasterization": 0,
|
||||||
|
"rbgShift": 0,
|
||||||
|
"saturationColor": 0.4983,
|
||||||
|
"screenCurvature": 0,
|
||||||
|
"frameColor": "#ffffff",
|
||||||
|
"frameGloss": 0,
|
||||||
|
"staticNoise": 0.0955,
|
||||||
|
"windowOpacity": 0.7,
|
||||||
|
"margin": 0.1,
|
||||||
|
"blinkingCursor": false,
|
||||||
|
"frameMargin": 0
|
||||||
|
}'
|
||||||
builtin: true
|
builtin: true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -449,47 +729,51 @@ QtObject{
|
|||||||
function getProfileIndexByName(name) {
|
function getProfileIndexByName(name) {
|
||||||
for (var i = 0; i < profilesList.count; i++) {
|
for (var i = 0; i < profilesList.count; i++) {
|
||||||
if (profilesList.get(i).text === name)
|
if (profilesList.get(i).text === name)
|
||||||
return i;
|
return i
|
||||||
}
|
}
|
||||||
return -1;
|
return -1
|
||||||
}
|
}
|
||||||
|
|
||||||
Component.onCompleted: {
|
Component.onCompleted: {
|
||||||
// Manage the arguments from the QML side.
|
// Manage the arguments from the QML side.
|
||||||
var args = Qt.application.arguments;
|
var args = Qt.application.arguments
|
||||||
if (args.indexOf("--verbose") !== -1) {
|
if (args.indexOf("--verbose") !== -1) {
|
||||||
verbose = true;
|
verbose = true
|
||||||
}
|
}
|
||||||
if (args.indexOf("--default-settings") === -1) {
|
if (args.indexOf("--default-settings") === -1) {
|
||||||
loadSettings();
|
loadSettings()
|
||||||
}
|
}
|
||||||
|
|
||||||
loadCustomProfiles();
|
loadCustomProfiles()
|
||||||
|
|
||||||
var profileArgPosition = args.indexOf("--profile");
|
var profileArgPosition = args.indexOf("--profile")
|
||||||
if (profileArgPosition !== -1) {
|
if (profileArgPosition !== -1) {
|
||||||
var profileIndex = getProfileIndexByName(args[profileArgPosition + 1]);
|
var profileIndex = getProfileIndexByName(
|
||||||
|
args[profileArgPosition + 1])
|
||||||
if (profileIndex !== -1)
|
if (profileIndex !== -1)
|
||||||
loadProfile(profileIndex);
|
loadProfile(profileIndex)
|
||||||
else
|
else
|
||||||
console.log("Warning: selected profile is not valid; ignoring it");
|
console.log("Warning: selected profile is not valid; ignoring it")
|
||||||
}
|
}
|
||||||
|
|
||||||
if (args.indexOf("--fullscreen") !== -1) {
|
if (args.indexOf("--fullscreen") !== -1) {
|
||||||
fullscreen = true;
|
fullscreen = true
|
||||||
showMenubar = false;
|
showMenubar = false
|
||||||
}
|
}
|
||||||
|
|
||||||
initializedSettings();
|
if (args.indexOf("-T") !== -1) {
|
||||||
|
wintitle = args[args.indexOf("-T") + 1]
|
||||||
|
}
|
||||||
|
|
||||||
|
initializedSettings()
|
||||||
}
|
}
|
||||||
Component.onDestruction: {
|
Component.onDestruction: {
|
||||||
storeSettings();
|
storeSettings()
|
||||||
storeCustomProfiles();
|
storeCustomProfiles()
|
||||||
// storage.dropSettings(); //DROPS THE SETTINGS!.. REMEMBER TO DISABLE ONCE ENABLED!!
|
// storage.dropSettings(); //DROPS THE SETTINGS!.. REMEMBER TO DISABLE ONCE ENABLED!!
|
||||||
}
|
}
|
||||||
|
|
||||||
// VARS ///////////////////////////////////////////////////////////////////
|
// VARS ///////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
property Label _sampleLabel: Label {
|
property Label _sampleLabel: Label {
|
||||||
text: "100%"
|
text: "100%"
|
||||||
}
|
}
|
||||||
|
161
app/qml/BurnInEffect.qml
Normal file
161
app/qml/BurnInEffect.qml
Normal file
@ -0,0 +1,161 @@
|
|||||||
|
/*******************************************************************************
|
||||||
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
|
*
|
||||||
|
* This file is part of cool-retro-term.
|
||||||
|
*
|
||||||
|
* cool-retro-term 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 <http://www.gnu.org/licenses/>.
|
||||||
|
*******************************************************************************/
|
||||||
|
import QtQuick 2.0
|
||||||
|
|
||||||
|
import "utils.js" as Utils
|
||||||
|
|
||||||
|
Loader {
|
||||||
|
id: burnInEffect
|
||||||
|
|
||||||
|
property ShaderEffectSource source: item ? item.source : null
|
||||||
|
|
||||||
|
property real lastUpdate: 0
|
||||||
|
property real prevLastUpdate: 0
|
||||||
|
|
||||||
|
property real burnIn: appSettings.burnIn
|
||||||
|
property real burnInFadeTime: 1 / Utils.lint(_minBurnInFadeTime, _maxBurnInFadeTime, burnIn)
|
||||||
|
property real _minBurnInFadeTime: appSettings.minBurnInFadeTime
|
||||||
|
property real _maxBurnInFadeTime: appSettings.maxBurnInFadeTime
|
||||||
|
|
||||||
|
active: appSettings.burnIn !== 0
|
||||||
|
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
function completelyUpdate() {
|
||||||
|
let newTime = timeManager.time
|
||||||
|
if (newTime > lastUpdate) {
|
||||||
|
prevLastUpdate = lastUpdate
|
||||||
|
lastUpdate = newTime
|
||||||
|
}
|
||||||
|
|
||||||
|
item.source.scheduleUpdate()
|
||||||
|
}
|
||||||
|
|
||||||
|
function restartBlurSource() {
|
||||||
|
prevLastUpdate = timeManager.time
|
||||||
|
lastUpdate = prevLastUpdate
|
||||||
|
completelyUpdate()
|
||||||
|
}
|
||||||
|
|
||||||
|
sourceComponent: Item {
|
||||||
|
property alias source: burnInEffectSource
|
||||||
|
|
||||||
|
ShaderEffectSource {
|
||||||
|
id: burnInEffectSource
|
||||||
|
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
sourceItem: burnInShaderEffect
|
||||||
|
live: false
|
||||||
|
recursive: true
|
||||||
|
hideSource: true
|
||||||
|
wrapMode: ShaderEffectSource.ClampToEdge
|
||||||
|
|
||||||
|
format: ShaderEffectSource.RGBA
|
||||||
|
smooth: true
|
||||||
|
|
||||||
|
visible: false
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
target: kterminal
|
||||||
|
|
||||||
|
onImagePainted: {
|
||||||
|
completelyUpdate()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Restart blurred source settings change.
|
||||||
|
Connections {
|
||||||
|
target: appSettings
|
||||||
|
|
||||||
|
onBurnInChanged: {
|
||||||
|
burnInEffect.restartBlurSource()
|
||||||
|
}
|
||||||
|
|
||||||
|
onTerminalFontChanged: {
|
||||||
|
burnInEffect.restartBlurSource()
|
||||||
|
}
|
||||||
|
|
||||||
|
onRasterizationChanged: {
|
||||||
|
burnInEffect.restartBlurSource()
|
||||||
|
}
|
||||||
|
|
||||||
|
onBurnInQualityChanged: {
|
||||||
|
burnInEffect.restartBlurSource()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ShaderLibrary {
|
||||||
|
id: shaderLibrary
|
||||||
|
}
|
||||||
|
|
||||||
|
ShaderEffect {
|
||||||
|
id: burnInShaderEffect
|
||||||
|
|
||||||
|
property variant txt_source: kterminalSource
|
||||||
|
property variant burnInSource: burnInEffectSource
|
||||||
|
property real burnInTime: burnInFadeTime
|
||||||
|
property real lastUpdate: burnInEffect.lastUpdate
|
||||||
|
property real prevLastUpdate: burnInEffect.prevLastUpdate
|
||||||
|
|
||||||
|
anchors.fill: parent
|
||||||
|
blending: false
|
||||||
|
|
||||||
|
fragmentShader:
|
||||||
|
"#ifdef GL_ES
|
||||||
|
precision mediump float;
|
||||||
|
#endif\n" +
|
||||||
|
|
||||||
|
"uniform lowp float qt_Opacity;" +
|
||||||
|
"uniform lowp sampler2D txt_source;" +
|
||||||
|
|
||||||
|
"varying highp vec2 qt_TexCoord0;
|
||||||
|
|
||||||
|
uniform lowp sampler2D burnInSource;
|
||||||
|
uniform highp float burnInTime;
|
||||||
|
|
||||||
|
uniform highp float lastUpdate;
|
||||||
|
|
||||||
|
uniform highp float prevLastUpdate;" +
|
||||||
|
|
||||||
|
shaderLibrary.rgb2grey +
|
||||||
|
|
||||||
|
"void main() {
|
||||||
|
vec2 coords = qt_TexCoord0;
|
||||||
|
|
||||||
|
vec3 txtColor = texture2D(txt_source, coords).rgb;
|
||||||
|
vec4 accColor = texture2D(burnInSource, coords);
|
||||||
|
|
||||||
|
float prevMask = accColor.a;
|
||||||
|
float currMask = rgb2grey(txtColor);
|
||||||
|
|
||||||
|
highp float blurDecay = clamp((lastUpdate - prevLastUpdate) * burnInTime, 0.0, 1.0);
|
||||||
|
blurDecay = max(0.0, blurDecay - prevMask);
|
||||||
|
vec3 blurColor = accColor.rgb - vec3(blurDecay);
|
||||||
|
vec3 color = max(blurColor, txtColor);
|
||||||
|
|
||||||
|
gl_FragColor = vec4(color, currMask);
|
||||||
|
}
|
||||||
|
"
|
||||||
|
|
||||||
|
onStatusChanged: if (log) console.log(log) //Print warning messages
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,51 +0,0 @@
|
|||||||
import QtQuick 2.2
|
|
||||||
import QtQuick.Controls 1.1
|
|
||||||
|
|
||||||
MenuBar {
|
|
||||||
id: defaultMenuBar
|
|
||||||
property bool visible: true
|
|
||||||
Menu {
|
|
||||||
title: qsTr("File")
|
|
||||||
visible: defaultMenuBar.visible
|
|
||||||
MenuItem {action: quitAction}
|
|
||||||
}
|
|
||||||
Menu {
|
|
||||||
title: qsTr("Edit")
|
|
||||||
visible: defaultMenuBar.visible
|
|
||||||
MenuItem {action: copyAction}
|
|
||||||
MenuItem {action: pasteAction}
|
|
||||||
MenuSeparator{visible: Qt.platform.os !== "osx"}
|
|
||||||
MenuItem {action: showsettingsAction}
|
|
||||||
}
|
|
||||||
Menu{
|
|
||||||
title: qsTr("View")
|
|
||||||
visible: defaultMenuBar.visible
|
|
||||||
MenuItem {action: fullscreenAction; visible: fullscreenAction.enabled}
|
|
||||||
MenuItem {action: showMenubarAction; visible: showMenubarAction.enabled}
|
|
||||||
MenuSeparator{visible: showMenubarAction.enabled}
|
|
||||||
MenuItem {action: zoomIn}
|
|
||||||
MenuItem {action: zoomOut}
|
|
||||||
}
|
|
||||||
Menu{
|
|
||||||
id: profilesMenu
|
|
||||||
title: qsTr("Profiles")
|
|
||||||
visible: defaultMenuBar.visible
|
|
||||||
Instantiator{
|
|
||||||
model: appSettings.profilesList
|
|
||||||
delegate: MenuItem {
|
|
||||||
text: model.text
|
|
||||||
onTriggered: {
|
|
||||||
appSettings.loadProfileString(obj_string);
|
|
||||||
appSettings.handleFontChanged();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
onObjectAdded: profilesMenu.insertItem(index, object)
|
|
||||||
onObjectRemoved: profilesMenu.removeItem(object)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Menu{
|
|
||||||
title: qsTr("Help")
|
|
||||||
visible: defaultMenuBar.visible
|
|
||||||
MenuItem {action: showAboutAction}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,9 +17,8 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.0
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.1
|
||||||
|
|
||||||
import "Components"
|
import "Components"
|
||||||
@ -28,20 +27,19 @@ RowLayout {
|
|||||||
property alias name: check.text
|
property alias name: check.text
|
||||||
|
|
||||||
property double value
|
property double value
|
||||||
property alias min_value: slider.minimumValue
|
property alias min_value: slider.from
|
||||||
property alias max_value: slider.maximumValue
|
property alias max_value: slider.to
|
||||||
property alias stepSize: slider.stepSize
|
property alias stepSize: slider.stepSize
|
||||||
|
|
||||||
signal newValue(real newValue);
|
signal newValue(real newValue)
|
||||||
|
|
||||||
id: setting_component
|
id: setting_component
|
||||||
anchors.left: parent.left
|
Layout.fillWidth: true
|
||||||
anchors.right: parent.right
|
|
||||||
|
|
||||||
onValueChanged: {
|
onValueChanged: {
|
||||||
check.checked = !(value == 0);
|
check.checked = !(value == 0)
|
||||||
if (check.checked)
|
if (check.checked)
|
||||||
slider.value = value;
|
slider.value = value
|
||||||
}
|
}
|
||||||
|
|
||||||
CheckBox {
|
CheckBox {
|
||||||
@ -49,13 +47,13 @@ RowLayout {
|
|||||||
implicitWidth: 160
|
implicitWidth: 160
|
||||||
onClicked: {
|
onClicked: {
|
||||||
if (!checked) {
|
if (!checked) {
|
||||||
checked = false;
|
checked = false
|
||||||
slider.enabled = false;
|
slider.enabled = false
|
||||||
newValue(0);
|
newValue(0)
|
||||||
} else {
|
} else {
|
||||||
checked = true;
|
checked = true
|
||||||
newValue(slider.value);
|
newValue(slider.value)
|
||||||
slider.enabled = true;
|
slider.enabled = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -64,11 +62,11 @@ RowLayout {
|
|||||||
stepSize: parent.stepSize
|
stepSize: parent.stepSize
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
onValueChanged: {
|
onValueChanged: {
|
||||||
newValue(value);
|
newValue(value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SizedLabel {
|
SizedLabel {
|
||||||
anchors { top: parent.top; bottom: parent.bottom }
|
text: Math.round(
|
||||||
text: Math.round(((value - min_value) / (max_value - min_value)) * 100) + "%"
|
((value - min_value) / (max_value - min_value)) * 100) + "%"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,7 +17,6 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Dialogs 1.1
|
import QtQuick.Dialogs 1.1
|
||||||
|
|
||||||
@ -35,15 +34,14 @@ Item {
|
|||||||
visible: false
|
visible: false
|
||||||
|
|
||||||
//This is a workaround to a Qt 5.2 bug.
|
//This is a workaround to a Qt 5.2 bug.
|
||||||
onColorChanged: if (Qt.platform.os !== "osx") colorSelected(color)
|
onColorChanged: if (!appSettings.isMacOS) colorSelected(color)
|
||||||
onAccepted: if (Qt.platform.os === "osx") colorSelected(color)
|
onAccepted: if (appSettings.isMacOS) colorSelected(color)
|
||||||
}
|
}
|
||||||
Rectangle {
|
Rectangle {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
radius: 10
|
radius: 10
|
||||||
color: rootItem.color
|
color: rootItem.color
|
||||||
border.color: "black"
|
|
||||||
Glossy {}
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
anchors.margins: parent.height * 0.25
|
anchors.margins: parent.height * 0.25
|
||||||
@ -59,6 +57,6 @@ Item {
|
|||||||
}
|
}
|
||||||
MouseArea {
|
MouseArea {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
onClicked: colorDialog.visible = true;
|
onClicked: colorDialog.visible = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,19 +19,14 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
|
|
||||||
import QtQuick 2.0
|
import QtQuick 2.0
|
||||||
import QtQuick.Controls 1.0
|
import QtQuick.Controls 2.0
|
||||||
|
import QtQuick.Layouts 1.0
|
||||||
|
|
||||||
// This component is simply a label with a predefined size.
|
// This component is simply a label with a predefined size.
|
||||||
// Used to improve alignment.
|
// Used to improve alignment.
|
||||||
|
|
||||||
Item {
|
|
||||||
property alias text: textfield.text
|
|
||||||
width: appSettings.labelWidth
|
|
||||||
Label {
|
Label {
|
||||||
id: textfield
|
id: textfield
|
||||||
anchors { right: parent.right; verticalCenter: parent.verticalCenter }
|
Layout.minimumWidth: appSettings.labelWidth
|
||||||
}
|
width: appSettings.labelWidth
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,7 +17,6 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
@ -35,10 +34,19 @@ QtObject{
|
|||||||
ListElement {
|
ListElement {
|
||||||
name: "COMMODORE_PET"
|
name: "COMMODORE_PET"
|
||||||
text: "Commodore PET (1977)"
|
text: "Commodore PET (1977)"
|
||||||
source: "fonts/1977-commodore-pet/COMMODORE_PET.ttf"
|
source: "fonts/1977-commodore-pet/PetMe.ttf"
|
||||||
lineSpacing: 2
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
|
fontWidth: 0.8
|
||||||
|
}
|
||||||
|
ListElement {
|
||||||
|
name: "IBM_PC"
|
||||||
|
text: "IBM PC (1981)"
|
||||||
|
source: "fonts/1981-ibm-pc/PxPlus_IBM_BIOS.ttf"
|
||||||
|
lineSpacing: 3
|
||||||
|
pixelSize: 8
|
||||||
|
baseScaling: 3.5
|
||||||
fontWidth: 0.8
|
fontWidth: 0.8
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
@ -47,13 +55,13 @@ QtObject{
|
|||||||
source: "fonts/modern-proggy-tiny/ProggyTiny.ttf"
|
source: "fonts/modern-proggy-tiny/ProggyTiny.ttf"
|
||||||
lineSpacing: 1
|
lineSpacing: 1
|
||||||
pixelSize: 16
|
pixelSize: 16
|
||||||
baseScaling: 4.0
|
baseScaling: 3.3
|
||||||
fontWidth: 0.9
|
fontWidth: 0.9
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "TERMINUS_SCALED"
|
name: "TERMINUS_SCALED"
|
||||||
text: "Terminus (Modern)"
|
text: "Terminus (Modern)"
|
||||||
source: "fonts/modern-terminus/TerminusTTF-4.38.2.ttf"
|
source: "fonts/modern-terminus/TerminusTTF-4.46.0.ttf"
|
||||||
lineSpacing: 1
|
lineSpacing: 1
|
||||||
pixelSize: 12
|
pixelSize: 12
|
||||||
baseScaling: 3.0
|
baseScaling: 3.0
|
||||||
@ -74,25 +82,25 @@ QtObject{
|
|||||||
source: "fonts/1977-apple2/PrintChar21.ttf"
|
source: "fonts/1977-apple2/PrintChar21.ttf"
|
||||||
lineSpacing: 2
|
lineSpacing: 2
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
fontWidth: 0.9
|
fontWidth: 0.9
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "ATARI_400"
|
name: "ATARI_400"
|
||||||
text: "Atari 400-800 (1979)"
|
text: "Atari 400-800 (1979)"
|
||||||
source: "fonts/1979-atari-400-800/ATARI400800_original.TTF"
|
source: "fonts/1979-atari-400-800/AtariClassic-Regular.ttf"
|
||||||
lineSpacing: 3
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
fontWidth: 0.8
|
fontWidth: 0.8
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "COMMODORE_64"
|
name: "COMMODORE_64"
|
||||||
text: "Commodore 64 (1982)"
|
text: "Commodore 64 (1982)"
|
||||||
source: "fonts/1982-commodore64/C64_Pro_Mono_v1.0-STYLE.ttf"
|
source: "fonts/1982-commodore64/C64_Pro_Mono-STYLE.ttf"
|
||||||
lineSpacing: 3
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
fontWidth: 0.8
|
fontWidth: 0.8
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,7 +17,6 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
@ -35,25 +34,34 @@ QtObject{
|
|||||||
ListElement {
|
ListElement {
|
||||||
name: "COMMODORE_PET"
|
name: "COMMODORE_PET"
|
||||||
text: "Commodore PET (1977)"
|
text: "Commodore PET (1977)"
|
||||||
source: "fonts/1977-commodore-pet/COMMODORE_PET.ttf"
|
source: "fonts/1977-commodore-pet/PetMe.ttf"
|
||||||
lineSpacing: 2
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
fontWidth: 0.7
|
fontWidth: 0.7
|
||||||
}
|
}
|
||||||
|
ListElement {
|
||||||
|
name: "IBM_PC"
|
||||||
|
text: "IBM PC (1981)"
|
||||||
|
source: "fonts/1981-ibm-pc/PxPlus_IBM_BIOS.ttf"
|
||||||
|
lineSpacing: 3
|
||||||
|
pixelSize: 8
|
||||||
|
baseScaling: 3.5
|
||||||
|
fontWidth: 0.8
|
||||||
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "PROGGY_TINY"
|
name: "PROGGY_TINY"
|
||||||
text: "Proggy Tiny (Modern)"
|
text: "Proggy Tiny (Modern)"
|
||||||
source: "fonts/modern-proggy-tiny/ProggyTiny.ttf"
|
source: "fonts/modern-proggy-tiny/ProggyTiny.ttf"
|
||||||
lineSpacing: 1
|
lineSpacing: 1
|
||||||
pixelSize: 16
|
pixelSize: 16
|
||||||
baseScaling: 4.0
|
baseScaling: 3.3
|
||||||
fontWidth: 0.9
|
fontWidth: 0.9
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "TERMINUS_SCALED"
|
name: "TERMINUS_SCALED"
|
||||||
text: "Terminus (Modern)"
|
text: "Terminus (Modern)"
|
||||||
source: "fonts/modern-terminus/TerminusTTF-4.38.2.ttf"
|
source: "fonts/modern-terminus/TerminusTTF-4.46.0.ttf"
|
||||||
lineSpacing: 1
|
lineSpacing: 1
|
||||||
pixelSize: 12
|
pixelSize: 12
|
||||||
baseScaling: 3.0
|
baseScaling: 3.0
|
||||||
@ -72,27 +80,27 @@ QtObject{
|
|||||||
name: "APPLE_II"
|
name: "APPLE_II"
|
||||||
text: "Apple ][ (1977)"
|
text: "Apple ][ (1977)"
|
||||||
source: "fonts/1977-apple2/PrintChar21.ttf"
|
source: "fonts/1977-apple2/PrintChar21.ttf"
|
||||||
lineSpacing: 2
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
fontWidth: 0.8
|
fontWidth: 0.8
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "ATARI_400"
|
name: "ATARI_400"
|
||||||
text: "Atari 400-800 (1979)"
|
text: "Atari 400-800 (1979)"
|
||||||
source: "fonts/1979-atari-400-800/ATARI400800_original.TTF"
|
source: "fonts/1979-atari-400-800/AtariClassic-Regular.ttf"
|
||||||
lineSpacing: 3
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
fontWidth: 0.7
|
fontWidth: 0.7
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "COMMODORE_64"
|
name: "COMMODORE_64"
|
||||||
text: "Commodore 64 (1982)"
|
text: "Commodore 64 (1982)"
|
||||||
source: "fonts/1982-commodore64/C64_Pro_Mono_v1.0-STYLE.ttf"
|
source: "fonts/1982-commodore64/C64_Pro_Mono-STYLE.ttf"
|
||||||
lineSpacing: 3
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 4.0
|
baseScaling: 3.5
|
||||||
fontWidth: 0.7
|
fontWidth: 0.7
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,7 +17,6 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
@ -27,20 +26,18 @@ QtObject{
|
|||||||
property var _font: fontlist.get(selectedFontIndex)
|
property var _font: fontlist.get(selectedFontIndex)
|
||||||
property bool lowResolutionFont: _font.lowResolutionFont
|
property bool lowResolutionFont: _font.lowResolutionFont
|
||||||
|
|
||||||
property int pixelSize: lowResolutionFont
|
property int pixelSize: lowResolutionFont ? _font.pixelSize : _font.pixelSize * scaling
|
||||||
? _font.pixelSize
|
|
||||||
: _font.pixelSize * scaling
|
|
||||||
|
|
||||||
property int lineSpacing: lowResolutionFont
|
property int lineSpacing: lowResolutionFont ? _font.lineSpacing : pixelSize * _font.lineSpacing
|
||||||
? _font.lineSpacing
|
|
||||||
: pixelSize * _font.lineSpacing
|
|
||||||
|
|
||||||
property real screenScaling: lowResolutionFont
|
property real screenScaling: lowResolutionFont ? _font.baseScaling * scaling : 1.0
|
||||||
? _font.baseScaling * scaling
|
|
||||||
: 1.0
|
|
||||||
|
|
||||||
property real defaultFontWidth: fontlist.get(selectedFontIndex).fontWidth
|
property real defaultFontWidth: fontlist.get(selectedFontIndex).fontWidth
|
||||||
|
|
||||||
|
property string family: fontlist.get(selectedFontIndex).family
|
||||||
|
|
||||||
|
property bool isSystemFont: fontlist.get(selectedFontIndex).isSystemFont
|
||||||
|
|
||||||
// There are two kind of fonts: low resolution and high resolution.
|
// There are two kind of fonts: low resolution and high resolution.
|
||||||
// Low resolution font sets the lowResolutionFont property to true.
|
// Low resolution font sets the lowResolutionFont property to true.
|
||||||
// They are rendered at a fixed pixel size and the texture is upscaled
|
// They are rendered at a fixed pixel size and the texture is upscaled
|
||||||
@ -48,17 +45,18 @@ QtObject{
|
|||||||
// High resolution fonts are instead drawn on a texture which has the
|
// High resolution fonts are instead drawn on a texture which has the
|
||||||
// size of the screen, and the scaling directly controls their pixels size.
|
// size of the screen, and the scaling directly controls their pixels size.
|
||||||
// Those are slower to render but are not pixelated.
|
// Those are slower to render but are not pixelated.
|
||||||
|
|
||||||
property ListModel fontlist: ListModel {
|
property ListModel fontlist: ListModel {
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "TERMINUS_SCALED"
|
name: "TERMINUS_SCALED"
|
||||||
text: "Terminus (Modern)"
|
text: "Terminus (Modern)"
|
||||||
source: "fonts/modern-terminus/TerminusTTF-4.38.2.ttf"
|
source: "fonts/modern-terminus/TerminusTTF-4.46.0.ttf"
|
||||||
lineSpacing: 1
|
lineSpacing: 1
|
||||||
pixelSize: 12
|
pixelSize: 12
|
||||||
baseScaling: 3.0
|
baseScaling: 3.0
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "PRO_FONT_SCALED"
|
name: "PRO_FONT_SCALED"
|
||||||
@ -69,6 +67,8 @@ QtObject{
|
|||||||
baseScaling: 3.0
|
baseScaling: 3.0
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "EXCELSIOR_SCALED"
|
name: "EXCELSIOR_SCALED"
|
||||||
@ -79,16 +79,20 @@ QtObject{
|
|||||||
baseScaling: 2.4
|
baseScaling: 2.4
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "COMMODORE_PET_SCALED"
|
name: "COMMODORE_PET_SCALED"
|
||||||
text: "Commodore PET (1977)"
|
text: "Commodore PET (1977)"
|
||||||
source: "fonts/1977-commodore-pet/COMMODORE_PET.ttf"
|
source: "fonts/1977-commodore-pet/PetMe.ttf"
|
||||||
lineSpacing: 2
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 3.5
|
baseScaling: 3.5
|
||||||
fontWidth: 0.7
|
fontWidth: 0.7
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "PROGGY_TINY_SCALED"
|
name: "PROGGY_TINY_SCALED"
|
||||||
@ -96,77 +100,93 @@ QtObject{
|
|||||||
source: "fonts/modern-proggy-tiny/ProggyTiny.ttf"
|
source: "fonts/modern-proggy-tiny/ProggyTiny.ttf"
|
||||||
lineSpacing: 1
|
lineSpacing: 1
|
||||||
pixelSize: 16
|
pixelSize: 16
|
||||||
baseScaling: 3.0
|
baseScaling: 3.3
|
||||||
fontWidth: 0.9
|
fontWidth: 0.9
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "APPLE_II_SCALED"
|
name: "APPLE_II_SCALED"
|
||||||
text: "Apple ][ (1977)"
|
text: "Apple ][ (1977)"
|
||||||
source: "fonts/1977-apple2/PrintChar21.ttf"
|
source: "fonts/1977-apple2/PrintChar21.ttf"
|
||||||
lineSpacing: 2
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 3.5
|
baseScaling: 3.5
|
||||||
fontWidth: 0.8
|
fontWidth: 0.8
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "ATARI_400_SCALED"
|
name: "ATARI_400_SCALED"
|
||||||
text: "Atari 400-800 (1979)"
|
text: "Atari 400-800 (1979)"
|
||||||
source: "fonts/1979-atari-400-800/ATARI400800_original.TTF"
|
source: "fonts/1979-atari-400-800/AtariClassic-Regular.ttf"
|
||||||
lineSpacing: 3
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 3.5
|
baseScaling: 3.5
|
||||||
fontWidth: 0.7
|
fontWidth: 0.7
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
|
}
|
||||||
|
ListElement {
|
||||||
|
name: "IBM_PC_SCALED"
|
||||||
|
text: "IBM PC (1981)"
|
||||||
|
source: "fonts/1981-ibm-pc/PxPlus_IBM_BIOS.ttf"
|
||||||
|
lineSpacing: 3
|
||||||
|
pixelSize: 8
|
||||||
|
baseScaling: 3.5
|
||||||
|
fontWidth: 0.8
|
||||||
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "COMMODORE_64_SCALED"
|
name: "COMMODORE_64_SCALED"
|
||||||
text: "Commodore 64 (1982)"
|
text: "Commodore 64 (1982)"
|
||||||
source: "fonts/1982-commodore64/C64_Pro_Mono_v1.0-STYLE.ttf"
|
source: "fonts/1982-commodore64/C64_Pro_Mono-STYLE.ttf"
|
||||||
lineSpacing: 3
|
lineSpacing: 3
|
||||||
pixelSize: 8
|
pixelSize: 8
|
||||||
baseScaling: 3.5
|
baseScaling: 3.5
|
||||||
fontWidth: 0.7
|
fontWidth: 0.7
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
}
|
isSystemFont: false
|
||||||
ListElement{
|
family: ""
|
||||||
name: "ATARI_ST_SCALED"
|
|
||||||
text: "Atari ST (1985)"
|
|
||||||
source: "fonts/1985-atari-st/AtariST8x16SystemFont.ttf"
|
|
||||||
lineSpacing: 3
|
|
||||||
pixelSize: 16
|
|
||||||
baseScaling: 2.0
|
|
||||||
fontWidth: 1.0
|
|
||||||
lowResolutionFont: true
|
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "IBM_DOS"
|
name: "IBM_DOS"
|
||||||
text: "IBM DOS (1985)"
|
text: "IBM DOS (1985)"
|
||||||
source: "fonts/1985-ibm-pc-vga/Perfect DOS VGA 437 Win.ttf"
|
source: "fonts/1985-ibm-pc-vga/PxPlus_IBM_VGA8.ttf"
|
||||||
lineSpacing: 3
|
lineSpacing: 3
|
||||||
pixelSize: 16
|
pixelSize: 16
|
||||||
baseScaling: 2.0
|
baseScaling: 2.0
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: true
|
lowResolutionFont: true
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "HERMIT"
|
name: "HERMIT"
|
||||||
text: "HD: Hermit (Modern)"
|
text: "HD: Hermit (Modern)"
|
||||||
source: "fonts/modern-hermit/Hermit-medium.otf"
|
source: "fonts/modern-hermit/Hermit-medium.otf"
|
||||||
lineSpacing: 0.05
|
lineSpacing: 0.05
|
||||||
pixelSize: 28
|
pixelSize: 27
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: false
|
lowResolutionFont: false
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "TERMINUS"
|
name: "TERMINUS"
|
||||||
text: "HD: Terminus (Modern)"
|
text: "HD: Terminus (Modern)"
|
||||||
source: "fonts/modern-terminus/TerminusTTF-4.38.2.ttf"
|
source: "fonts/modern-terminus/TerminusTTF-4.46.0.ttf"
|
||||||
lineSpacing: 0.1
|
lineSpacing: 0.1
|
||||||
pixelSize: 35
|
pixelSize: 35
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: false
|
lowResolutionFont: false
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "PRO_FONT"
|
name: "PRO_FONT"
|
||||||
@ -176,15 +196,8 @@ QtObject{
|
|||||||
pixelSize: 35
|
pixelSize: 35
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: false
|
lowResolutionFont: false
|
||||||
}
|
isSystemFont: false
|
||||||
ListElement{
|
family: ""
|
||||||
name: "MONACO"
|
|
||||||
text: "HD: Monaco (Modern)"
|
|
||||||
source: "fonts/modern-monaco/monaco.ttf"
|
|
||||||
lineSpacing: 0.1
|
|
||||||
pixelSize: 30
|
|
||||||
fontWidth: 1.0
|
|
||||||
lowResolutionFont: false
|
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "INCONSOLATA"
|
name: "INCONSOLATA"
|
||||||
@ -194,15 +207,46 @@ QtObject{
|
|||||||
pixelSize: 35
|
pixelSize: 35
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: false
|
lowResolutionFont: false
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
name: "IBM_3278"
|
name: "IBM_3278"
|
||||||
text: "HD: IBM 3278 (1971)"
|
text: "HD: IBM 3278 (1971)"
|
||||||
source: "fonts/1971-ibm-3278/3270Medium.ttf"
|
source: "fonts/1971-ibm-3278/3270-Regular.ttf"
|
||||||
lineSpacing: 0.2
|
lineSpacing: 0.2
|
||||||
pixelSize: 32
|
pixelSize: 32
|
||||||
fontWidth: 1.0
|
fontWidth: 1.0
|
||||||
lowResolutionFont: false
|
lowResolutionFont: false
|
||||||
|
isSystemFont: false
|
||||||
|
family: ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Component.onCompleted: addSystemFonts()
|
||||||
|
|
||||||
|
function addSystemFonts() {
|
||||||
|
var families = monospaceSystemFonts
|
||||||
|
for (var i = 0; i < families.length; i++) {
|
||||||
|
if (verbose) {
|
||||||
|
console.log("Adding system font: ", families[i])
|
||||||
|
}
|
||||||
|
fontlist.append(convertToListElement(families[i]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function convertToListElement(family) {
|
||||||
|
return {
|
||||||
|
"name": "System: " + family,
|
||||||
|
"text": qsTr("System: ") + family,
|
||||||
|
"source": "",
|
||||||
|
"lineSpacing": 0.1,
|
||||||
|
"pixelSize": 30,
|
||||||
|
"fontWidth": 1.0,
|
||||||
|
"baseScaling": 1.0,
|
||||||
|
"lowResolutionFont": false,
|
||||||
|
"isSystemFont": true,
|
||||||
|
"family": family
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
import QtQuick 2.2
|
|
||||||
|
|
||||||
Rectangle {
|
|
||||||
anchors.centerIn: parent
|
|
||||||
width: parent.width - parent.border.width
|
|
||||||
height: parent.height - parent.border.width
|
|
||||||
radius:parent.radius - parent.border.width/2
|
|
||||||
smooth: true
|
|
||||||
|
|
||||||
border.width: parent.border.width/2
|
|
||||||
border.color: "#22FFFFFF"
|
|
||||||
|
|
||||||
gradient: Gradient {
|
|
||||||
GradientStop { position: 0; color: "#88FFFFFF" }
|
|
||||||
GradientStop { position: .1; color: "#55FFFFFF" }
|
|
||||||
GradientStop { position: .5; color: "#33FFFFFF" }
|
|
||||||
GradientStop { position: .501; color: "#11000000" }
|
|
||||||
GradientStop { position: .8; color: "#11FFFFFF" }
|
|
||||||
GradientStop { position: 1; color: "#55FFFFFF" }
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,10 +17,9 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Window 2.0
|
import QtQuick.Window 2.0
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.0
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.1
|
||||||
import QtQuick.Dialogs 1.1
|
import QtQuick.Dialogs 1.1
|
||||||
|
|
||||||
@ -40,23 +39,25 @@ Window{
|
|||||||
visible: false
|
visible: false
|
||||||
|
|
||||||
function showError(message) {
|
function showError(message) {
|
||||||
text = message;
|
text = message
|
||||||
open();
|
open()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function validateName(name) {
|
function validateName(name) {
|
||||||
var profile_list = appSettings.profilesList;
|
var profile_list = appSettings.profilesList
|
||||||
if (name === "")
|
if (name === "")
|
||||||
return 1;
|
return 1
|
||||||
return 0;
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
anchors.margins: 10
|
anchors.margins: 10
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
RowLayout {
|
RowLayout {
|
||||||
Label{text: qsTr("Name")}
|
Label {
|
||||||
|
text: qsTr("Name")
|
||||||
|
}
|
||||||
TextField {
|
TextField {
|
||||||
id: namefield
|
id: namefield
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
@ -65,21 +66,21 @@ Window{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
RowLayout {
|
RowLayout {
|
||||||
anchors.right: parent.right
|
Layout.alignment: Qt.AlignBottom | Qt.AlignRight
|
||||||
anchors.bottom: parent.bottom
|
|
||||||
Button {
|
Button {
|
||||||
id: okbutton
|
id: okbutton
|
||||||
text: qsTr("OK")
|
text: qsTr("OK")
|
||||||
onClicked: clickAction()
|
onClicked: clickAction()
|
||||||
function clickAction() {
|
function clickAction() {
|
||||||
var name = namefield.text;
|
var name = namefield.text
|
||||||
switch (validateName(name)) {
|
switch (validateName(name)) {
|
||||||
case 1:
|
case 1:
|
||||||
errorDialog.showError(qsTr("The name you inserted is empty. Please choose a different one."));
|
errorDialog.showError(
|
||||||
break;
|
qsTr("The name you inserted is empty. Please choose a different one."))
|
||||||
|
break
|
||||||
default:
|
default:
|
||||||
nameSelected(name);
|
nameSelected(name)
|
||||||
close();
|
close()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -19,77 +19,115 @@
|
|||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.0
|
||||||
|
|
||||||
import QMLTermWidget 1.0
|
import QMLTermWidget 1.0
|
||||||
|
|
||||||
|
import "menus"
|
||||||
import "utils.js" as Utils
|
import "utils.js" as Utils
|
||||||
|
|
||||||
Item{
|
Item{
|
||||||
id: terminalContainer
|
id: terminalContainer
|
||||||
|
|
||||||
property size virtualResolution: Qt.size(kterminal.width, kterminal.height)
|
property size virtualResolution: Qt.size(kterminal.totalWidth, kterminal.totalHeight)
|
||||||
property alias mainTerminal: kterminal
|
property alias mainTerminal: kterminal
|
||||||
property ShaderEffectSource mainSource: kterminalSource
|
|
||||||
property ShaderEffectSource blurredSource: blurredSourceLoader.item
|
|
||||||
|
|
||||||
|
property ShaderEffectSource mainSource: kterminalSource
|
||||||
|
property BurnInEffect burnInEffect: burnInEffect
|
||||||
property real fontWidth: 1.0
|
property real fontWidth: 1.0
|
||||||
property real screenScaling: 1.0
|
property real screenScaling: 1.0
|
||||||
property real scaleTexture: 1.0
|
property real scaleTexture: 1.0
|
||||||
property alias title: ksession.title
|
property alias title: ksession.title
|
||||||
property alias kterminal: kterminal
|
property alias kterminal: kterminal
|
||||||
|
|
||||||
anchors.leftMargin: frame.displacementLeft * appSettings.windowScaling
|
|
||||||
anchors.rightMargin: frame.displacementRight * appSettings.windowScaling
|
|
||||||
anchors.topMargin: frame.displacementTop * appSettings.windowScaling
|
|
||||||
anchors.bottomMargin: frame.displacementBottom * appSettings.windowScaling
|
|
||||||
|
|
||||||
//Parameters for the burnIn effect.
|
|
||||||
property real burnIn: appSettings.burnIn
|
|
||||||
property real fps: appSettings.fps !== 0 ? appSettings.fps : 60
|
|
||||||
property real burnInFadeTime: Utils.lint(_minBurnInFadeTime, _maxBurnInFadeTime, burnIn)
|
|
||||||
property real motionBlurCoefficient: 1.0 / (fps * burnInFadeTime)
|
|
||||||
property real _minBurnInFadeTime: 0.16
|
|
||||||
property real _maxBurnInFadeTime: 1.6
|
|
||||||
|
|
||||||
property size terminalSize: kterminal.terminalSize
|
property size terminalSize: kterminal.terminalSize
|
||||||
property size fontMetrics: kterminal.fontMetrics
|
property size fontMetrics: kterminal.fontMetrics
|
||||||
|
|
||||||
// Manage copy and paste
|
// Manage copy and paste
|
||||||
Connections {
|
Connections {
|
||||||
target: copyAction
|
target: copyAction
|
||||||
onTriggered: kterminal.copyClipboard();
|
|
||||||
|
onTriggered: {
|
||||||
|
kterminal.copyClipboard()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Connections {
|
Connections {
|
||||||
target: pasteAction
|
target: pasteAction
|
||||||
onTriggered: kterminal.pasteClipboard()
|
|
||||||
|
onTriggered: {
|
||||||
|
kterminal.pasteClipboard()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Connections {
|
||||||
|
target: pasteActionAlt
|
||||||
|
|
||||||
|
onTriggered: {
|
||||||
|
kterminal.pasteSelection()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//When settings are updated sources need to be redrawn.
|
//When settings are updated sources need to be redrawn.
|
||||||
Connections {
|
Connections {
|
||||||
target: appSettings
|
target: appSettings
|
||||||
onFontScalingChanged: terminalContainer.updateSources();
|
|
||||||
onFontWidthChanged: terminalContainer.updateSources();
|
onFontScalingChanged: {
|
||||||
|
terminalContainer.updateSources()
|
||||||
|
}
|
||||||
|
|
||||||
|
onFontWidthChanged: {
|
||||||
|
terminalContainer.updateSources()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Connections {
|
Connections {
|
||||||
target: terminalContainer
|
target: terminalContainer
|
||||||
onWidthChanged: terminalContainer.updateSources();
|
|
||||||
onHeightChanged: terminalContainer.updateSources();
|
onWidthChanged: {
|
||||||
|
terminalContainer.updateSources()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onHeightChanged: {
|
||||||
|
terminalContainer.updateSources()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Connections {
|
||||||
|
target: terminalWindow
|
||||||
|
|
||||||
|
onActiveChanged: {
|
||||||
|
kterminal.forceActiveFocus()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function updateSources() {
|
function updateSources() {
|
||||||
kterminal.update();
|
kterminal.update()
|
||||||
}
|
}
|
||||||
|
|
||||||
QMLTermWidget {
|
QMLTermWidget {
|
||||||
id: kterminal
|
id: kterminal
|
||||||
width: Math.floor(parent.width / (screenScaling * fontWidth))
|
|
||||||
height: Math.floor(parent.height / screenScaling)
|
property int textureResolutionScale: appSettings.lowResolutionFont ? devicePixelRatio : 1
|
||||||
|
property int margin: appSettings.totalMargin / screenScaling
|
||||||
|
property int totalWidth: Math.floor(parent.width / (screenScaling * fontWidth))
|
||||||
|
property int totalHeight: Math.floor(parent.height / screenScaling)
|
||||||
|
|
||||||
|
property int rawWidth: totalWidth - 2 * margin
|
||||||
|
property int rawHeight: totalHeight - 2 * margin
|
||||||
|
|
||||||
|
textureSize: Qt.size(width / textureResolutionScale, height / textureResolutionScale)
|
||||||
|
|
||||||
|
width: ensureMultiple(rawWidth, devicePixelRatio)
|
||||||
|
height: ensureMultiple(rawHeight, devicePixelRatio)
|
||||||
|
|
||||||
|
/** Ensure size is a multiple of factor. This is needed for pixel perfect scaling on highdpi screens. */
|
||||||
|
function ensureMultiple(size, factor) {
|
||||||
|
return Math.round(size / factor) * factor;
|
||||||
|
}
|
||||||
|
|
||||||
colorScheme: "cool-retro-term"
|
colorScheme: "cool-retro-term"
|
||||||
|
|
||||||
smooth: !appSettings.lowResolutionFont
|
smooth: !appSettings.lowResolutionFont
|
||||||
enableBold: false
|
enableBold: false
|
||||||
fullCursorHeight: true
|
fullCursorHeight: true
|
||||||
|
blinkingCursor: appSettings.blinkingCursor
|
||||||
|
|
||||||
session: QMLTermSession {
|
session: QMLTermSession {
|
||||||
id: ksession
|
id: ksession
|
||||||
@ -114,14 +152,10 @@ Item{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
FontLoader{ id: fontLoader }
|
function handleFontChanged(fontFamily, pixelSize, lineSpacing, screenScaling, fontWidth) {
|
||||||
|
|
||||||
function handleFontChange(fontSource, pixelSize, lineSpacing, screenScaling, fontWidth){
|
|
||||||
fontLoader.source = fontSource;
|
|
||||||
|
|
||||||
kterminal.antialiasText = !appSettings.lowResolutionFont;
|
kterminal.antialiasText = !appSettings.lowResolutionFont;
|
||||||
font.pixelSize = pixelSize;
|
font.pixelSize = pixelSize;
|
||||||
font.family = fontLoader.name;
|
font.family = fontFamily;
|
||||||
|
|
||||||
terminalContainer.fontWidth = fontWidth;
|
terminalContainer.fontWidth = fontWidth;
|
||||||
terminalContainer.screenScaling = screenScaling;
|
terminalContainer.screenScaling = screenScaling;
|
||||||
@ -129,6 +163,7 @@ Item{
|
|||||||
|
|
||||||
kterminal.lineSpacing = lineSpacing;
|
kterminal.lineSpacing = lineSpacing;
|
||||||
}
|
}
|
||||||
|
|
||||||
function startSession() {
|
function startSession() {
|
||||||
appSettings.initializedSettings.disconnect(startSession);
|
appSettings.initializedSettings.disconnect(startSession);
|
||||||
|
|
||||||
@ -140,7 +175,7 @@ Item{
|
|||||||
var args = Utils.tokenizeCommandLine(appSettings.customCommand);
|
var args = Utils.tokenizeCommandLine(appSettings.customCommand);
|
||||||
ksession.setShellProgram(args[0]);
|
ksession.setShellProgram(args[0]);
|
||||||
ksession.setArgs(args.slice(1));
|
ksession.setArgs(args.slice(1));
|
||||||
} else if (!defaultCmd && Qt.platform.os === "osx") {
|
} else if (!defaultCmd && appSettings.isMacOS) {
|
||||||
// OSX Requires the following default parameters for auto login.
|
// OSX Requires the following default parameters for auto login.
|
||||||
ksession.setArgs(["-i", "-l"]);
|
ksession.setArgs(["-i", "-l"]);
|
||||||
}
|
}
|
||||||
@ -152,38 +187,31 @@ Item{
|
|||||||
forceActiveFocus();
|
forceActiveFocus();
|
||||||
}
|
}
|
||||||
Component.onCompleted: {
|
Component.onCompleted: {
|
||||||
appSettings.terminalFontChanged.connect(handleFontChange);
|
appSettings.terminalFontChanged.connect(handleFontChanged);
|
||||||
appSettings.initializedSettings.connect(startSession);
|
appSettings.initializedSettings.connect(startSession);
|
||||||
|
appSettings.handleFontChanged()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Component {
|
Component {
|
||||||
id: linuxContextMenu
|
id: shortContextMenu
|
||||||
Menu{
|
ShortContextMenu { }
|
||||||
id: contextmenu
|
|
||||||
MenuItem{action: copyAction}
|
|
||||||
MenuItem{action: pasteAction}
|
|
||||||
MenuSeparator{}
|
|
||||||
MenuItem{action: fullscreenAction}
|
|
||||||
MenuItem{action: showMenubarAction}
|
|
||||||
MenuSeparator{visible: !appSettings.showMenubar}
|
|
||||||
CRTMainMenuBar{visible: !appSettings.showMenubar}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Component {
|
Component {
|
||||||
id: osxContextMenu
|
id: fullContextMenu
|
||||||
Menu{
|
FullContextMenu { }
|
||||||
id: contextmenu
|
|
||||||
MenuItem{action: copyAction}
|
|
||||||
MenuItem{action: pasteAction}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Loader {
|
Loader {
|
||||||
id: menuLoader
|
id: menuLoader
|
||||||
sourceComponent: (Qt.platform.os === "osx" ? osxContextMenu : linuxContextMenu)
|
sourceComponent: (appSettings.isMacOS || appSettings.showMenubar ? shortContextMenu : fullContextMenu)
|
||||||
}
|
}
|
||||||
property alias contextmenu: menuLoader.item
|
property alias contextmenu: menuLoader.item
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
|
property real margin: appSettings.totalMargin
|
||||||
|
|
||||||
acceptedButtons: Qt.LeftButton | Qt.MiddleButton | Qt.RightButton
|
acceptedButtons: Qt.LeftButton | Qt.MiddleButton | Qt.RightButton
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
cursorShape: kterminal.terminalUsesMouse ? Qt.ArrowCursor : Qt.IBeamCursor
|
cursorShape: kterminal.terminalUsesMouse ? Qt.ArrowCursor : Qt.IBeamCursor
|
||||||
@ -217,135 +245,42 @@ Item{
|
|||||||
}
|
}
|
||||||
|
|
||||||
function correctDistortion(x, y){
|
function correctDistortion(x, y){
|
||||||
x = x / width;
|
x = (x - margin) / width;
|
||||||
y = y / height;
|
y = (y - margin) / height;
|
||||||
|
|
||||||
var cc = Qt.size(0.5 - x, 0.5 - y);
|
var cc = Qt.size(0.5 - x, 0.5 - y);
|
||||||
var distortion = (cc.height * cc.height + cc.width * cc.width) * appSettings.screenCurvature;
|
var distortion = (cc.height * cc.height + cc.width * cc.width) * appSettings.screenCurvature * appSettings.screenCurvatureSize;
|
||||||
|
|
||||||
return Qt.point((x - cc.width * (1+distortion) * distortion) * kterminal.width,
|
return Qt.point((x - cc.width * (1+distortion) * distortion) * (kterminal.totalWidth),
|
||||||
(y - cc.height * (1+distortion) * distortion) * kterminal.height)
|
(y - cc.height * (1+distortion) * distortion) * (kterminal.totalHeight))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ShaderEffectSource{
|
ShaderEffectSource{
|
||||||
id: kterminalSource
|
id: kterminalSource
|
||||||
sourceItem: kterminal
|
sourceItem: kterminal
|
||||||
hideSource: true
|
hideSource: true
|
||||||
wrapMode: ShaderEffectSource.ClampToEdge
|
wrapMode: ShaderEffectSource.Repeat
|
||||||
visible: false
|
visible: false
|
||||||
textureSize: Qt.size(kterminal.width * scaleTexture, kterminal.height * scaleTexture);
|
textureSize: Qt.size(kterminal.totalWidth * scaleTexture, kterminal.totalHeight * scaleTexture)
|
||||||
}
|
sourceRect: Qt.rect(-kterminal.margin, -kterminal.margin, kterminal.totalWidth, kterminal.totalHeight)
|
||||||
Loader{
|
|
||||||
id: blurredSourceLoader
|
|
||||||
asynchronous: true
|
|
||||||
active: burnIn !== 0
|
|
||||||
|
|
||||||
sourceComponent: ShaderEffectSource{
|
|
||||||
property bool updateBurnIn: false
|
|
||||||
|
|
||||||
id: _blurredSourceEffect
|
|
||||||
sourceItem: blurredTerminalLoader.item
|
|
||||||
recursive: true
|
|
||||||
live: false
|
|
||||||
hideSource: true
|
|
||||||
wrapMode: kterminalSource.wrapMode
|
|
||||||
|
|
||||||
visible: false
|
|
||||||
|
|
||||||
function restartBlurSource(){
|
|
||||||
livetimer.restart();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// This updates the burnin synched with the timer.
|
Item {
|
||||||
Connections {
|
id: burnInContainer
|
||||||
target: updateBurnIn ? mainShader : null
|
|
||||||
ignoreUnknownSignals: false
|
|
||||||
onTimeChanged: _blurredSourceEffect.scheduleUpdate();
|
|
||||||
}
|
|
||||||
|
|
||||||
Timer{
|
|
||||||
id: livetimer
|
|
||||||
|
|
||||||
// The interval assumes 60 fps. This is the time needed burnout a white pixel.
|
|
||||||
// We multiply 1.1 to have a little bit of margin over the theoretical value.
|
|
||||||
// This solution is not extremely clean, but it's probably the best to avoid measuring fps.
|
|
||||||
|
|
||||||
interval: burnInFadeTime * 1000 * 1.1
|
|
||||||
running: true
|
|
||||||
onTriggered: _blurredSourceEffect.updateBurnIn = false;
|
|
||||||
}
|
|
||||||
Connections{
|
|
||||||
target: kterminal
|
|
||||||
onImagePainted:{
|
|
||||||
_blurredSourceEffect.scheduleUpdate();
|
|
||||||
_blurredSourceEffect.updateBurnIn = true;
|
|
||||||
livetimer.restart();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Restart blurred source settings change.
|
|
||||||
Connections{
|
|
||||||
target: appSettings
|
|
||||||
onBurnInChanged: _blurredSourceEffect.restartBlurSource();
|
|
||||||
onTerminalFontChanged: _blurredSourceEffect.restartBlurSource();
|
|
||||||
onRasterizationChanged: _blurredSourceEffect.restartBlurSource();
|
|
||||||
onBurnInQualityChanged: _blurredSourceEffect.restartBlurSource();
|
|
||||||
}
|
|
||||||
Connections {
|
|
||||||
target: kterminalScrollbar
|
|
||||||
onOpacityChanged: _blurredSourceEffect.restartBlurSource();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Loader{
|
|
||||||
id: blurredTerminalLoader
|
|
||||||
|
|
||||||
property int burnInScaling: scaleTexture * appSettings.burnInQuality
|
property int burnInScaling: scaleTexture * appSettings.burnInQuality
|
||||||
|
|
||||||
width: appSettings.lowResolutionFont
|
width: Math.round(appSettings.lowResolutionFont
|
||||||
? kterminal.width * Math.max(1, burnInScaling)
|
? kterminal.totalWidth * Math.max(1, burnInScaling)
|
||||||
: kterminal.width * scaleTexture * appSettings.burnInQuality
|
: kterminal.totalWidth * scaleTexture * appSettings.burnInQuality)
|
||||||
height: appSettings.lowResolutionFont
|
|
||||||
? kterminal.height * Math.max(1, burnInScaling)
|
|
||||||
: kterminal.height * scaleTexture * appSettings.burnInQuality
|
|
||||||
|
|
||||||
active: burnIn !== 0
|
height: Math.round(appSettings.lowResolutionFont
|
||||||
asynchronous: true
|
? kterminal.totalHeight * Math.max(1, burnInScaling)
|
||||||
|
: kterminal.totalHeight * scaleTexture * appSettings.burnInQuality)
|
||||||
|
|
||||||
sourceComponent: ShaderEffect {
|
|
||||||
property variant txt_source: kterminalSource
|
|
||||||
property variant blurredSource: blurredSourceLoader.item
|
|
||||||
property real blurCoefficient: motionBlurCoefficient
|
|
||||||
|
|
||||||
blending: false
|
BurnInEffect {
|
||||||
|
id: burnInEffect
|
||||||
fragmentShader:
|
|
||||||
"#ifdef GL_ES
|
|
||||||
precision mediump float;
|
|
||||||
#endif\n" +
|
|
||||||
|
|
||||||
"uniform lowp float qt_Opacity;" +
|
|
||||||
"uniform lowp sampler2D txt_source;" +
|
|
||||||
|
|
||||||
"varying highp vec2 qt_TexCoord0;
|
|
||||||
|
|
||||||
uniform lowp sampler2D blurredSource;
|
|
||||||
uniform highp float blurCoefficient;" +
|
|
||||||
|
|
||||||
"float rgb2grey(vec3 v){
|
|
||||||
return dot(v, vec3(0.21, 0.72, 0.04));
|
|
||||||
}" +
|
|
||||||
|
|
||||||
"void main() {" +
|
|
||||||
"vec2 coords = qt_TexCoord0;" +
|
|
||||||
"vec3 origColor = texture2D(txt_source, coords).rgb;" +
|
|
||||||
"vec3 blur_color = texture2D(blurredSource, coords).rgb - vec3(blurCoefficient);" +
|
|
||||||
"vec3 color = min(origColor + blur_color, max(origColor, blur_color));" +
|
|
||||||
|
|
||||||
"gl_FragColor = vec4(color, rgb2grey(color - origColor));" +
|
|
||||||
"}"
|
|
||||||
|
|
||||||
onStatusChanged: if (log) console.log(log) //Print warning messages
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
159
app/qml/SettingsAdvancedTab.qml
Normal file
159
app/qml/SettingsAdvancedTab.qml
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
/*******************************************************************************
|
||||||
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
|
*
|
||||||
|
* This file is part of cool-retro-term.
|
||||||
|
*
|
||||||
|
* cool-retro-term 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 <http://www.gnu.org/licenses/>.
|
||||||
|
*******************************************************************************/
|
||||||
|
import QtQuick 2.2
|
||||||
|
import QtQuick.Controls 2.0
|
||||||
|
import QtQuick.Layouts 1.1
|
||||||
|
import QtQml 2.0
|
||||||
|
|
||||||
|
import "Components"
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
GroupBox {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
title: qsTr("Command")
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
anchors.fill: parent
|
||||||
|
CheckBox {
|
||||||
|
id: useCustomCommand
|
||||||
|
text: qsTr("Use custom command instead of shell at startup")
|
||||||
|
checked: appSettings.useCustomCommand
|
||||||
|
onCheckedChanged: appSettings.useCustomCommand = checked
|
||||||
|
}
|
||||||
|
// Workaround for QTBUG-31627 for pre 5.3.0
|
||||||
|
Binding {
|
||||||
|
target: useCustomCommand
|
||||||
|
property: "checked"
|
||||||
|
value: appSettings.useCustomCommand
|
||||||
|
}
|
||||||
|
TextField {
|
||||||
|
id: customCommand
|
||||||
|
Layout.fillWidth: true
|
||||||
|
text: appSettings.customCommand
|
||||||
|
enabled: useCustomCommand.checked
|
||||||
|
onEditingFinished: appSettings.customCommand = text
|
||||||
|
|
||||||
|
// Save text even if user forgets to press enter or unfocus
|
||||||
|
function saveSetting() {
|
||||||
|
appSettings.customCommand = text
|
||||||
|
}
|
||||||
|
Component.onCompleted: settings_window.closing.connect(
|
||||||
|
saveSetting)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GroupBox {
|
||||||
|
title: qsTr("Performance")
|
||||||
|
Layout.fillWidth: true
|
||||||
|
GridLayout {
|
||||||
|
anchors.fill: parent
|
||||||
|
columns: 4
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: qsTr("Effects FPS")
|
||||||
|
}
|
||||||
|
Slider {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.columnSpan: 2
|
||||||
|
id: fpsSlider
|
||||||
|
onValueChanged: {
|
||||||
|
if (enabled) {
|
||||||
|
appSettings.fps = value !== 60 ? value + 1 : 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
stepSize: 1
|
||||||
|
enabled: false
|
||||||
|
Component.onCompleted: {
|
||||||
|
from = 0
|
||||||
|
to = 60
|
||||||
|
value = appSettings.fps !== 0 ? appSettings.fps - 1 : 60
|
||||||
|
enabled = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SizedLabel {
|
||||||
|
text: appSettings.fps !== 0 ? appSettings.fps : qsTr("Max")
|
||||||
|
}
|
||||||
|
Label {
|
||||||
|
text: qsTr("Texture Quality")
|
||||||
|
}
|
||||||
|
Slider {
|
||||||
|
id: txtslider
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.columnSpan: 2
|
||||||
|
onValueChanged: if (enabled)
|
||||||
|
appSettings.windowScaling = value
|
||||||
|
stepSize: 0.05
|
||||||
|
enabled: false
|
||||||
|
Component.onCompleted: {
|
||||||
|
from = 0.25 //Without this value gets set to 0.5
|
||||||
|
value = appSettings.windowScaling
|
||||||
|
enabled = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SizedLabel {
|
||||||
|
text: Math.round(txtslider.value * 100) + "%"
|
||||||
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: qsTr("Bloom Quality")
|
||||||
|
}
|
||||||
|
Slider {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.columnSpan: 2
|
||||||
|
id: bloomSlider
|
||||||
|
onValueChanged: if (enabled)
|
||||||
|
appSettings.bloomQuality = value
|
||||||
|
stepSize: 0.05
|
||||||
|
enabled: false
|
||||||
|
Component.onCompleted: {
|
||||||
|
from = 0.25
|
||||||
|
value = appSettings.bloomQuality
|
||||||
|
enabled = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SizedLabel {
|
||||||
|
text: Math.round(bloomSlider.value * 100) + "%"
|
||||||
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: qsTr("BurnIn Quality")
|
||||||
|
}
|
||||||
|
Slider {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
id: burnInSlider
|
||||||
|
Layout.columnSpan: 2
|
||||||
|
onValueChanged: if (enabled)
|
||||||
|
appSettings.burnInQuality = value
|
||||||
|
stepSize: 0.05
|
||||||
|
enabled: false
|
||||||
|
Component.onCompleted: {
|
||||||
|
from = 0.25
|
||||||
|
value = appSettings.burnInQuality
|
||||||
|
enabled = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SizedLabel {
|
||||||
|
text: Math.round(burnInSlider.value * 100) + "%"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,18 +17,20 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.0
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.1
|
||||||
|
|
||||||
Tab{
|
ColumnLayout {
|
||||||
|
spacing: 2
|
||||||
|
|
||||||
GroupBox {
|
GroupBox {
|
||||||
title: qsTr("Effects")
|
title: qsTr("Effects")
|
||||||
anchors.fill: parent
|
Layout.fillWidth: true
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
spacing: 2
|
|
||||||
CheckableSlider {
|
CheckableSlider {
|
||||||
name: qsTr("Bloom")
|
name: qsTr("Bloom")
|
||||||
onNewValue: appSettings.bloom = newValue
|
onNewValue: appSettings.bloom = newValue
|
||||||
@ -51,35 +53,34 @@ Tab{
|
|||||||
}
|
}
|
||||||
CheckableSlider {
|
CheckableSlider {
|
||||||
name: qsTr("Glow Line")
|
name: qsTr("Glow Line")
|
||||||
onNewValue: appSettings.glowingLine = newValue;
|
onNewValue: appSettings.glowingLine = newValue
|
||||||
value: appSettings.glowingLine
|
value: appSettings.glowingLine
|
||||||
}
|
}
|
||||||
CheckableSlider {
|
CheckableSlider {
|
||||||
name: qsTr("Screen Curvature")
|
name: qsTr("Screen Curvature")
|
||||||
onNewValue: appSettings.screenCurvature = newValue;
|
onNewValue: appSettings.screenCurvature = newValue
|
||||||
value: appSettings.screenCurvature;
|
value: appSettings.screenCurvature
|
||||||
}
|
}
|
||||||
CheckableSlider {
|
CheckableSlider {
|
||||||
name: qsTr("Ambient Light")
|
name: qsTr("Ambient Light")
|
||||||
onNewValue: appSettings.ambientLight = newValue;
|
onNewValue: appSettings.ambientLight = newValue
|
||||||
value: appSettings.ambientLight
|
value: appSettings.ambientLight
|
||||||
enabled: appSettings.framesIndex !== 0
|
enabled: appSettings.framesIndex !== 0
|
||||||
}
|
}
|
||||||
CheckableSlider {
|
CheckableSlider {
|
||||||
name: qsTr("Flickering")
|
name: qsTr("Flickering")
|
||||||
onNewValue: appSettings.flickering = newValue;
|
onNewValue: appSettings.flickering = newValue
|
||||||
value: appSettings.flickering;
|
value: appSettings.flickering
|
||||||
}
|
}
|
||||||
CheckableSlider {
|
CheckableSlider {
|
||||||
name: qsTr("Horizontal Sync")
|
name: qsTr("Horizontal Sync")
|
||||||
onNewValue: appSettings.horizontalSync = newValue;
|
onNewValue: appSettings.horizontalSync = newValue
|
||||||
value: appSettings.horizontalSync;
|
value: appSettings.horizontalSync
|
||||||
}
|
}
|
||||||
CheckableSlider {
|
CheckableSlider {
|
||||||
name: qsTr("RGB Shift")
|
name: qsTr("RGB Shift")
|
||||||
onNewValue: appSettings.rbgShift = newValue;
|
onNewValue: appSettings.rbgShift = newValue
|
||||||
value: appSettings.rbgShift;
|
value: appSettings.rbgShift
|
||||||
enabled: appSettings.chromaColor !== 0
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,72 +17,74 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.4
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.1
|
||||||
import QtQuick.Dialogs 1.1
|
import QtQuick.Dialogs 1.1
|
||||||
|
|
||||||
Tab{
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
anchors.fill: parent
|
|
||||||
GroupBox {
|
GroupBox {
|
||||||
anchors {left: parent.left; right: parent.right}
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.fillHeight: true
|
|
||||||
title: qsTr("Profile")
|
title: qsTr("Profile")
|
||||||
RowLayout {
|
RowLayout {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
TableView {
|
ListView {
|
||||||
id: profilesView
|
id: profilesView
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
anchors { top: parent.top; bottom: parent.bottom; }
|
Layout.fillHeight: true
|
||||||
model: appSettings.profilesList
|
model: appSettings.profilesList
|
||||||
headerVisible: false
|
clip: true
|
||||||
TableViewColumn {
|
delegate: Rectangle {
|
||||||
title: qsTr("Profile")
|
width: label.width
|
||||||
role: "text"
|
height: label.height
|
||||||
width: parent.width * 0.5
|
color: (index == profilesView.currentIndex) ? palette.highlight : palette.base
|
||||||
|
Label {
|
||||||
|
id: label
|
||||||
|
text: appSettings.profilesList.get(index).text
|
||||||
|
MouseArea {
|
||||||
|
anchors.fill: parent
|
||||||
|
onClicked: profilesView.currentIndex = index
|
||||||
|
onDoubleClicked: appSettings.loadProfile(index)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
onActivated: {
|
|
||||||
appSettings.loadProfile(row);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
anchors { top: parent.top; bottom: parent.bottom }
|
Layout.fillHeight: true
|
||||||
Layout.fillWidth: false
|
Layout.fillWidth: false
|
||||||
Button {
|
Button {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: qsTr("New")
|
text: qsTr("Save")
|
||||||
onClicked: {
|
onClicked: {
|
||||||
insertname.profileName = "";
|
insertname.profileName = ""
|
||||||
insertname.show()
|
insertname.show()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Button {
|
Button {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
property alias currentIndex: profilesView.currentRow
|
property alias currentIndex: profilesView.currentIndex
|
||||||
enabled: currentIndex >= 0
|
enabled: currentIndex >= 0
|
||||||
text: qsTr("Load")
|
text: qsTr("Load")
|
||||||
onClicked: {
|
onClicked: {
|
||||||
var index = profilesView.currentRow;
|
var index = currentIndex
|
||||||
if (index >= 0)
|
if (index >= 0)
|
||||||
appSettings.loadProfile(index);
|
appSettings.loadProfile(index)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Button {
|
Button {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: qsTr("Remove")
|
text: qsTr("Remove")
|
||||||
property alias currentIndex: profilesView.currentRow
|
property alias currentIndex: profilesView.currentIndex
|
||||||
|
|
||||||
enabled: currentIndex >= 0 && !appSettings.profilesList.get(currentIndex).builtin
|
enabled: currentIndex >= 0 && !appSettings.profilesList.get(
|
||||||
|
currentIndex).builtin
|
||||||
onClicked: {
|
onClicked: {
|
||||||
appSettings.profilesList.remove(currentIndex);
|
appSettings.profilesList.remove(currentIndex)
|
||||||
profilesView.selection.clear();
|
profilesView.selection.clear()
|
||||||
|
|
||||||
// TODO This is a very ugly workaround. The view didn't update on Qt 5.3.2.
|
// TODO This is a very ugly workaround. The view didn't update on Qt 5.3.2.
|
||||||
profilesView.model = 0;
|
profilesView.model = 0
|
||||||
profilesView.model = appSettings.profilesList;
|
profilesView.model = appSettings.profilesList
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Item {
|
Item {
|
||||||
@ -93,66 +95,84 @@ Tab{
|
|||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: qsTr("Import")
|
text: qsTr("Import")
|
||||||
onClicked: {
|
onClicked: {
|
||||||
fileDialog.selectExisting = true;
|
fileDialog.selectExisting = true
|
||||||
fileDialog.callBack = function (url) {loadFile(url);};
|
fileDialog.callBack = function (url) {
|
||||||
fileDialog.open();
|
loadFile(url)
|
||||||
|
}
|
||||||
|
fileDialog.open()
|
||||||
}
|
}
|
||||||
function loadFile(url) {
|
function loadFile(url) {
|
||||||
try {
|
try {
|
||||||
if (appSettings.verbose)
|
if (appSettings.verbose)
|
||||||
console.log("Loading file: " + url);
|
console.log("Loading file: " + url)
|
||||||
|
|
||||||
var profileObject = JSON.parse(fileIO.read(url));
|
var profileObject = JSON.parse(fileIO.read(url))
|
||||||
var name = profileObject.name;
|
var name = profileObject.name
|
||||||
|
|
||||||
if (!name)
|
if (!name)
|
||||||
throw "Profile doesn't have a name";
|
throw "Profile doesn't have a name"
|
||||||
|
|
||||||
delete profileObject.name;
|
var version = profileObject.version
|
||||||
|
!== undefined ? profileObject.version : 1
|
||||||
|
if (version !== appSettings.profileVersion)
|
||||||
|
throw "This profile is not supported on this version of CRT."
|
||||||
|
|
||||||
appSettings.appendCustomProfile(name, JSON.stringify(profileObject));
|
delete profileObject.name
|
||||||
|
|
||||||
|
appSettings.appendCustomProfile(name,
|
||||||
|
JSON.stringify(
|
||||||
|
profileObject))
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.log(err);
|
messageDialog.text = qsTr(err)
|
||||||
messageDialog.text = qsTr("There has been an error reading the file.")
|
messageDialog.open()
|
||||||
messageDialog.open();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Button {
|
Button {
|
||||||
property alias currentIndex: profilesView.currentRow
|
property alias currentIndex: profilesView.currentIndex
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
text: qsTr("Export")
|
text: qsTr("Export")
|
||||||
enabled: currentIndex >= 0 && !appSettings.profilesList.get(currentIndex).builtin
|
enabled: currentIndex >= 0 && !appSettings.profilesList.get(
|
||||||
|
currentIndex).builtin
|
||||||
onClicked: {
|
onClicked: {
|
||||||
fileDialog.selectExisting = false;
|
fileDialog.selectExisting = false
|
||||||
fileDialog.callBack = function (url) {storeFile(url);};
|
fileDialog.callBack = function (url) {
|
||||||
fileDialog.open();
|
storeFile(url)
|
||||||
|
}
|
||||||
|
fileDialog.open()
|
||||||
}
|
}
|
||||||
function storeFile(url) {
|
function storeFile(url) {
|
||||||
try {
|
try {
|
||||||
var urlString = url.toString();
|
var urlString = url.toString()
|
||||||
|
|
||||||
// Fix the extension if it's missing.
|
// Fix the extension if it's missing.
|
||||||
var extension = urlString.substring(urlString.length - 5, urlString.length);
|
var extension = urlString.substring(
|
||||||
var urlTail = (extension === ".json" ? "" : ".json");
|
urlString.length - 5, urlString.length)
|
||||||
url += urlTail;
|
var urlTail = (extension === ".json" ? "" : ".json")
|
||||||
|
url += urlTail
|
||||||
|
|
||||||
if (true)
|
if (true)
|
||||||
console.log("Storing file: " + url);
|
console.log("Storing file: " + url)
|
||||||
|
|
||||||
var profileObject = appSettings.profilesList.get(currentIndex);
|
var profileObject = appSettings.profilesList.get(
|
||||||
var profileSettings = JSON.parse(profileObject.obj_string);
|
currentIndex)
|
||||||
profileSettings["name"] = profileObject.text;
|
var profileSettings = JSON.parse(
|
||||||
|
profileObject.obj_string)
|
||||||
|
profileSettings["name"] = profileObject.text
|
||||||
|
profileSettings["version"] = appSettings.profileVersion
|
||||||
|
|
||||||
var result = fileIO.write(url, JSON.stringify(profileSettings, undefined, 2));
|
var result = fileIO.write(url, JSON.stringify(
|
||||||
|
profileSettings,
|
||||||
|
undefined, 2))
|
||||||
if (!result)
|
if (!result)
|
||||||
throw "The file could not be written.";
|
throw "The file could not be written."
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.log(err);
|
console.log(err)
|
||||||
messageDialog.text = qsTr("There has been an error storing the file.")
|
messageDialog.text = qsTr(
|
||||||
messageDialog.open();
|
"There has been an error storing the file.")
|
||||||
|
messageDialog.open()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -161,34 +181,47 @@ Tab{
|
|||||||
}
|
}
|
||||||
|
|
||||||
GroupBox {
|
GroupBox {
|
||||||
anchors {left: parent.left; right: parent.right}
|
title: qsTr("Screen")
|
||||||
title: qsTr("Command")
|
Layout.fillWidth: true
|
||||||
ColumnLayout {
|
GridLayout {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
CheckBox{
|
columns: 2
|
||||||
id: useCustomCommand
|
Label {
|
||||||
text: qsTr("Use custom command instead of shell at startup")
|
text: qsTr("Brightness")
|
||||||
checked: appSettings.useCustomCommand
|
|
||||||
onCheckedChanged: appSettings.useCustomCommand = checked
|
|
||||||
}
|
}
|
||||||
// Workaround for QTBUG-31627 for pre 5.3.0
|
SimpleSlider {
|
||||||
Binding{
|
onValueChanged: appSettings.brightness = value
|
||||||
target: useCustomCommand
|
value: appSettings.brightness
|
||||||
property: "checked"
|
|
||||||
value: appSettings.useCustomCommand
|
|
||||||
}
|
}
|
||||||
TextField{
|
Label {
|
||||||
id: customCommand
|
text: qsTr("Contrast")
|
||||||
anchors {left: parent.left; right: parent.right}
|
|
||||||
text: appSettings.customCommand
|
|
||||||
enabled: useCustomCommand.checked
|
|
||||||
onEditingFinished: appSettings.customCommand = text
|
|
||||||
|
|
||||||
// Save text even if user forgets to press enter or unfocus
|
|
||||||
function saveSetting() {
|
|
||||||
appSettings.customCommand = text;
|
|
||||||
}
|
}
|
||||||
Component.onCompleted: settings_window.closing.connect(saveSetting)
|
SimpleSlider {
|
||||||
|
onValueChanged: appSettings.contrast = value
|
||||||
|
value: appSettings.contrast
|
||||||
|
}
|
||||||
|
Label {
|
||||||
|
text: qsTr("Margin")
|
||||||
|
}
|
||||||
|
SimpleSlider {
|
||||||
|
onValueChanged: appSettings._margin = value
|
||||||
|
value: appSettings._margin
|
||||||
|
}
|
||||||
|
Label {
|
||||||
|
text: qsTr("Frame size")
|
||||||
|
}
|
||||||
|
SimpleSlider {
|
||||||
|
onValueChanged: appSettings._frameMargin = value
|
||||||
|
value: appSettings._frameMargin
|
||||||
|
}
|
||||||
|
Label {
|
||||||
|
text: qsTr("Opacity")
|
||||||
|
visible: !appSettings.isMacOS
|
||||||
|
}
|
||||||
|
SimpleSlider {
|
||||||
|
onValueChanged: appSettings.windowOpacity = value
|
||||||
|
value: appSettings.windowOpacity
|
||||||
|
visible: !appSettings.isMacOS
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -197,14 +230,15 @@ Tab{
|
|||||||
InsertNameDialog {
|
InsertNameDialog {
|
||||||
id: insertname
|
id: insertname
|
||||||
onNameSelected: {
|
onNameSelected: {
|
||||||
appSettings.appendCustomProfile(name, appSettings.composeProfileString());
|
appSettings.appendCustomProfile(name,
|
||||||
|
appSettings.composeProfileString())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
MessageDialog {
|
MessageDialog {
|
||||||
id: messageDialog
|
id: messageDialog
|
||||||
title: qsTr("File Error")
|
title: qsTr("File Error")
|
||||||
onAccepted: {
|
onAccepted: {
|
||||||
messageDialog.close();
|
messageDialog.close()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Loader {
|
Loader {
|
||||||
@ -217,19 +251,18 @@ Tab{
|
|||||||
selectMultiple: false
|
selectMultiple: false
|
||||||
selectFolder: false
|
selectFolder: false
|
||||||
selectExisting: fileDialog.selectExisting
|
selectExisting: fileDialog.selectExisting
|
||||||
onAccepted: callBack(fileUrl);
|
onAccepted: callBack(fileUrl)
|
||||||
}
|
}
|
||||||
|
|
||||||
onSelectExistingChanged: reload()
|
onSelectExistingChanged: reload()
|
||||||
|
|
||||||
function open() {
|
function open() {
|
||||||
item.open();
|
item.open()
|
||||||
}
|
}
|
||||||
|
|
||||||
function reload() {
|
function reload() {
|
||||||
active = false;
|
active = false
|
||||||
active = true;
|
active = true
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,135 +0,0 @@
|
|||||||
/*******************************************************************************
|
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
|
||||||
*
|
|
||||||
* This file is part of cool-retro-term.
|
|
||||||
*
|
|
||||||
* cool-retro-term 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 <http://www.gnu.org/licenses/>.
|
|
||||||
*******************************************************************************/
|
|
||||||
|
|
||||||
import QtQuick 2.2
|
|
||||||
import QtQuick.Controls 1.1
|
|
||||||
import QtQuick.Layouts 1.1
|
|
||||||
|
|
||||||
import "Components"
|
|
||||||
|
|
||||||
Tab{
|
|
||||||
ColumnLayout{
|
|
||||||
anchors.fill: parent
|
|
||||||
GroupBox{
|
|
||||||
title: qsTr("General")
|
|
||||||
Layout.fillWidth: true
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.right: parent.right
|
|
||||||
GridLayout{
|
|
||||||
anchors.fill: parent
|
|
||||||
rows: 2
|
|
||||||
columns: 3
|
|
||||||
Label{text: qsTr("Effects FPS")}
|
|
||||||
Slider{
|
|
||||||
Layout.fillWidth: true
|
|
||||||
id: fpsSlider
|
|
||||||
onValueChanged: {
|
|
||||||
if (enabled) {
|
|
||||||
appSettings.fps = value !== 60 ? value + 1 : 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
stepSize: 1
|
|
||||||
enabled: false
|
|
||||||
Component.onCompleted: {
|
|
||||||
minimumValue = 0;
|
|
||||||
maximumValue = 60;
|
|
||||||
value = appSettings.fps !== 0 ? appSettings.fps - 1 : 60;
|
|
||||||
enabled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SizedLabel{text: appSettings.fps !== 0 ? appSettings.fps : qsTr("Max")}
|
|
||||||
Label{text: qsTr("Texture Quality")}
|
|
||||||
Slider{
|
|
||||||
Layout.fillWidth: true
|
|
||||||
id: txtslider
|
|
||||||
onValueChanged: if (enabled) appSettings.windowScaling = value;
|
|
||||||
stepSize: 0.05
|
|
||||||
enabled: false
|
|
||||||
Component.onCompleted: {
|
|
||||||
minimumValue = 0.25 //Without this value gets set to 0.5
|
|
||||||
value = appSettings.windowScaling;
|
|
||||||
enabled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SizedLabel{text: Math.round(txtslider.value * 100) + "%"}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
GroupBox{
|
|
||||||
title: qsTr("Bloom")
|
|
||||||
Layout.fillWidth: true
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.right: parent.right
|
|
||||||
GridLayout{
|
|
||||||
id: bloomQualityContainer
|
|
||||||
anchors.fill: parent
|
|
||||||
Label{text: qsTr("Bloom Quality")}
|
|
||||||
Slider{
|
|
||||||
Layout.fillWidth: true
|
|
||||||
id: bloomSlider
|
|
||||||
onValueChanged: if (enabled) appSettings.bloomQuality = value;
|
|
||||||
stepSize: 0.05
|
|
||||||
enabled: false
|
|
||||||
Component.onCompleted: {
|
|
||||||
minimumValue = 0.25
|
|
||||||
value = appSettings.bloomQuality;
|
|
||||||
enabled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SizedLabel{text: Math.round(bloomSlider.value * 100) + "%"}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
GroupBox{
|
|
||||||
title: qsTr("BurnIn")
|
|
||||||
Layout.fillWidth: true
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.right: parent.right
|
|
||||||
GridLayout{
|
|
||||||
id: blurQualityContainer
|
|
||||||
anchors.fill: parent
|
|
||||||
|
|
||||||
Label{text: qsTr("BurnIn Quality")}
|
|
||||||
Slider{
|
|
||||||
Layout.fillWidth: true
|
|
||||||
id: burnInSlider
|
|
||||||
onValueChanged: if (enabled) appSettings.burnInQuality = value;
|
|
||||||
stepSize: 0.05
|
|
||||||
enabled: false
|
|
||||||
Component.onCompleted: {
|
|
||||||
minimumValue = 0.25
|
|
||||||
value = appSettings.burnInQuality;
|
|
||||||
enabled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SizedLabel{text: Math.round(burnInSlider.value * 100) + "%"}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
GroupBox{
|
|
||||||
title: qsTr("Frame")
|
|
||||||
Layout.fillWidth: true
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.right: parent.right
|
|
||||||
CheckBox{
|
|
||||||
checked: appSettings._frameReflections
|
|
||||||
text: qsTr("Frame Reflections")
|
|
||||||
onCheckedChanged: appSettings._frameReflections = checked
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,94 +0,0 @@
|
|||||||
/*******************************************************************************
|
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
|
||||||
*
|
|
||||||
* This file is part of cool-retro-term.
|
|
||||||
*
|
|
||||||
* cool-retro-term 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 <http://www.gnu.org/licenses/>.
|
|
||||||
*******************************************************************************/
|
|
||||||
|
|
||||||
import QtQuick 2.2
|
|
||||||
import QtQuick.Controls 1.1
|
|
||||||
import QtQuick.Layouts 1.1
|
|
||||||
import QtQuick.Dialogs 1.1
|
|
||||||
|
|
||||||
Tab{
|
|
||||||
ColumnLayout{
|
|
||||||
anchors.fill: parent
|
|
||||||
GroupBox{
|
|
||||||
title: qsTr("Rasterization Mode")
|
|
||||||
Layout.fillWidth: true
|
|
||||||
ComboBox {
|
|
||||||
id: rasterizationBox
|
|
||||||
property string selectedElement: model[currentIndex]
|
|
||||||
anchors.fill: parent
|
|
||||||
model: [qsTr("Default"), qsTr("Scanlines"), qsTr("Pixels")]
|
|
||||||
currentIndex: appSettings.rasterization
|
|
||||||
onCurrentIndexChanged: {
|
|
||||||
appSettings.rasterization = currentIndex
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
GroupBox{
|
|
||||||
title: qsTr("Lights")
|
|
||||||
Layout.fillWidth: true
|
|
||||||
GridLayout{
|
|
||||||
anchors.fill: parent
|
|
||||||
columns: 2
|
|
||||||
Label{ text: qsTr("Brightness") }
|
|
||||||
SimpleSlider{
|
|
||||||
onValueChanged: appSettings.brightness = value
|
|
||||||
value: appSettings.brightness
|
|
||||||
}
|
|
||||||
Label{ text: qsTr("Contrast") }
|
|
||||||
SimpleSlider{
|
|
||||||
onValueChanged: appSettings.contrast = value
|
|
||||||
value: appSettings.contrast
|
|
||||||
}
|
|
||||||
Label{ text: qsTr("Opacity") }
|
|
||||||
SimpleSlider{
|
|
||||||
onValueChanged: appSettings.windowOpacity = value
|
|
||||||
value: appSettings.windowOpacity
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
GroupBox{
|
|
||||||
title: qsTr("Frame")
|
|
||||||
Layout.fillWidth: true
|
|
||||||
RowLayout{
|
|
||||||
anchors.fill: parent
|
|
||||||
ComboBox{
|
|
||||||
id: framescombobox
|
|
||||||
Layout.fillWidth: true
|
|
||||||
model: appSettings.framesList
|
|
||||||
currentIndex: appSettings.framesIndex
|
|
||||||
onActivated: {
|
|
||||||
appSettings.frameName = appSettings.framesList.get(index).name;
|
|
||||||
}
|
|
||||||
function updateIndex(){
|
|
||||||
var name = appSettings.frameName;
|
|
||||||
var index = appSettings.getFrameIndexByName(name);
|
|
||||||
if (index !== undefined)
|
|
||||||
currentIndex = index;
|
|
||||||
}
|
|
||||||
Component.onCompleted: updateIndex();
|
|
||||||
Connections {
|
|
||||||
target: appSettings
|
|
||||||
onFrameNameChanged: framescombobox.updateIndex();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,83 +17,95 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.1
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.1
|
||||||
|
import QtQml 2.0
|
||||||
|
|
||||||
import "Components"
|
import "Components"
|
||||||
|
|
||||||
Tab{
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
anchors.fill: parent
|
|
||||||
GroupBox {
|
GroupBox {
|
||||||
property var rasterization: [qsTr("Default"), qsTr("Scanlines"), qsTr("Pixels")][appSettings.rasterization]
|
title: qsTr("Font")
|
||||||
title: qsTr("Font" + "(" + rasterization + ")")
|
Layout.fillWidth: true
|
||||||
anchors { left: parent.left; right: parent.right }
|
|
||||||
GridLayout {
|
GridLayout {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
columns: 2
|
columns: 2
|
||||||
Label{ text: qsTr("Name") }
|
Label {
|
||||||
|
text: qsTr("Rasterization")
|
||||||
|
}
|
||||||
|
ComboBox {
|
||||||
|
id: rasterizationBox
|
||||||
|
|
||||||
|
property string selectedElement: model[currentIndex]
|
||||||
|
|
||||||
|
Layout.fillWidth: true
|
||||||
|
model: [qsTr("Default"), qsTr("Scanlines"), qsTr("Pixels"), qsTr("Sub-Pixels")]
|
||||||
|
currentIndex: appSettings.rasterization
|
||||||
|
onCurrentIndexChanged: {
|
||||||
|
appSettings.rasterization = currentIndex
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Label {
|
||||||
|
text: qsTr("Name")
|
||||||
|
}
|
||||||
ComboBox {
|
ComboBox {
|
||||||
id: fontChanger
|
id: fontChanger
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
model: appSettings.fontlist
|
model: appSettings.fontlist
|
||||||
|
textRole: "text"
|
||||||
onActivated: {
|
onActivated: {
|
||||||
var name = appSettings.fontlist.get(index).name;
|
var name = appSettings.fontlist.get(index).name
|
||||||
appSettings.fontNames[appSettings.rasterization] = name;
|
appSettings.fontNames[appSettings.rasterization] = name
|
||||||
appSettings.handleFontChanged();
|
appSettings.handleFontChanged()
|
||||||
}
|
}
|
||||||
function updateIndex() {
|
function updateIndex() {
|
||||||
var name = appSettings.fontNames[appSettings.rasterization];
|
var name = appSettings.fontNames[appSettings.rasterization]
|
||||||
var index = appSettings.getIndexByName(name);
|
var index = appSettings.getIndexByName(name)
|
||||||
if (index !== undefined)
|
if (index !== undefined)
|
||||||
currentIndex = index;
|
currentIndex = index
|
||||||
}
|
}
|
||||||
Connections {
|
Connections {
|
||||||
target: appSettings
|
target: appSettings
|
||||||
onTerminalFontChanged: fontChanger.updateIndex();
|
|
||||||
|
onTerminalFontChanged: {
|
||||||
|
fontChanger.updateIndex()
|
||||||
}
|
}
|
||||||
Component.onCompleted: updateIndex();
|
|
||||||
}
|
}
|
||||||
Label{ text: qsTr("Scaling") }
|
Component.onCompleted: updateIndex()
|
||||||
|
}
|
||||||
|
Label {
|
||||||
|
text: qsTr("Scaling")
|
||||||
|
}
|
||||||
RowLayout {
|
RowLayout {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Slider {
|
Slider {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
id: fontScalingChanger
|
id: fontScalingChanger
|
||||||
onValueChanged: if(enabled) appSettings.fontScaling = value
|
onValueChanged: appSettings.fontScaling = value
|
||||||
|
value: appSettings.fontScaling
|
||||||
stepSize: 0.05
|
stepSize: 0.05
|
||||||
enabled: false // Another trick to fix initial bad behavior.
|
from: appSettings.minimumFontScaling
|
||||||
Component.onCompleted: {
|
to: appSettings.maximumFontScaling
|
||||||
minimumValue = appSettings.minimumFontScaling;
|
|
||||||
maximumValue = appSettings.maximumFontScaling;
|
|
||||||
value = appSettings.fontScaling;
|
|
||||||
enabled = true;
|
|
||||||
}
|
|
||||||
Connections{
|
|
||||||
target: appSettings
|
|
||||||
onFontScalingChanged: fontScalingChanger.value = appSettings.fontScaling;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
SizedLabel {
|
SizedLabel {
|
||||||
text: Math.round(fontScalingChanger.value * 100) + "%"
|
text: Math.round(fontScalingChanger.value * 100) + "%"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Label{ text: qsTr("Font Width") }
|
Label {
|
||||||
|
text: qsTr("Font Width")
|
||||||
|
}
|
||||||
RowLayout {
|
RowLayout {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Slider {
|
Slider {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
id: widthChanger
|
id: widthChanger
|
||||||
onValueChanged: appSettings.fontWidth = value;
|
onValueChanged: appSettings.fontWidth = value
|
||||||
value: appSettings.fontWidth
|
value: appSettings.fontWidth
|
||||||
stepSize: 0.05
|
stepSize: 0.05
|
||||||
Component.onCompleted: {
|
from: 0.5
|
||||||
// This is needed to avoid unnecessary chnaged events.
|
to: 1.5
|
||||||
minimumValue = 0.5;
|
|
||||||
maximumValue = 1.5;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
SizedLabel {
|
SizedLabel {
|
||||||
text: Math.round(widthChanger.value * 100) + "%"
|
text: Math.round(widthChanger.value * 100) + "%"
|
||||||
@ -101,9 +113,27 @@ Tab{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
GroupBox {
|
||||||
|
title: qsTr("Cursor")
|
||||||
|
Layout.fillWidth: true
|
||||||
|
ColumnLayout {
|
||||||
|
anchors.fill: parent
|
||||||
|
CheckBox {
|
||||||
|
id: blinkingCursor
|
||||||
|
text: qsTr("Blinking Cursor")
|
||||||
|
checked: appSettings.blinkingCursor
|
||||||
|
onCheckedChanged: appSettings.blinkingCursor = checked
|
||||||
|
}
|
||||||
|
Binding {
|
||||||
|
target: blinkingCursor
|
||||||
|
property: "checked"
|
||||||
|
value: appSettings.blinkingCursor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
GroupBox {
|
GroupBox {
|
||||||
title: qsTr("Colors")
|
title: qsTr("Colors")
|
||||||
anchors { left: parent.left; right: parent.right }
|
Layout.fillWidth: true
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
@ -119,6 +149,11 @@ Tab{
|
|||||||
value: appSettings.saturationColor
|
value: appSettings.saturationColor
|
||||||
enabled: appSettings.chromaColor !== 0
|
enabled: appSettings.chromaColor !== 0
|
||||||
}
|
}
|
||||||
|
CheckableSlider {
|
||||||
|
name: qsTr("Frame Gloss")
|
||||||
|
onNewValue: appSettings.frameGloss = newValue
|
||||||
|
value: appSettings.frameGloss
|
||||||
|
}
|
||||||
}
|
}
|
||||||
RowLayout {
|
RowLayout {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
@ -126,16 +161,22 @@ Tab{
|
|||||||
name: qsTr("Font")
|
name: qsTr("Font")
|
||||||
height: 50
|
height: 50
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
onColorSelected: appSettings._fontColor = color;
|
onColorSelected: appSettings._fontColor = color
|
||||||
color: appSettings._fontColor
|
color: appSettings._fontColor
|
||||||
}
|
}
|
||||||
ColorButton {
|
ColorButton {
|
||||||
name: qsTr("Background")
|
name: qsTr("Background")
|
||||||
height: 50
|
height: 50
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
onColorSelected: appSettings._backgroundColor = color;
|
onColorSelected: appSettings._backgroundColor = color
|
||||||
color: appSettings._backgroundColor
|
color: appSettings._backgroundColor
|
||||||
}
|
}
|
||||||
|
ColorButton {
|
||||||
|
name: qsTr("Frame")
|
||||||
|
height: 50
|
||||||
|
Layout.fillWidth: true
|
||||||
|
onColorSelected: appSettings._frameColor = color
|
||||||
|
color: appSettings._frameColor
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -19,52 +19,57 @@
|
|||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.1
|
||||||
import QtQuick.Window 2.1
|
import QtQuick.Window 2.1
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.3
|
||||||
import QtQuick.Dialogs 1.1
|
import QtQuick.Dialogs 1.1
|
||||||
|
|
||||||
Window {
|
Window {
|
||||||
id: settings_window
|
id: settings_window
|
||||||
title: qsTr("Settings")
|
title: qsTr("Settings")
|
||||||
width: 580
|
width: 640
|
||||||
height: 400
|
height: 640
|
||||||
|
|
||||||
property int tabmargins: 15
|
property int tabmargins: 15
|
||||||
|
|
||||||
TabView{
|
Item {
|
||||||
id: tabView
|
anchors { fill: parent; margins: tabmargins }
|
||||||
|
|
||||||
|
TabBar {
|
||||||
|
id: bar
|
||||||
|
anchors { left: parent.left; right: parent.right; top: parent.top; }
|
||||||
|
TabButton {
|
||||||
|
text: qsTr("General")
|
||||||
|
}
|
||||||
|
TabButton {
|
||||||
|
text: qsTr("Terminal")
|
||||||
|
}
|
||||||
|
TabButton {
|
||||||
|
text: qsTr("Effects")
|
||||||
|
}
|
||||||
|
TabButton {
|
||||||
|
text: qsTr("Advanced")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Frame {
|
||||||
|
anchors {
|
||||||
|
top: bar.bottom
|
||||||
|
left: parent.left
|
||||||
|
right: parent.right
|
||||||
|
bottom: parent.bottom
|
||||||
|
}
|
||||||
|
|
||||||
|
StackLayout {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
anchors.margins: 10
|
|
||||||
SettingsGeneralTab{
|
currentIndex: bar.currentIndex
|
||||||
id: generalTab
|
|
||||||
title: qsTr("General")
|
SettingsGeneralTab { }
|
||||||
anchors.fill: parent
|
SettingsTerminalTab { }
|
||||||
anchors.margins: tabmargins
|
SettingsEffectsTab { }
|
||||||
}
|
SettingsAdvancedTab { }
|
||||||
SettingsScreenTab{
|
}
|
||||||
id: screenTab
|
|
||||||
title: qsTr("Screen")
|
|
||||||
anchors.fill: parent
|
|
||||||
anchors.margins: tabmargins
|
|
||||||
}
|
|
||||||
SettingsTerminalTab{
|
|
||||||
id: terminalTab
|
|
||||||
title: qsTr("Terminal")
|
|
||||||
anchors.fill: parent
|
|
||||||
anchors.margins: tabmargins
|
|
||||||
}
|
|
||||||
SettingsEffectsTab{
|
|
||||||
id: effectsTab
|
|
||||||
title: qsTr("Effects")
|
|
||||||
anchors.fill: parent
|
|
||||||
anchors.margins: tabmargins
|
|
||||||
}
|
|
||||||
SettingsPerformanceTab{
|
|
||||||
id: performanceTab
|
|
||||||
title: qsTr("Performance")
|
|
||||||
anchors.fill: parent
|
|
||||||
anchors.margins: tabmargins
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
91
app/qml/ShaderLibrary.qml
Normal file
91
app/qml/ShaderLibrary.qml
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
import QtQuick 2.0
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
property string rasterizationShader:
|
||||||
|
(appSettings.rasterization === appSettings.no_rasterization ? "
|
||||||
|
lowp vec3 applyRasterization(vec2 screenCoords, lowp vec3 texel, vec2 virtualResolution, float intensity) {
|
||||||
|
return texel;
|
||||||
|
}" : "") +
|
||||||
|
|
||||||
|
(appSettings.rasterization === appSettings.scanline_rasterization ? "
|
||||||
|
#define INTENSITY 0.30
|
||||||
|
#define BRIGHTBOOST 0.30
|
||||||
|
|
||||||
|
lowp vec3 applyRasterization(vec2 screenCoords, lowp vec3 texel, vec2 virtualResolution, float intensity) {
|
||||||
|
lowp vec3 pixelHigh = ((1.0 + BRIGHTBOOST) - (0.2 * texel)) * texel;
|
||||||
|
lowp vec3 pixelLow = ((1.0 - INTENSITY) + (0.1 * texel)) * texel;
|
||||||
|
|
||||||
|
vec2 coords = fract(screenCoords * virtualResolution) * 2.0 - vec2(1.0);
|
||||||
|
lowp float mask = 1.0 - abs(coords.y);
|
||||||
|
|
||||||
|
lowp vec3 rasterizationColor = mix(pixelLow, pixelHigh, mask);
|
||||||
|
return mix(texel, rasterizationColor, intensity);
|
||||||
|
}" : "") +
|
||||||
|
|
||||||
|
(appSettings.rasterization === appSettings.pixel_rasterization ? "
|
||||||
|
#define INTENSITY 0.30
|
||||||
|
#define BRIGHTBOOST 0.30
|
||||||
|
|
||||||
|
lowp vec3 applyRasterization(vec2 screenCoords, lowp vec3 texel, vec2 virtualResolution, float intensity) {
|
||||||
|
lowp vec3 result = texel;
|
||||||
|
|
||||||
|
lowp vec3 pixelHigh = ((1.0 + BRIGHTBOOST) - (0.2 * texel)) * texel;
|
||||||
|
lowp vec3 pixelLow = ((1.0 - INTENSITY) + (0.1 * texel)) * texel;
|
||||||
|
|
||||||
|
vec2 coords = fract(screenCoords * virtualResolution) * 2.0 - vec2(1.0);
|
||||||
|
coords = coords * coords;
|
||||||
|
lowp float mask = 1.0 - coords.x - coords.y;
|
||||||
|
|
||||||
|
lowp vec3 rasterizationColor = mix(pixelLow, pixelHigh, mask);
|
||||||
|
return mix(texel, rasterizationColor, intensity);
|
||||||
|
}" : "") +
|
||||||
|
|
||||||
|
(appSettings.rasterization === appSettings.subpixel_rasterization ? "
|
||||||
|
#define INTENSITY 0.30
|
||||||
|
#define BRIGHTBOOST 0.30
|
||||||
|
#define SUBPIXELS 3.0
|
||||||
|
const vec3 offsets = vec3(3.141592654) * vec3(1.0/2.0,1.0/2.0 - 2.0/3.0,1.0/2.0-4.0/3.0);
|
||||||
|
|
||||||
|
lowp vec3 applyRasterization(vec2 screenCoords, lowp vec3 texel, vec2 virtualResolution, float intensity) {
|
||||||
|
vec2 omega = vec2(3.141592654) * vec2(2.0) * virtualResolution;
|
||||||
|
vec2 angle = screenCoords * omega;
|
||||||
|
vec3 xfactors = (SUBPIXELS + sin(angle.x + offsets)) / (SUBPIXELS + 1.0);
|
||||||
|
|
||||||
|
lowp vec3 result = texel * xfactors;
|
||||||
|
lowp vec3 pixelHigh = ((1.0 + BRIGHTBOOST) - (0.2 * result)) * result;
|
||||||
|
lowp vec3 pixelLow = ((1.0 - INTENSITY) + (0.1 * result)) * result;
|
||||||
|
|
||||||
|
vec2 coords = fract(screenCoords * virtualResolution) * 2.0 - vec2(1.0);
|
||||||
|
lowp float mask = 1.0 - abs(coords.y);
|
||||||
|
|
||||||
|
lowp vec3 rasterizationColor = mix(pixelLow, pixelHigh, mask);
|
||||||
|
return mix(texel, rasterizationColor, intensity);
|
||||||
|
}" : "") +
|
||||||
|
|
||||||
|
"\n\n"
|
||||||
|
|
||||||
|
property string min2: "
|
||||||
|
float min2(vec2 v) {
|
||||||
|
return min(v.x, v.y);
|
||||||
|
}\n\n"
|
||||||
|
|
||||||
|
property string rgb2grey: "
|
||||||
|
float rgb2grey(vec3 v) {
|
||||||
|
return dot(v, vec3(0.21, 0.72, 0.04));
|
||||||
|
}\n\n"
|
||||||
|
|
||||||
|
property string max2: "
|
||||||
|
float max2(vec2 v) {
|
||||||
|
return max(v.x, v.y);
|
||||||
|
}\n\n"
|
||||||
|
|
||||||
|
property string prod2: "
|
||||||
|
float prod2(vec2 v) {
|
||||||
|
return v.x * v.y;
|
||||||
|
}\n\n"
|
||||||
|
|
||||||
|
property string sum2: "
|
||||||
|
float sum2(vec2 v) {
|
||||||
|
return v.x + v.y;
|
||||||
|
}\n\n"
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -23,62 +23,78 @@ import QtGraphicalEffects 1.0
|
|||||||
|
|
||||||
import "utils.js" as Utils
|
import "utils.js" as Utils
|
||||||
|
|
||||||
ShaderEffect {
|
Item {
|
||||||
property ShaderEffectSource source
|
property ShaderEffectSource source
|
||||||
property ShaderEffectSource blurredSource
|
property BurnInEffect burnInEffect
|
||||||
property ShaderEffectSource bloomSource
|
property ShaderEffectSource bloomSource
|
||||||
|
|
||||||
property color fontColor: appSettings.fontColor
|
property color fontColor: appSettings.fontColor
|
||||||
property color backgroundColor: appSettings.backgroundColor
|
property color backgroundColor: appSettings.backgroundColor
|
||||||
property real bloom: appSettings.bloom * 2.5
|
|
||||||
|
|
||||||
property real burnIn: appSettings.burnIn
|
property real screenCurvature: appSettings.screenCurvature * appSettings.screenCurvatureSize
|
||||||
|
|
||||||
property real jitter: appSettings.jitter * 0.007
|
property real chromaColor: appSettings.chromaColor
|
||||||
property real staticNoise: appSettings.staticNoise
|
|
||||||
property size scaleNoiseSize: Qt.size((width) / (noiseTexture.width * appSettings.windowScaling * appSettings.fontScaling),
|
|
||||||
(height) / (noiseTexture.height * appSettings.windowScaling * appSettings.fontScaling))
|
|
||||||
|
|
||||||
property real screenCurvature: appSettings.screenCurvature
|
property real ambientLight: appSettings.ambientLight * 0.2
|
||||||
property real glowingLine: appSettings.glowingLine * 0.2
|
|
||||||
|
|
||||||
property real chromaColor: appSettings.chromaColor;
|
property real frameGloss: appSettings.frameGloss
|
||||||
|
|
||||||
property real rbgShift: appSettings.rbgShift * 0.2
|
property size virtualResolution
|
||||||
|
property size screenResolution
|
||||||
|
|
||||||
|
property real _screenDensity: Math.min(
|
||||||
|
screenResolution.width / virtualResolution.width,
|
||||||
|
screenResolution.height / virtualResolution.height
|
||||||
|
)
|
||||||
|
|
||||||
|
ShaderEffect {
|
||||||
|
id: dynamicShader
|
||||||
|
|
||||||
|
property ShaderLibrary shaderLibrary: ShaderLibrary { }
|
||||||
|
|
||||||
|
property ShaderEffectSource screenBuffer: frameBuffer
|
||||||
|
property ShaderEffectSource burnInSource: burnInEffect.source
|
||||||
|
property ShaderEffectSource frameSource: terminalFrameLoader.item
|
||||||
|
|
||||||
|
property color fontColor: parent.fontColor
|
||||||
|
property color backgroundColor: parent.backgroundColor
|
||||||
|
property real screenCurvature: parent.screenCurvature
|
||||||
|
property real chromaColor: parent.chromaColor
|
||||||
|
property real ambientLight: parent.ambientLight
|
||||||
|
property real frameGloss: parent.frameGloss
|
||||||
|
|
||||||
property real flickering: appSettings.flickering
|
property real flickering: appSettings.flickering
|
||||||
property real horizontalSync: appSettings.horizontalSync * 0.5
|
property real horizontalSync: appSettings.horizontalSync
|
||||||
|
property real horizontalSyncStrength: Utils.lint(0.05, 0.35, horizontalSync)
|
||||||
|
property real glowingLine: appSettings.glowingLine * 0.2
|
||||||
|
|
||||||
property bool frameReflections: appSettings.frameReflections
|
// Fast burnin properties
|
||||||
|
property real burnIn: appSettings.burnIn
|
||||||
|
property real burnInLastUpdate: burnInEffect.lastUpdate
|
||||||
|
property real burnInTime: burnInEffect.burnInFadeTime
|
||||||
|
|
||||||
property real disp_top: (frame.displacementTop * appSettings.windowScaling) / height
|
property real jitter: appSettings.jitter
|
||||||
property real disp_bottom: (frame.displacementBottom * appSettings.windowScaling) / height
|
property size jitterDisplacement: Qt.size(0.007 * jitter, 0.002 * jitter)
|
||||||
property real disp_left: (frame.displacementLeft * appSettings.windowScaling) / width
|
property real shadowLength: 0.25 * screenCurvature * Utils.lint(0.50, 1.5, ambientLight)
|
||||||
property real disp_right: (frame.displacementRight * appSettings.windowScaling) / width
|
property real staticNoise: appSettings.staticNoise
|
||||||
|
property size scaleNoiseSize: Qt.size((width * 0.75) / (noiseTexture.width * appSettings.windowScaling * appSettings.totalFontScaling),
|
||||||
|
(height * 0.75) / (noiseTexture.height * appSettings.windowScaling * appSettings.totalFontScaling))
|
||||||
|
|
||||||
property real screen_brightness: appSettings.brightness * 1.5 + 0.5
|
property size virtualResolution: parent.virtualResolution
|
||||||
|
|
||||||
// This is the average value of the abs(sin) function. Needed to avoid aliasing.
|
// Rasterization might display oversamping issues if virtual resolution is close to physical display resolution.
|
||||||
readonly property real absSinAvg: 0.63661828335466886
|
// We progressively disable rasterization from 4x up to 2x resolution.
|
||||||
property size rasterizationSmooth: Qt.size(
|
property real rasterizationIntensity: Utils.smoothstep(2.0, 4.0, _screenDensity)
|
||||||
Utils.clamp(2.0 * virtual_resolution.width / (width * devicePixelRatio), 0.0, 1.0),
|
|
||||||
Utils.clamp(2.0 * virtual_resolution.height / (height * devicePixelRatio), 0.0, 1.0))
|
|
||||||
|
|
||||||
property real dispX
|
property real displayTerminalFrame: appSettings._frameMargin > 0 || appSettings.screenCurvature > 0
|
||||||
property real dispY
|
|
||||||
property size virtual_resolution
|
|
||||||
|
|
||||||
TimeManager{
|
property real time: timeManager.time
|
||||||
id: timeManager
|
|
||||||
enableTimer: terminalWindow.visible
|
|
||||||
}
|
|
||||||
|
|
||||||
property alias time: timeManager.time
|
|
||||||
property ShaderEffectSource noiseSource: noiseShaderSource
|
property ShaderEffectSource noiseSource: noiseShaderSource
|
||||||
|
|
||||||
// If something goes wrong activate the fallback version of the shader.
|
// If something goes wrong activate the fallback version of the shader.
|
||||||
property bool fallBack: false
|
property bool fallBack: false
|
||||||
|
|
||||||
|
anchors.fill: parent
|
||||||
blending: false
|
blending: false
|
||||||
|
|
||||||
//Smooth random texture used for flickering effect.
|
//Smooth random texture used for flickering effect.
|
||||||
@ -107,11 +123,6 @@ ShaderEffect {
|
|||||||
uniform highp mat4 qt_Matrix;
|
uniform highp mat4 qt_Matrix;
|
||||||
uniform highp float time;
|
uniform highp float time;
|
||||||
|
|
||||||
uniform highp float disp_left;
|
|
||||||
uniform highp float disp_right;
|
|
||||||
uniform highp float disp_top;
|
|
||||||
uniform highp float disp_bottom;
|
|
||||||
|
|
||||||
attribute highp vec4 qt_Vertex;
|
attribute highp vec4 qt_Vertex;
|
||||||
attribute highp vec2 qt_MultiTexCoord0;
|
attribute highp vec2 qt_MultiTexCoord0;
|
||||||
|
|
||||||
@ -123,27 +134,28 @@ ShaderEffect {
|
|||||||
(!fallBack && flickering !== 0.0 ?"
|
(!fallBack && flickering !== 0.0 ?"
|
||||||
varying lowp float brightness;
|
varying lowp float brightness;
|
||||||
uniform lowp float flickering;" : "") +
|
uniform lowp float flickering;" : "") +
|
||||||
|
|
||||||
(!fallBack && horizontalSync !== 0.0 ?"
|
(!fallBack && horizontalSync !== 0.0 ?"
|
||||||
uniform lowp float horizontalSync;
|
uniform lowp float horizontalSyncStrength;
|
||||||
varying lowp float distortionScale;
|
varying lowp float distortionScale;
|
||||||
varying lowp float distortionFreq;" : "") +
|
varying lowp float distortionFreq;" : "") +
|
||||||
|
|
||||||
"
|
"
|
||||||
void main() {
|
void main() {
|
||||||
qt_TexCoord0.x = (qt_MultiTexCoord0.x - disp_left) / (1.0 - disp_left - disp_right);
|
qt_TexCoord0 = qt_MultiTexCoord0;
|
||||||
qt_TexCoord0.y = (qt_MultiTexCoord0.y - disp_top) / (1.0 - disp_top - disp_bottom);
|
|
||||||
vec2 coords = vec2(fract(time/(1024.0*2.0)), fract(time/(1024.0*1024.0)));" +
|
vec2 coords = vec2(fract(time/(1024.0*2.0)), fract(time/(1024.0*1024.0)));" +
|
||||||
|
|
||||||
(!fallBack && (flickering !== 0.0 || horizontalSync !== 0.0) ?
|
(!fallBack && (flickering !== 0.0 || horizontalSync !== 0.0) ?
|
||||||
"vec4 initialNoiseTexel = texture2D(noiseSource, coords);"
|
"vec4 initialNoiseTexel = texture2D(noiseSource, coords);"
|
||||||
: "") +
|
: "") +
|
||||||
|
|
||||||
(!fallBack && flickering !== 0.0 ? "
|
(!fallBack && flickering !== 0.0 ? "
|
||||||
brightness = 1.0 + (initialNoiseTexel.g - 0.5) * flickering;"
|
brightness = 1.0 + (initialNoiseTexel.g - 0.5) * flickering;"
|
||||||
: "") +
|
: "") +
|
||||||
|
|
||||||
(!fallBack && horizontalSync !== 0.0 ? "
|
(!fallBack && horizontalSync !== 0.0 ? "
|
||||||
float randval = horizontalSync - initialNoiseTexel.r;
|
float randval = horizontalSyncStrength - initialNoiseTexel.r;
|
||||||
distortionScale = step(0.0, randval) * randval * horizontalSync;
|
distortionScale = step(0.0, randval) * randval * horizontalSyncStrength;
|
||||||
distortionFreq = mix(4.0, 40.0, initialNoiseTexel.g);"
|
distortionFreq = mix(4.0, 40.0, initialNoiseTexel.g);"
|
||||||
: "") +
|
: "") +
|
||||||
|
|
||||||
@ -155,31 +167,30 @@ ShaderEffect {
|
|||||||
precision mediump float;
|
precision mediump float;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
uniform sampler2D source;
|
uniform sampler2D screenBuffer;
|
||||||
uniform highp float qt_Opacity;
|
uniform highp float qt_Opacity;
|
||||||
uniform highp float time;
|
uniform highp float time;
|
||||||
varying highp vec2 qt_TexCoord0;
|
varying highp vec2 qt_TexCoord0;
|
||||||
|
|
||||||
uniform highp vec4 fontColor;
|
uniform highp vec4 fontColor;
|
||||||
uniform highp vec4 backgroundColor;
|
uniform highp vec4 backgroundColor;
|
||||||
uniform lowp float screen_brightness;
|
uniform lowp float shadowLength;
|
||||||
|
uniform lowp float frameGloss;
|
||||||
|
|
||||||
uniform highp vec2 virtual_resolution;
|
uniform highp vec2 virtualResolution;
|
||||||
uniform highp vec2 rasterizationSmooth;
|
uniform lowp float rasterizationIntensity;\n" +
|
||||||
uniform highp float dispX;
|
|
||||||
uniform highp float dispY;" +
|
|
||||||
|
|
||||||
(bloom !== 0 ? "
|
|
||||||
uniform highp sampler2D bloomSource;
|
|
||||||
uniform lowp float bloom;" : "") +
|
|
||||||
(burnIn !== 0 ? "
|
(burnIn !== 0 ? "
|
||||||
uniform sampler2D blurredSource;" : "") +
|
uniform sampler2D burnInSource;
|
||||||
|
uniform highp float burnInLastUpdate;
|
||||||
|
uniform highp float burnInTime;" : "") +
|
||||||
(staticNoise !== 0 ? "
|
(staticNoise !== 0 ? "
|
||||||
uniform highp float staticNoise;" : "") +
|
uniform highp float staticNoise;" : "") +
|
||||||
(((staticNoise !== 0 || jitter !== 0 || rbgShift)
|
(((staticNoise !== 0 || jitter !== 0) ||(fallBack && (flickering || horizontalSync))) ? "
|
||||||
||(fallBack && (flickering || horizontalSync))) ? "
|
|
||||||
uniform lowp sampler2D noiseSource;
|
uniform lowp sampler2D noiseSource;
|
||||||
uniform highp vec2 scaleNoiseSize;" : "") +
|
uniform highp vec2 scaleNoiseSize;" : "") +
|
||||||
|
(displayTerminalFrame ? "
|
||||||
|
uniform lowp sampler2D frameSource;" : "") +
|
||||||
(screenCurvature !== 0 ? "
|
(screenCurvature !== 0 ? "
|
||||||
uniform highp float screenCurvature;" : "") +
|
uniform highp float screenCurvature;" : "") +
|
||||||
(glowingLine !== 0 ? "
|
(glowingLine !== 0 ? "
|
||||||
@ -187,12 +198,12 @@ ShaderEffect {
|
|||||||
(chromaColor !== 0 ? "
|
(chromaColor !== 0 ? "
|
||||||
uniform lowp float chromaColor;" : "") +
|
uniform lowp float chromaColor;" : "") +
|
||||||
(jitter !== 0 ? "
|
(jitter !== 0 ? "
|
||||||
uniform lowp float jitter;" : "") +
|
uniform lowp vec2 jitterDisplacement;" : "") +
|
||||||
(rbgShift !== 0 ? "
|
(ambientLight !== 0 ? "
|
||||||
uniform lowp float rbgShift;" : "") +
|
uniform lowp float ambientLight;" : "") +
|
||||||
|
|
||||||
(fallBack && horizontalSync !== 0 ? "
|
(fallBack && horizontalSync !== 0 ? "
|
||||||
uniform lowp float horizontalSync;" : "") +
|
uniform lowp float horizontalSyncStrength;" : "") +
|
||||||
(fallBack && flickering !== 0.0 ?"
|
(fallBack && flickering !== 0.0 ?"
|
||||||
uniform lowp float flickering;" : "") +
|
uniform lowp float flickering;" : "") +
|
||||||
(!fallBack && flickering !== 0 ? "
|
(!fallBack && flickering !== 0 ? "
|
||||||
@ -204,24 +215,47 @@ ShaderEffect {
|
|||||||
|
|
||||||
(glowingLine !== 0 ? "
|
(glowingLine !== 0 ? "
|
||||||
float randomPass(vec2 coords){
|
float randomPass(vec2 coords){
|
||||||
return fract(smoothstep(-120.0, 0.0, coords.y - (virtual_resolution.y + 120.0) * fract(time * 0.00015)));
|
return fract(smoothstep(-120.0, 0.0, coords.y - (virtualResolution.y + 120.0) * fract(time * 0.00015)));
|
||||||
}" : "") +
|
}" : "") +
|
||||||
|
|
||||||
"highp float getScanlineIntensity(vec2 coords) {
|
shaderLibrary.min2 +
|
||||||
highp float result = 1.0;" +
|
shaderLibrary.rgb2grey +
|
||||||
|
shaderLibrary.rasterizationShader +
|
||||||
|
|
||||||
(appSettings.rasterization != appSettings.no_rasterization ?
|
"
|
||||||
"float val = abs(sin(coords.y * virtual_resolution.y * "+Math.PI+"));
|
float isInScreen(vec2 v) {
|
||||||
result *= mix(val, " + absSinAvg + ", rasterizationSmooth.y);" : "") +
|
return min2(step(0.0, v) - step(1.0, v));
|
||||||
(appSettings.rasterization == appSettings.pixel_rasterization ?
|
|
||||||
"val = abs(sin(coords.x * virtual_resolution.x * "+Math.PI+"));
|
|
||||||
result *= mix(val, " + absSinAvg + ", rasterizationSmooth.x);" : "") + "
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
float rgb2grey(vec3 v){
|
vec2 barrel(vec2 v, vec2 cc) {" +
|
||||||
return dot(v, vec3(0.21, 0.72, 0.04));
|
|
||||||
|
(screenCurvature !== 0 ? "
|
||||||
|
float distortion = dot(cc, cc) * screenCurvature;
|
||||||
|
return (v - cc * (1.0 + distortion) * distortion);"
|
||||||
|
:
|
||||||
|
"return v;") +
|
||||||
|
"}" +
|
||||||
|
|
||||||
|
"vec3 convertWithChroma(vec3 inColor) {
|
||||||
|
vec3 outColor = inColor;" +
|
||||||
|
|
||||||
|
(chromaColor !== 0 ?
|
||||||
|
"outColor = fontColor.rgb * mix(vec3(rgb2grey(inColor)), inColor, chromaColor);"
|
||||||
|
:
|
||||||
|
"outColor = fontColor.rgb * rgb2grey(inColor);") +
|
||||||
|
|
||||||
|
" return outColor;
|
||||||
|
}" +
|
||||||
|
|
||||||
|
//pseudo-random vector
|
||||||
|
//https://stackoverflow.com/a/10625698
|
||||||
|
"float random( vec2 p )
|
||||||
|
{
|
||||||
|
vec2 K1 = vec2(
|
||||||
|
23.14069263277926, // e^pi (Gelfond's constant)
|
||||||
|
2.665144142690225 // 2^sqrt(2) (Gelfond-Schneider constant)
|
||||||
|
);
|
||||||
|
return fract( cos( dot(p,K1) ) * 12345.6789 );
|
||||||
}" +
|
}" +
|
||||||
|
|
||||||
"void main() {" +
|
"void main() {" +
|
||||||
@ -237,8 +271,8 @@ ShaderEffect {
|
|||||||
float brightness = 1.0 + (initialNoiseTexel.g - 0.5) * flickering;"
|
float brightness = 1.0 + (initialNoiseTexel.g - 0.5) * flickering;"
|
||||||
: "") +
|
: "") +
|
||||||
(fallBack && horizontalSync !== 0.0 ? "
|
(fallBack && horizontalSync !== 0.0 ? "
|
||||||
float randval = horizontalSync - initialNoiseTexel.r;
|
float randval = horizontalSyncStrength - initialNoiseTexel.r;
|
||||||
float distortionScale = step(0.0, randval) * randval * horizontalSync;
|
float distortionScale = step(0.0, randval) * randval * horizontalSyncStrength;
|
||||||
float distortionFreq = mix(4.0, 40.0, initialNoiseTexel.g);"
|
float distortionFreq = mix(4.0, 40.0, initialNoiseTexel.g);"
|
||||||
: "") +
|
: "") +
|
||||||
|
|
||||||
@ -246,82 +280,70 @@ ShaderEffect {
|
|||||||
float noise = staticNoise;" : "") +
|
float noise = staticNoise;" : "") +
|
||||||
|
|
||||||
(screenCurvature !== 0 ? "
|
(screenCurvature !== 0 ? "
|
||||||
float distortion = dot(cc, cc) * screenCurvature;
|
vec2 staticCoords = barrel(qt_TexCoord0, cc);"
|
||||||
vec2 staticCoords = (qt_TexCoord0 - cc * (1.0 + distortion) * distortion);"
|
|
||||||
:"
|
:"
|
||||||
vec2 staticCoords = qt_TexCoord0;") +
|
vec2 staticCoords = qt_TexCoord0;") +
|
||||||
|
|
||||||
"vec2 coords = staticCoords;" +
|
"vec2 coords = qt_TexCoord0;" +
|
||||||
|
|
||||||
(horizontalSync !== 0 ? "
|
(horizontalSync !== 0 ? "
|
||||||
float dst = sin((coords.y + time * 0.001) * distortionFreq);
|
float dst = sin((coords.y + time * 0.001) * distortionFreq);
|
||||||
coords.x += dst * distortionScale;" +
|
coords.x += dst * distortionScale;" +
|
||||||
|
|
||||||
(staticNoise ? "
|
(staticNoise ? "
|
||||||
noise += distortionScale * 7.0;" : "")
|
noise += distortionScale * 7.0;" : "")
|
||||||
|
|
||||||
: "") +
|
: "") +
|
||||||
|
|
||||||
(jitter !== 0 || staticNoise !== 0 ?
|
(jitter !== 0 || staticNoise !== 0 ?
|
||||||
"vec4 noiseTexel = texture2D(noiseSource, scaleNoiseSize * coords + vec2(fract(time / 51.0), fract(time / 237.0)));"
|
"vec4 noiseTexel = texture2D(
|
||||||
|
noiseSource, scaleNoiseSize * coords
|
||||||
|
+ vec2(0.0, random(vec2(fract(time / 237.0), 822.9582)))
|
||||||
|
+ vec2(fract(time / 31.0), fract(time / 177.0))
|
||||||
|
);"
|
||||||
: "") +
|
: "") +
|
||||||
|
|
||||||
(jitter !== 0 ? "
|
(jitter !== 0 ? "
|
||||||
vec2 offset = vec2(noiseTexel.b, noiseTexel.a) - vec2(0.5);
|
vec2 offset = vec2(noiseTexel.b, noiseTexel.a) - vec2(0.5);
|
||||||
vec2 txt_coords = coords + offset * jitter;"
|
vec2 txt_coords = coords + offset * jitterDisplacement;"
|
||||||
: "vec2 txt_coords = coords;") +
|
: "vec2 txt_coords = coords;") +
|
||||||
|
|
||||||
"float color = 0.0;" +
|
"float color = 0.0001;" +
|
||||||
|
|
||||||
(staticNoise !== 0 ? "
|
(staticNoise !== 0 ? "
|
||||||
float noiseVal = noiseTexel.a;
|
float noiseVal = noiseTexel.a;
|
||||||
color += noiseVal * noise * (1.0 - distance * 1.3);" : "") +
|
color += noiseVal * noise * (1.0 - distance * 1.3);" : "") +
|
||||||
|
|
||||||
(glowingLine !== 0 ? "
|
(glowingLine !== 0 ? "
|
||||||
color += randomPass(coords * virtual_resolution) * glowingLine;" : "") +
|
color += randomPass(coords * virtualResolution) * glowingLine;" : "") +
|
||||||
|
|
||||||
"vec3 txt_color = texture2D(source, txt_coords).rgb;" +
|
"vec3 txt_color = texture2D(screenBuffer, txt_coords).rgb;" +
|
||||||
|
|
||||||
(burnIn !== 0 ? "
|
(burnIn !== 0 ? "
|
||||||
vec4 txt_blur = texture2D(blurredSource, txt_coords);
|
vec4 txt_blur = texture2D(burnInSource, staticCoords);
|
||||||
txt_color = txt_color + txt_blur.rgb * txt_blur.a;"
|
float blurDecay = clamp((time - burnInLastUpdate) * burnInTime, 0.0, 1.0);
|
||||||
|
vec3 burnInColor = 0.65 * (txt_blur.rgb - vec3(blurDecay));
|
||||||
|
txt_color = max(txt_color, convertWithChroma(burnInColor));"
|
||||||
: "") +
|
: "") +
|
||||||
|
|
||||||
"float greyscale_color = rgb2grey(txt_color) + color;" +
|
"txt_color += fontColor.rgb * vec3(color);" +
|
||||||
|
|
||||||
(chromaColor !== 0 ?
|
"txt_color = applyRasterization(staticCoords, txt_color, virtualResolution, rasterizationIntensity);\n" +
|
||||||
(rbgShift !== 0 ? "
|
|
||||||
float rgb_noise = abs(texture2D(noiseSource, vec2(fract(time/(1024.0 * 256.0)), fract(time/(1024.0*1024.0)))).a - 0.5);
|
|
||||||
float rcolor = texture2D(source, txt_coords + vec2(0.1, 0.0) * rbgShift * rgb_noise).r;
|
|
||||||
float bcolor = texture2D(source, txt_coords - vec2(0.1, 0.0) * rbgShift * rgb_noise).b;
|
|
||||||
txt_color.r = rcolor;
|
|
||||||
txt_color.b = bcolor;
|
|
||||||
greyscale_color = 0.33 * (rcolor + bcolor);" : "") +
|
|
||||||
|
|
||||||
"vec3 mixedColor = mix(fontColor.rgb, txt_color * fontColor.rgb, chromaColor);
|
"vec3 finalColor = txt_color;" +
|
||||||
vec3 finalBackColor = mix(backgroundColor.rgb, mixedColor, greyscale_color);
|
|
||||||
vec3 finalColor = mix(finalBackColor, fontColor.rgb, color).rgb;"
|
|
||||||
:
|
|
||||||
"vec3 finalColor = mix(backgroundColor.rgb, fontColor.rgb, greyscale_color);") +
|
|
||||||
|
|
||||||
"finalColor *= getScanlineIntensity(coords);" +
|
|
||||||
|
|
||||||
(bloom !== 0 ?
|
|
||||||
"vec4 bloomFullColor = texture2D(bloomSource, coords);
|
|
||||||
vec3 bloomColor = bloomFullColor.rgb;
|
|
||||||
float bloomAlpha = bloomFullColor.a;" +
|
|
||||||
(chromaColor !== 0 ?
|
|
||||||
"bloomColor = fontColor.rgb * mix(vec3(rgb2grey(bloomColor)), bloomColor, chromaColor);"
|
|
||||||
:
|
|
||||||
"bloomColor = fontColor.rgb * rgb2grey(bloomColor);") +
|
|
||||||
"finalColor += bloomColor * bloom * bloomAlpha;"
|
|
||||||
: "") +
|
|
||||||
|
|
||||||
"finalColor *= smoothstep(-dispX, 0.0, staticCoords.x) - smoothstep(1.0, 1.0 + dispX, staticCoords.x);
|
|
||||||
finalColor *= smoothstep(-dispY, 0.0, staticCoords.y) - smoothstep(1.0, 1.0 + dispY, staticCoords.y);" +
|
|
||||||
|
|
||||||
(flickering !== 0 ? "
|
(flickering !== 0 ? "
|
||||||
finalColor *= brightness;" : "") +
|
finalColor *= brightness;" : "") +
|
||||||
|
|
||||||
"gl_FragColor = vec4(finalColor * screen_brightness, qt_Opacity);" +
|
(ambientLight !== 0 ? "
|
||||||
|
finalColor += vec3(ambientLight) * (1.0 - distance) * (1.0 - distance);" : "") +
|
||||||
|
|
||||||
|
(displayTerminalFrame ?
|
||||||
|
"vec4 frameColor = texture2D(frameSource, qt_TexCoord0);
|
||||||
|
finalColor = mix(finalColor, frameColor.rgb + (finalColor*frameGloss), frameColor.a);"
|
||||||
|
: "") +
|
||||||
|
|
||||||
|
"gl_FragColor = vec4(finalColor, qt_Opacity);" +
|
||||||
"}"
|
"}"
|
||||||
|
|
||||||
onStatusChanged: {
|
onStatusChanged: {
|
||||||
@ -335,3 +357,176 @@ ShaderEffect {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loader {
|
||||||
|
id: terminalFrameLoader
|
||||||
|
|
||||||
|
active: dynamicShader.displayTerminalFrame
|
||||||
|
|
||||||
|
width: staticShader.width
|
||||||
|
height: staticShader.height
|
||||||
|
|
||||||
|
sourceComponent: ShaderEffectSource {
|
||||||
|
|
||||||
|
sourceItem: terminalFrame
|
||||||
|
hideSource: true
|
||||||
|
visible: false
|
||||||
|
format: ShaderEffectSource.RGBA
|
||||||
|
|
||||||
|
TerminalFrame {
|
||||||
|
id: terminalFrame
|
||||||
|
blending: false
|
||||||
|
anchors.fill: parent
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ShaderLibrary {
|
||||||
|
id: shaderLibrary
|
||||||
|
}
|
||||||
|
|
||||||
|
ShaderEffect {
|
||||||
|
id: staticShader
|
||||||
|
|
||||||
|
width: parent.width * appSettings.windowScaling
|
||||||
|
height: parent.height * appSettings.windowScaling
|
||||||
|
|
||||||
|
property ShaderEffectSource source: parent.source
|
||||||
|
property ShaderEffectSource bloomSource: parent.bloomSource
|
||||||
|
|
||||||
|
property color fontColor: parent.fontColor
|
||||||
|
property color backgroundColor: parent.backgroundColor
|
||||||
|
property real bloom: appSettings.bloom * 2.5
|
||||||
|
|
||||||
|
property real screenCurvature: parent.screenCurvature
|
||||||
|
|
||||||
|
property real chromaColor: appSettings.chromaColor;
|
||||||
|
|
||||||
|
property real rbgShift: (appSettings.rbgShift / width) * appSettings.totalFontScaling // TODO FILIPPO width here is wrong.
|
||||||
|
|
||||||
|
property int rasterization: appSettings.rasterization
|
||||||
|
|
||||||
|
property real screen_brightness: Utils.lint(0.5, 1.5, appSettings.brightness)
|
||||||
|
|
||||||
|
property real ambientLight: parent.ambientLight
|
||||||
|
|
||||||
|
property size virtualResolution: parent.virtualResolution
|
||||||
|
|
||||||
|
blending: false
|
||||||
|
visible: false
|
||||||
|
|
||||||
|
//Print the number with a reasonable precision for the shader.
|
||||||
|
function str(num){
|
||||||
|
return num.toFixed(8);
|
||||||
|
}
|
||||||
|
|
||||||
|
fragmentShader: "
|
||||||
|
#ifdef GL_ES
|
||||||
|
precision mediump float;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
uniform sampler2D source;
|
||||||
|
uniform highp float qt_Opacity;
|
||||||
|
varying highp vec2 qt_TexCoord0;
|
||||||
|
|
||||||
|
uniform highp vec4 fontColor;
|
||||||
|
uniform highp vec4 backgroundColor;
|
||||||
|
uniform lowp float screen_brightness;
|
||||||
|
|
||||||
|
uniform highp vec2 virtualResolution;" +
|
||||||
|
|
||||||
|
(bloom !== 0 ? "
|
||||||
|
uniform highp sampler2D bloomSource;
|
||||||
|
uniform lowp float bloom;" : "") +
|
||||||
|
|
||||||
|
(screenCurvature !== 0 ? "
|
||||||
|
uniform highp float screenCurvature;" : "") +
|
||||||
|
|
||||||
|
(chromaColor !== 0 ? "
|
||||||
|
uniform lowp float chromaColor;" : "") +
|
||||||
|
|
||||||
|
(rbgShift !== 0 ? "
|
||||||
|
uniform lowp float rbgShift;" : "") +
|
||||||
|
|
||||||
|
(ambientLight !== 0 ? "
|
||||||
|
uniform lowp float ambientLight;" : "") +
|
||||||
|
|
||||||
|
shaderLibrary.min2 +
|
||||||
|
shaderLibrary.sum2 +
|
||||||
|
shaderLibrary.rgb2grey +
|
||||||
|
|
||||||
|
"vec3 convertWithChroma(vec3 inColor) {
|
||||||
|
vec3 outColor = inColor;" +
|
||||||
|
|
||||||
|
(chromaColor !== 0 ?
|
||||||
|
"outColor = fontColor.rgb * mix(vec3(rgb2grey(inColor)), inColor, chromaColor);"
|
||||||
|
:
|
||||||
|
"outColor = fontColor.rgb * rgb2grey(inColor);") +
|
||||||
|
|
||||||
|
" return outColor;
|
||||||
|
}" +
|
||||||
|
|
||||||
|
shaderLibrary.rasterizationShader +
|
||||||
|
|
||||||
|
"void main() {" +
|
||||||
|
"vec2 cc = vec2(0.5) - qt_TexCoord0;" +
|
||||||
|
|
||||||
|
(screenCurvature !== 0 ? "
|
||||||
|
float distortion = dot(cc, cc) * screenCurvature;
|
||||||
|
vec2 curvatureCoords = (qt_TexCoord0 - cc * (1.0 + distortion) * distortion);
|
||||||
|
vec2 txt_coords = - 2.0 * curvatureCoords + 3.0 * step(vec2(0.0), curvatureCoords) * curvatureCoords - 3.0 * step(vec2(1.0), curvatureCoords) * curvatureCoords;"
|
||||||
|
:"
|
||||||
|
vec2 txt_coords = qt_TexCoord0;") +
|
||||||
|
|
||||||
|
"vec3 txt_color = texture2D(source, txt_coords).rgb;" +
|
||||||
|
|
||||||
|
(rbgShift !== 0 ? "
|
||||||
|
vec2 displacement = vec2(12.0, 0.0) * rbgShift;
|
||||||
|
vec3 rightColor = texture2D(source, txt_coords + displacement).rgb;
|
||||||
|
vec3 leftColor = texture2D(source, txt_coords - displacement).rgb;
|
||||||
|
txt_color.r = leftColor.r * 0.10 + rightColor.r * 0.30 + txt_color.r * 0.60;
|
||||||
|
txt_color.g = leftColor.g * 0.20 + rightColor.g * 0.20 + txt_color.g * 0.60;
|
||||||
|
txt_color.b = leftColor.b * 0.30 + rightColor.b * 0.10 + txt_color.b * 0.60;
|
||||||
|
" : "") +
|
||||||
|
|
||||||
|
"txt_color += vec3(0.0001);" +
|
||||||
|
"float greyscale_color = rgb2grey(txt_color);" +
|
||||||
|
|
||||||
|
(screenCurvature !== 0 ? "
|
||||||
|
float reflectionMask = sum2(step(vec2(0.0), curvatureCoords) - step(vec2(1.0), curvatureCoords));
|
||||||
|
reflectionMask = clamp(reflectionMask, 0.0, 1.0);"
|
||||||
|
:
|
||||||
|
"float reflectionMask = 1.0;") +
|
||||||
|
|
||||||
|
(chromaColor !== 0 ?
|
||||||
|
"vec3 foregroundColor = mix(fontColor.rgb, txt_color * fontColor.rgb / greyscale_color, chromaColor);
|
||||||
|
vec3 finalColor = mix(backgroundColor.rgb, foregroundColor, greyscale_color * reflectionMask);"
|
||||||
|
:
|
||||||
|
"vec3 finalColor = mix(backgroundColor.rgb, fontColor.rgb, greyscale_color * reflectionMask);") +
|
||||||
|
|
||||||
|
(bloom !== 0 ?
|
||||||
|
"vec4 bloomFullColor = texture2D(bloomSource, txt_coords);
|
||||||
|
vec3 bloomColor = bloomFullColor.rgb;
|
||||||
|
float bloomAlpha = bloomFullColor.a;
|
||||||
|
bloomColor = convertWithChroma(bloomColor);
|
||||||
|
finalColor += clamp(bloomColor * bloom * bloomAlpha, 0.0, 0.5);"
|
||||||
|
: "") +
|
||||||
|
|
||||||
|
"finalColor *= screen_brightness;" +
|
||||||
|
|
||||||
|
"gl_FragColor = vec4(finalColor, qt_Opacity);" +
|
||||||
|
"}"
|
||||||
|
|
||||||
|
onStatusChanged: {
|
||||||
|
// Print warning messages
|
||||||
|
if (log) console.log(log);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ShaderEffectSource {
|
||||||
|
id: frameBuffer
|
||||||
|
visible: false
|
||||||
|
sourceItem: staticShader
|
||||||
|
hideSource: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,9 +17,8 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtQuick.Controls 1.1
|
import QtQuick.Controls 2.0
|
||||||
import QtQuick.Layouts 1.1
|
import QtQuick.Layouts 1.1
|
||||||
|
|
||||||
import "Components"
|
import "Components"
|
||||||
@ -27,8 +26,8 @@ import "Components"
|
|||||||
RowLayout {
|
RowLayout {
|
||||||
property alias value: slider.value
|
property alias value: slider.value
|
||||||
property alias stepSize: slider.stepSize
|
property alias stepSize: slider.stepSize
|
||||||
property alias minimumValue: slider.minimumValue
|
property alias minimumValue: slider.from
|
||||||
property alias maximumValue: slider.maximumValue
|
property alias maximumValue: slider.to
|
||||||
property real maxMultiplier: 100
|
property real maxMultiplier: 100
|
||||||
|
|
||||||
id: setting_component
|
id: setting_component
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,12 +17,12 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
property size terminalSize
|
property size terminalSize
|
||||||
property real topOpacity: 0.6
|
property real topOpacity: 0.6
|
||||||
|
|
||||||
width: textSize.width * 2
|
width: textSize.width * 2
|
||||||
height: textSize.height * 2
|
height: textSize.height * 2
|
||||||
radius: 5
|
radius: 5
|
||||||
@ -31,7 +31,11 @@ Rectangle{
|
|||||||
color: "black"
|
color: "black"
|
||||||
opacity: sizetimer.running ? 0.6 : 0.0
|
opacity: sizetimer.running ? 0.6 : 0.0
|
||||||
|
|
||||||
Behavior on opacity{NumberAnimation{duration: 200}}
|
Behavior on opacity {
|
||||||
|
NumberAnimation {
|
||||||
|
duration: 200
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onTerminalSizeChanged: sizetimer.restart()
|
onTerminalSizeChanged: sizetimer.restart()
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -22,10 +22,12 @@ import QtQuick 2.2
|
|||||||
import QtQuick.LocalStorage 2.0
|
import QtQuick.LocalStorage 2.0
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
|
readonly property string dbMajorVersion: "1"
|
||||||
|
readonly property string dbMinorVersion: "1.0"
|
||||||
property bool initialized: false
|
property bool initialized: false
|
||||||
|
|
||||||
function getDatabase() {
|
function getDatabase() {
|
||||||
return LocalStorage.openDatabaseSync("coolretroterm", "1.0", "StorageDatabase", 100000);
|
return LocalStorage.openDatabaseSync("coolretroterm" + dbMajorVersion, dbMinorVersion, "StorageDatabase", 100000)
|
||||||
}
|
}
|
||||||
|
|
||||||
function initialize() {
|
function initialize() {
|
||||||
@ -33,9 +35,10 @@ QtObject {
|
|||||||
db.transaction(
|
db.transaction(
|
||||||
function(tx) {
|
function(tx) {
|
||||||
tx.executeSql('CREATE TABLE IF NOT EXISTS settings(setting TEXT UNIQUE, value TEXT)');
|
tx.executeSql('CREATE TABLE IF NOT EXISTS settings(setting TEXT UNIQUE, value TEXT)');
|
||||||
});
|
}
|
||||||
|
)
|
||||||
|
|
||||||
initialized = true;
|
initialized = true
|
||||||
}
|
}
|
||||||
|
|
||||||
function setSetting(setting, value) {
|
function setSetting(setting, value) {
|
||||||
@ -43,7 +46,8 @@ QtObject {
|
|||||||
|
|
||||||
var db = getDatabase();
|
var db = getDatabase();
|
||||||
var res = "";
|
var res = "";
|
||||||
db.transaction(function(tx) {
|
db.transaction(
|
||||||
|
function(tx) {
|
||||||
var rs = tx.executeSql('INSERT OR REPLACE INTO settings VALUES (?,?);', [setting,value]);
|
var rs = tx.executeSql('INSERT OR REPLACE INTO settings VALUES (?,?);', [setting,value]);
|
||||||
//console.log(rs.rowsAffected)
|
//console.log(rs.rowsAffected)
|
||||||
if (rs.rowsAffected > 0) {
|
if (rs.rowsAffected > 0) {
|
||||||
@ -52,23 +56,25 @@ QtObject {
|
|||||||
res = "Error";
|
res = "Error";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
)
|
||||||
// The function returns “OK” if it was successful, or “Error” if it wasn't
|
// The function returns “OK” if it was successful, or “Error” if it wasn't
|
||||||
return res;
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
function getSetting(setting) {
|
function getSetting(setting) {
|
||||||
if(!initialized) initialize();
|
if(!initialized) initialize();
|
||||||
var db = getDatabase();
|
var db = getDatabase();
|
||||||
var res = "";
|
var res = "";
|
||||||
db.transaction(function(tx) {
|
db.transaction(
|
||||||
|
function(tx) {
|
||||||
var rs = tx.executeSql('SELECT value FROM settings WHERE setting=?;', [setting]);
|
var rs = tx.executeSql('SELECT value FROM settings WHERE setting=?;', [setting]);
|
||||||
if (rs.rows.length > 0) {
|
if (rs.rows.length > 0) {
|
||||||
res = rs.rows.item(0).value;
|
res = rs.rows.item(0).value;
|
||||||
} else {
|
} else {
|
||||||
res = undefined;
|
res = undefined;
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
|
)
|
||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -77,6 +83,7 @@ QtObject {
|
|||||||
db.transaction(
|
db.transaction(
|
||||||
function(tx) {
|
function(tx) {
|
||||||
tx.executeSql('DROP TABLE settings');
|
tx.executeSql('DROP TABLE settings');
|
||||||
});
|
}
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,3 +1,22 @@
|
|||||||
|
/*******************************************************************************
|
||||||
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
|
*
|
||||||
|
* This file is part of cool-retro-term.
|
||||||
|
*
|
||||||
|
* cool-retro-term 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 <http://www.gnu.org/licenses/>.
|
||||||
|
*******************************************************************************/
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
import QtGraphicalEffects 1.0
|
import QtGraphicalEffects 1.0
|
||||||
|
|
||||||
@ -7,31 +26,22 @@ ShaderTerminal{
|
|||||||
property alias title: terminal.title
|
property alias title: terminal.title
|
||||||
property alias terminalSize: terminal.terminalSize
|
property alias terminalSize: terminal.terminalSize
|
||||||
|
|
||||||
|
property real devicePixelRatio: terminalWindow.screen.devicePixelRatio
|
||||||
|
|
||||||
id: mainShader
|
id: mainShader
|
||||||
opacity: appSettings.windowOpacity * 0.3 + 0.7
|
opacity: appSettings.windowOpacity * 0.3 + 0.7
|
||||||
|
|
||||||
blending: false
|
|
||||||
|
|
||||||
source: terminal.mainSource
|
source: terminal.mainSource
|
||||||
blurredSource: terminal.blurredSource
|
burnInEffect: terminal.burnInEffect
|
||||||
dispX: (12 / width) * appSettings.windowScaling
|
virtualResolution: terminal.virtualResolution
|
||||||
dispY: (12 / height) * appSettings.windowScaling
|
screenResolution: Qt.size(
|
||||||
virtual_resolution: terminal.virtualResolution
|
terminalWindow.width * devicePixelRatio * appSettings.windowScaling,
|
||||||
|
terminalWindow.height * devicePixelRatio * appSettings.windowScaling
|
||||||
|
)
|
||||||
|
|
||||||
Loader{
|
TimeManager {
|
||||||
id: frame
|
id: timeManager
|
||||||
anchors.fill: parent
|
enableTimer: terminalWindow.visible
|
||||||
|
|
||||||
property real displacementLeft: item ? item.displacementLeft : 0
|
|
||||||
property real displacementTop: item ? item.displacementTop : 0
|
|
||||||
property real displacementRight: item ? item.displacementRight : 0
|
|
||||||
property real displacementBottom: item ? item.displacementBottom : 0
|
|
||||||
|
|
||||||
asynchronous: true
|
|
||||||
visible: status === Loader.Ready
|
|
||||||
|
|
||||||
z: 2.1
|
|
||||||
source: appSettings.frameSource
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PreprocessedTerminal {
|
PreprocessedTerminal {
|
||||||
@ -40,7 +50,6 @@ ShaderTerminal{
|
|||||||
}
|
}
|
||||||
|
|
||||||
// EFFECTS ////////////////////////////////////////////////////////////////
|
// EFFECTS ////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
Loader {
|
Loader {
|
||||||
id: bloomEffectLoader
|
id: bloomEffectLoader
|
||||||
active: appSettings.bloom
|
active: appSettings.bloom
|
||||||
@ -49,7 +58,7 @@ ShaderTerminal{
|
|||||||
height: parent.height * appSettings.bloomQuality
|
height: parent.height * appSettings.bloomQuality
|
||||||
|
|
||||||
sourceComponent: FastBlur {
|
sourceComponent: FastBlur {
|
||||||
radius: Utils.lint(16, 48, appSettings.bloomQuality * appSettings.windowScaling);
|
radius: Utils.lint(16, 64, appSettings.bloomQuality)
|
||||||
source: terminal.mainSource
|
source: terminal.mainSource
|
||||||
transparentBorder: true
|
transparentBorder: true
|
||||||
}
|
}
|
||||||
@ -68,65 +77,4 @@ ShaderTerminal{
|
|||||||
}
|
}
|
||||||
|
|
||||||
bloomSource: bloomSourceLoader.item
|
bloomSource: bloomSourceLoader.item
|
||||||
|
|
||||||
// This shader might be useful in the future. Since we used it only for a couple
|
|
||||||
// of calculations is probably best to move those in the main shader. If in the future
|
|
||||||
// we need to store another fullScreen channel this might be handy.
|
|
||||||
|
|
||||||
// ShaderEffect {
|
|
||||||
// id: rasterizationEffect
|
|
||||||
// width: parent.width
|
|
||||||
// height: parent.height
|
|
||||||
// property real outColor: 0.0
|
|
||||||
// property real dispX: (5 / width) * appSettings.windowScaling
|
|
||||||
// property real dispY: (5 / height) * appSettings.windowScaling
|
|
||||||
// property size virtual_resolution: terminal.virtualResolution
|
|
||||||
|
|
||||||
// blending: false
|
|
||||||
|
|
||||||
// fragmentShader:
|
|
||||||
// "uniform lowp float qt_Opacity;" +
|
|
||||||
|
|
||||||
// "varying highp vec2 qt_TexCoord0;
|
|
||||||
// uniform highp vec2 virtual_resolution;
|
|
||||||
// uniform highp float dispX;
|
|
||||||
// uniform highp float dispY;
|
|
||||||
// uniform mediump float outColor;
|
|
||||||
|
|
||||||
// highp float getScanlineIntensity(vec2 coords) {
|
|
||||||
// highp float result = 1.0;" +
|
|
||||||
|
|
||||||
// (appSettings.rasterization != appSettings.no_rasterization ?
|
|
||||||
// "result *= abs(sin(coords.y * virtual_resolution.y * "+Math.PI+"));" : "") +
|
|
||||||
// (appSettings.rasterization == appSettings.pixel_rasterization ?
|
|
||||||
// "result *= abs(sin(coords.x * virtual_resolution.x * "+Math.PI+"));" : "") + "
|
|
||||||
|
|
||||||
// return result;
|
|
||||||
// }" +
|
|
||||||
|
|
||||||
// "void main() {" +
|
|
||||||
// "highp float color = getScanlineIntensity(qt_TexCoord0);" +
|
|
||||||
|
|
||||||
// "float distance = length(vec2(0.5) - qt_TexCoord0);" +
|
|
||||||
// "color = mix(color, 0.0, 1.2 * distance * distance);" +
|
|
||||||
|
|
||||||
// "color *= outColor + smoothstep(0.00, dispX, qt_TexCoord0.x) * (1.0 - outColor);" +
|
|
||||||
// "color *= outColor + smoothstep(0.00, dispY, qt_TexCoord0.y) * (1.0 - outColor);" +
|
|
||||||
// "color *= outColor + (1.0 - smoothstep(1.00 - dispX, 1.00, qt_TexCoord0.x)) * (1.0 - outColor);" +
|
|
||||||
// "color *= outColor + (1.0 - smoothstep(1.00 - dispY, 1.00, qt_TexCoord0.y)) * (1.0 - outColor);" +
|
|
||||||
|
|
||||||
// "gl_FragColor.a = color;" +
|
|
||||||
// "}"
|
|
||||||
|
|
||||||
// onStatusChanged: if (log) console.log(log) //Print warning messages
|
|
||||||
// }
|
|
||||||
|
|
||||||
// rasterizationSource: ShaderEffectSource{
|
|
||||||
// id: rasterizationEffectSource
|
|
||||||
// sourceItem: rasterizationEffect
|
|
||||||
// hideSource: true
|
|
||||||
// smooth: true
|
|
||||||
// wrapMode: ShaderEffectSource.ClampToEdge
|
|
||||||
// visible: false
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
|
104
app/qml/TerminalFrame.qml
Normal file
104
app/qml/TerminalFrame.qml
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
/*******************************************************************************
|
||||||
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
|
*
|
||||||
|
* This file is part of cool-retro-term.
|
||||||
|
*
|
||||||
|
* cool-retro-term 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 <http://www.gnu.org/licenses/>.
|
||||||
|
*******************************************************************************/
|
||||||
|
import QtQuick 2.0
|
||||||
|
|
||||||
|
import "utils.js" as Utils
|
||||||
|
|
||||||
|
ShaderEffect {
|
||||||
|
property color _frameColor: appSettings.frameColor
|
||||||
|
property color _backgroundColor: appSettings.backgroundColor
|
||||||
|
property color _fontColor: appSettings.fontColor
|
||||||
|
property color _lightColor: Utils.mix(_fontColor, _backgroundColor, 0.2)
|
||||||
|
property real _ambientLight: Utils.lint(0.2, 0.8, appSettings.ambientLight)
|
||||||
|
|
||||||
|
property color frameColor: Utils.mix(_frameColor, _lightColor, _ambientLight)
|
||||||
|
property real screenCurvature: appSettings.screenCurvature * appSettings.screenCurvatureSize
|
||||||
|
|
||||||
|
// Coefficient of the log curve used to approximate shadowing
|
||||||
|
property real screenShadowCoeff: Utils.lint(20.0, 10.0, _ambientLight)
|
||||||
|
property real frameShadowCoeff: Utils.lint(20.0, 10.0, _ambientLight)
|
||||||
|
|
||||||
|
property size margin: Qt.size(
|
||||||
|
appSettings.frameMargin / width * appSettings.windowScaling,
|
||||||
|
appSettings.frameMargin / height * appSettings.windowScaling
|
||||||
|
)
|
||||||
|
|
||||||
|
ShaderLibrary {
|
||||||
|
id: shaderLibrary
|
||||||
|
}
|
||||||
|
|
||||||
|
fragmentShader: "
|
||||||
|
#ifdef GL_ES
|
||||||
|
precision mediump float;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
uniform lowp float screenCurvature;
|
||||||
|
uniform lowp float screenShadowCoeff;
|
||||||
|
uniform lowp float frameShadowCoeff;
|
||||||
|
uniform highp float qt_Opacity;
|
||||||
|
uniform lowp vec4 frameColor;
|
||||||
|
uniform mediump vec2 margin;
|
||||||
|
|
||||||
|
varying highp vec2 qt_TexCoord0;
|
||||||
|
|
||||||
|
vec2 distortCoordinates(vec2 coords){
|
||||||
|
vec2 cc = (coords - vec2(0.5));
|
||||||
|
float dist = dot(cc, cc) * screenCurvature;
|
||||||
|
return (coords + cc * (1.0 + dist) * dist);
|
||||||
|
}
|
||||||
|
" +
|
||||||
|
|
||||||
|
shaderLibrary.max2 +
|
||||||
|
shaderLibrary.min2 +
|
||||||
|
shaderLibrary.prod2 +
|
||||||
|
shaderLibrary.sum2 +
|
||||||
|
|
||||||
|
"
|
||||||
|
|
||||||
|
vec2 positiveLog(vec2 x) {
|
||||||
|
return clamp(log(x), vec2(0.0), vec2(100.0));
|
||||||
|
}
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
vec2 staticCoords = qt_TexCoord0;
|
||||||
|
vec2 coords = distortCoordinates(staticCoords) * (vec2(1.0) + margin * 2.0) - margin;
|
||||||
|
|
||||||
|
vec2 vignetteCoords = staticCoords * (1.0 - staticCoords.yx);
|
||||||
|
float vignette = pow(prod2(vignetteCoords) * 15.0, 0.25);
|
||||||
|
|
||||||
|
vec3 color = frameColor.rgb * vec3(1.0 - vignette);
|
||||||
|
float alpha = 0.0;
|
||||||
|
|
||||||
|
float frameShadow = max2(positiveLog(-coords * frameShadowCoeff + vec2(1.0)) + positiveLog(coords * frameShadowCoeff - (vec2(frameShadowCoeff) - vec2(1.0))));
|
||||||
|
frameShadow = clamp(sqrt(frameShadow), 0.0, 1.0);
|
||||||
|
color *= frameShadow;
|
||||||
|
alpha = sum2(1.0 - step(vec2(0.0), coords) + step(vec2(1.0), coords));
|
||||||
|
alpha = clamp(alpha, 0.0, 1.0);
|
||||||
|
alpha *= mix(1.0, 0.9, frameShadow);
|
||||||
|
|
||||||
|
float screenShadow = 1.0 - prod2(positiveLog(coords * screenShadowCoeff + vec2(1.0)) * positiveLog(-coords * screenShadowCoeff + vec2(screenShadowCoeff + 1.0)));
|
||||||
|
alpha = max(0.8 * screenShadow, alpha);
|
||||||
|
|
||||||
|
gl_FragColor = vec4(color * alpha, alpha);
|
||||||
|
}
|
||||||
|
"
|
||||||
|
|
||||||
|
onStatusChanged: if (log) console.log(log) //Print warning messages
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
* Copyright (c) 2013 "Filippo Scognamiglio"
|
* Copyright (c) 2013-2021 "Filippo Scognamiglio"
|
||||||
* https://github.com/Swordfish90/cool-retro-term
|
* https://github.com/Swordfish90/cool-retro-term
|
||||||
*
|
*
|
||||||
* This file is part of cool-retro-term.
|
* This file is part of cool-retro-term.
|
||||||
@ -17,7 +17,6 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
|
|
||||||
import QtQuick 2.2
|
import QtQuick 2.2
|
||||||
|
|
||||||
Timer {
|
Timer {
|
||||||
|
BIN
app/qml/fonts/1971-ibm-3278/3270-Regular.ttf
Normal file
BIN
app/qml/fonts/1971-ibm-3278/3270-Regular.ttf
Normal file
Binary file not shown.
Binary file not shown.
@ -33,7 +33,7 @@ packaged version may not be the latest version, but it's good enough for
|
|||||||
most purposes. For those who don't have the luxury of a proper
|
most purposes. For those who don't have the luxury of a proper
|
||||||
system-managed package, Adobe Type 1, TTF, OTF and WOFF versions are
|
system-managed package, Adobe Type 1, TTF, OTF and WOFF versions are
|
||||||
available for download on
|
available for download on
|
||||||
http://s3.amazonaws.com/3270font/3270_fonts_d250fd9.zip (although this
|
http://s3.amazonaws.com/3270font/3270_fonts_4cfe95c.zip (although this
|
||||||
URL may not always reflect the latest version).
|
URL may not always reflect the latest version).
|
||||||
|
|
||||||

|
https://raw.githubusercontent.com/wiki/rbanffy/3270font/powerline.png)
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
If you are running Windows, you'll probably need something like
|
If you are running Windows, you'll probably need something like
|
||||||
Cygwin, but, in the end, the font works correctly (with some very
|
Cygwin, but, in the end, the font works correctly (with some very
|
||||||
@ -82,13 +83,15 @@ Contributing
|
|||||||
|
|
||||||
I fear GitHub's pull-request mechanism may not be very
|
I fear GitHub's pull-request mechanism may not be very
|
||||||
FontForge-friendly. If you want to contribute (there are a lot of
|
FontForge-friendly. If you want to contribute (there are a lot of
|
||||||
missing glyphs, such as the APL set and most non-latin alphabets which
|
missing glyphs, such as most non-latin alphabets which most likely were
|
||||||
most likely were never built into 3270 terminals), the best workflow
|
never built into 3270 terminals), the best workflow would probably be to
|
||||||
would be to make add the encoding slots (if needed), add/make the
|
add the encoding slots (if needed), add/make the changes, remove the
|
||||||
changes, reencode it in "Unicode, Full", compact it and validate
|
unchanged glyphs and save it as a different file. If, in doubt, get in
|
||||||
it. Check if the `git diff` command gives out something sensible (does
|
touch and we will figure out how to do it right.
|
||||||
not change things you didn't intend to) and make a pull request. If, in
|
|
||||||
doubt, get in touch and we will figure out how to do it right.
|
In order to generate the sample image and the grids for FontForge,
|
||||||
|
you'll need a Python 3 environment with PIL or pillow installed. The
|
||||||
|
requirements.txt file lists everything you need to do it.
|
||||||
|
|
||||||
Known problems
|
Known problems
|
||||||
--------------
|
--------------
|
||||||
@ -99,3 +102,20 @@ symbols are at the end of the font, along with some glyphs useful for
|
|||||||
building others.
|
building others.
|
||||||
|
|
||||||
Please refer to http://x3270.bgp.nu/Charset.html for a complete map.
|
Please refer to http://x3270.bgp.nu/Charset.html for a complete map.
|
||||||
|
|
||||||
|
Future improvements
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
A grid generator is provided for producing various grid sizes for the
|
||||||
|
font. Those grids are not used yet, but they are intended to be used to
|
||||||
|
align font features to provide better rendering at common font size
|
||||||
|
choices. The captures below exemplify these choices:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|

|
||||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
BIN
app/qml/fonts/1979-atari-400-800/AtariClassic-Regular.ttf
Normal file
BIN
app/qml/fonts/1979-atari-400-800/AtariClassic-Regular.ttf
Normal file
Binary file not shown.
BIN
app/qml/fonts/1979-atari-400-800/Read Me.pdf
Normal file
BIN
app/qml/fonts/1979-atari-400-800/Read Me.pdf
Normal file
Binary file not shown.
@ -1,317 +0,0 @@
|
|||||||
{\rtf1\ansi\deff0\deftab720{\fonttbl{\f0\fswiss MS Sans Serif;}{\f1\froman\fcharset2 Symbol;}{\f2\fmodern\fprq1 Atari Classic Chunky;}{\f3\froman Times New Roman;}{\f4\fswiss\fprq2 Arial;}}
|
|
||||||
{\colortbl\red0\green0\blue0;\red0\green0\blue255;}
|
|
||||||
\deflang1033\pard\plain\f2\fs24\cf1 Atari Classic TrueType Fonts \plain\f2\fs24\cf0
|
|
||||||
\par \plain\f4\fs16\cf0 (Windows Version 1.1)
|
|
||||||
\par Created by Mark Simonson (v.1.0-1998, v.1.1-2001)
|
|
||||||
\par marksim@bitstream.net
|
|
||||||
\par Website: Mac/Atari Fusion--Atari Home Computer Resources for Mac Users
|
|
||||||
\par http://www2.bitstream.net/~marksim/atarimac/
|
|
||||||
\par Macintosh version also available.
|
|
||||||
\par
|
|
||||||
\par With these fonts installed, you can view and print Atari text files in any text editor that allows you to change fonts (WordPad, for example). Tip: In order to get the correct line breaks, you will need to change the ATASCII return character (155) to the DOS LF character. (In the Character Map accessory, the ATASCII return is the blank character that comes just before the inverse up-arrow.)
|
|
||||||
\par
|
|
||||||
\par There are three different fonts. \plain\f4\fs16\cf0\b Atari Classic Chunky \plain\f4\fs16\cf0 is a pixel-for-pixel copy of the original ATASCII character set. \plain\f4\fs16\cf0\b Atari Classic Smooth \plain\f4\fs16\cf0 interprets the pixel aliasing (stair steps) as diagonal lines. \plain\f4\fs16\cf0\b Atari Classic Extrasmooth \plain\f4\fs16\cf0 refines this idea further with the addition of curves. \plain\f4\fs16\cf0\b Smooth\plain\f4\fs16\cf0 and \plain\f4\fs16\cf0\b Extrasmooth\plain\f4\fs16\cf0 were designed for better appearance and legibility at larger sizes and on print-outs. Use the one that looks best to you.
|
|
||||||
\par
|
|
||||||
\par These fonts will tend to look uneven at font sizes that do not correspond to the 8-by-8 pixel grid that the characters are based on. Because Windows assumes 96ppi screen resolution, they will look best in a font size that is a multiple of 6 (i.e., 6pt, 12pt, 18pt, etc.). (In Windows, 6 points = 8 pixels.)
|
|
||||||
\par
|
|
||||||
\par The Atari Classic TrueType fonts duplicate the ATASCII character set on a low-level basis. Unlike a normal Windows font, ATASCII utilizes all character codes from $00 to $FF (0 to 255). The lower half are normal characters; the upper half are inverse versions of the lower half. The basic ASCII characters ($00 to $7F) correspond fairly closely except for the first 32, which don't normally contain characters in a Windows font.
|
|
||||||
\par
|
|
||||||
\par Due to differences between the way Windows and the Atari use character codes, not all characters will display properly in Windows. In fact, some characters will not display at all (though they do exist in the font). Unfortunately, this is due to certain character codes being reserved in Windows and there doesn't appear to be any way to work around it. The character codes affected are: $00-$1F (0-31), $7F-$81 (127-129), $8D-$90 (141-144), $9D (157), and $9F (158).
|
|
||||||
\par
|
|
||||||
\par Not all characters can be typed from the keyboard. You can however copy characters as needed from this document (see tables below). The Character Map desk accessory can help also.
|
|
||||||
\par
|
|
||||||
\par \plain\f4\fs16\cf0\b ATASCII CHARACTER SET TABLES
|
|
||||||
\par \plain\f4\fs16\cf0
|
|
||||||
\par In order to see the ATASCII character set with these tables, the Atari Classic TrueType fonts must be installed. Characters that are not displayed properly are due to character code usage differences between ATASCII and Windows (see above).
|
|
||||||
\par
|
|
||||||
\par
|
|
||||||
\par \plain\f4\fs16\cf0\b TABLE 1: ATASCII Character Dump Block
|
|
||||||
\par \plain\f4\fs16\cf0
|
|
||||||
\par All characters (ATASCII $00 thru $FF) 16 characters per
|
|
||||||
\par line.
|
|
||||||
\par
|
|
||||||
\par
|
|
||||||
\par \plain\f2\fs12\cf0 \'01\'02\'03\'04\'05\'06\'07\'08\tab
|
|
||||||
\par \'0b\'0c
|
|
||||||
\par \'0e\'0f
|
|
||||||
\par \'10\'11\'12\'13\'14\'15\'16\'17\'18\'19\'1a\'1b\'1c\'1d\'1e\'1f
|
|
||||||
\par !"#$%&'()*+,-./
|
|
||||||
\par 0123456789:;<=>?
|
|
||||||
\par @ABCDEFGHIJKLMNO
|
|
||||||
\par PQRSTUVWXYZ[\\]^_
|
|
||||||
\par `abcdefghijklmno
|
|
||||||
\par pqrstuvwxyz\{|\}~
|
|
||||||
\par \'80\'81\'82\'83\'84\'85\'86\'87\'88\'89\'8a\'8b\'8c\'8d\'8e\'8f
|
|
||||||
\par \'90''""\bullet \endash \emdash \'98\'99\'9a \'9c\'9d\'9e\'9f
|
|
||||||
\par \~\'a1\'a2\'a3\'a4\'a5\'a6\'a7\'a8\'a9\'aa\'ab\'ac\'ad\'ae\'af
|
|
||||||
\par \'b0\'b1\'b2\'b3\'b4\'b5\'b6\'b7\'b8\'b9\'ba\'bb\'bc\'bd\'be\'bf
|
|
||||||
\par \'c0\'c1\'c2\'c3\'c4\'c5\'c6\'c7\'c8\'c9\'ca\'cb\'cc\'cd\'ce\'cf
|
|
||||||
\par \'d0\'d1\'d2\'d3\'d4\'d5\'d6\'d7\'d8\'d9\'da\'db\'dc\'dd\'de\'df
|
|
||||||
\par \'e0\'e1\'e2\'e3\'e4\'e5\'e6\'e7\'e8\'e9\'ea\'eb\'ec\'ed\'ee\'ef
|
|
||||||
\par \'f0\'f1\'f2\'f3\'f4\'f5\'f6\'f7\'f8\'f9\'fa\'fb\'fc\'fd\'fe\'ff
|
|
||||||
\par \plain\f4\fs16\cf0
|
|
||||||
\par
|
|
||||||
\par \plain\f4\fs16\cf0\b TABLE 2: ATASCII Character Dump List
|
|
||||||
\par \plain\f4\fs16\cf0
|
|
||||||
\par All characters (ATASCII $00 thru $FF) one character per
|
|
||||||
\par line with hexadecimal value indicated on the left.
|
|
||||||
\par
|
|
||||||
\par \plain\f2\fs12\cf0 00=
|
|
||||||
\par 01=\'01
|
|
||||||
\par 02=\'02
|
|
||||||
\par 03=\'03
|
|
||||||
\par 04=\'04
|
|
||||||
\par 05=\'05
|
|
||||||
\par 06=\'06
|
|
||||||
\par 07=\'07
|
|
||||||
\par 08=\'08
|
|
||||||
\par 09=\tab
|
|
||||||
\par 0A=
|
|
||||||
\par
|
|
||||||
\par 0B=\'0b
|
|
||||||
\par 0C=\'0c
|
|
||||||
\par 0D=
|
|
||||||
\par 0E=\'0e
|
|
||||||
\par 0F=\'0f
|
|
||||||
\par 10=\'10
|
|
||||||
\par 11=\'11
|
|
||||||
\par 12=\'12
|
|
||||||
\par 13=\'13
|
|
||||||
\par 14=\'14
|
|
||||||
\par 15=\'15
|
|
||||||
\par 16=\'16
|
|
||||||
\par 17=\'17
|
|
||||||
\par 18=\'18
|
|
||||||
\par 19=\'19
|
|
||||||
\par 1A=\'1a
|
|
||||||
\par 1B=\'1b
|
|
||||||
\par 1C=\'1c
|
|
||||||
\par 1D=\'1d
|
|
||||||
\par 1E=\'1e
|
|
||||||
\par 1F=\'1f
|
|
||||||
\par 20=
|
|
||||||
\par 21=!
|
|
||||||
\par 22="
|
|
||||||
\par 23=#
|
|
||||||
\par 24=$
|
|
||||||
\par 25=%
|
|
||||||
\par 26=&
|
|
||||||
\par 27='
|
|
||||||
\par 28=(
|
|
||||||
\par 29=)
|
|
||||||
\par 2A=*
|
|
||||||
\par 2B=+
|
|
||||||
\par 2C=,
|
|
||||||
\par 2D=-
|
|
||||||
\par 2E=.
|
|
||||||
\par 2F=/
|
|
||||||
\par 30=0
|
|
||||||
\par 31=1
|
|
||||||
\par 32=2
|
|
||||||
\par 33=3
|
|
||||||
\par 34=4
|
|
||||||
\par 35=5
|
|
||||||
\par 36=6
|
|
||||||
\par 37=7
|
|
||||||
\par 38=8
|
|
||||||
\par 39=9
|
|
||||||
\par 3A=:
|
|
||||||
\par 3B=;
|
|
||||||
\par 3C=<
|
|
||||||
\par 3D==
|
|
||||||
\par 3E=>
|
|
||||||
\par 3F=?
|
|
||||||
\par 40=@
|
|
||||||
\par 41=A
|
|
||||||
\par 42=B
|
|
||||||
\par 43=C
|
|
||||||
\par 44=D
|
|
||||||
\par 45=E
|
|
||||||
\par 46=F
|
|
||||||
\par 47=G
|
|
||||||
\par 48=H
|
|
||||||
\par 49=I
|
|
||||||
\par 4A=J
|
|
||||||
\par 4B=K
|
|
||||||
\par 4C=L
|
|
||||||
\par 4D=M
|
|
||||||
\par 4E=N
|
|
||||||
\par 4F=O
|
|
||||||
\par 50=P
|
|
||||||
\par 51=Q
|
|
||||||
\par 52=R
|
|
||||||
\par 53=S
|
|
||||||
\par 54=T
|
|
||||||
\par 55=U
|
|
||||||
\par 56=V
|
|
||||||
\par 57=W
|
|
||||||
\par 58=X
|
|
||||||
\par 59=Y
|
|
||||||
\par 5A=Z
|
|
||||||
\par 5B=[
|
|
||||||
\par 5C=\\
|
|
||||||
\par 5D=]
|
|
||||||
\par 5E=^
|
|
||||||
\par 5F=_
|
|
||||||
\par 60=`
|
|
||||||
\par 61=a
|
|
||||||
\par 62=b
|
|
||||||
\par 63=c
|
|
||||||
\par 64=d
|
|
||||||
\par 65=e
|
|
||||||
\par 66=f
|
|
||||||
\par 67=g
|
|
||||||
\par 68=h
|
|
||||||
\par 69=i
|
|
||||||
\par 6A=j
|
|
||||||
\par 6B=k
|
|
||||||
\par 6C=l
|
|
||||||
\par 6D=m
|
|
||||||
\par 6E=n
|
|
||||||
\par 6F=o
|
|
||||||
\par 70=p
|
|
||||||
\par 71=q
|
|
||||||
\par 72=r
|
|
||||||
\par 73=s
|
|
||||||
\par 74=t
|
|
||||||
\par 75=u
|
|
||||||
\par 76=v
|
|
||||||
\par 77=w
|
|
||||||
\par 78=x
|
|
||||||
\par 79=y
|
|
||||||
\par 7A=z
|
|
||||||
\par 7B=\{
|
|
||||||
\par 7C=|
|
|
||||||
\par 7D=\}
|
|
||||||
\par 7E=~
|
|
||||||
\par 7F=
|
|
||||||
\par 80=\'80
|
|
||||||
\par 81=\'81
|
|
||||||
\par 82=\'82
|
|
||||||
\par 83=\'83
|
|
||||||
\par 84=\'84
|
|
||||||
\par 85=\'85
|
|
||||||
\par 86=\'86
|
|
||||||
\par 87=\'87
|
|
||||||
\par 88=\'88
|
|
||||||
\par 89=\'89
|
|
||||||
\par 8A=\'8a
|
|
||||||
\par 8B=\'8b
|
|
||||||
\par 8C=\'8c
|
|
||||||
\par 8D=\'8d
|
|
||||||
\par 8E=\'8e
|
|
||||||
\par 8F=\'8f
|
|
||||||
\par 90=\'90
|
|
||||||
\par 91='
|
|
||||||
\par 92='
|
|
||||||
\par 93="
|
|
||||||
\par 94="
|
|
||||||
\par 95=\bullet
|
|
||||||
\par 96=\endash
|
|
||||||
\par 97=\emdash
|
|
||||||
\par 98=\'98
|
|
||||||
\par 99=\'99
|
|
||||||
\par 9A=\'9a
|
|
||||||
\par 9B=
|
|
||||||
\par 9C=\'9c
|
|
||||||
\par 9D=\'9d
|
|
||||||
\par 9E=\'9e
|
|
||||||
\par 9F=\'9f
|
|
||||||
\par A0=\~
|
|
||||||
\par A1=\'a1
|
|
||||||
\par A2=\'a2
|
|
||||||
\par A3=\'a3
|
|
||||||
\par A4=\'a4
|
|
||||||
\par A5=\'a5
|
|
||||||
\par A6=\'a6
|
|
||||||
\par A7=\'a7
|
|
||||||
\par A8=\'a8
|
|
||||||
\par A9=\'a9
|
|
||||||
\par AA=\'aa
|
|
||||||
\par AB=\'ab
|
|
||||||
\par AC=\'ac
|
|
||||||
\par AD=\'ad
|
|
||||||
\par AE=\'ae
|
|
||||||
\par AF=\'af
|
|
||||||
\par B0=\'b0
|
|
||||||
\par B1=\'b1
|
|
||||||
\par B2=\'b2
|
|
||||||
\par B3=\'b3
|
|
||||||
\par B4=\'b4
|
|
||||||
\par B5=\'b5
|
|
||||||
\par B6=\'b6
|
|
||||||
\par B7=\'b7
|
|
||||||
\par B8=\'b8
|
|
||||||
\par B9=\'b9
|
|
||||||
\par BA=\'ba
|
|
||||||
\par BB=\'bb
|
|
||||||
\par BC=\'bc
|
|
||||||
\par BD=\'bd
|
|
||||||
\par BE=\'be
|
|
||||||
\par BF=\'bf
|
|
||||||
\par C0=\'c0
|
|
||||||
\par C1=\'c1
|
|
||||||
\par C2=\'c2
|
|
||||||
\par C3=\'c3
|
|
||||||
\par C4=\'c4
|
|
||||||
\par C5=\'c5
|
|
||||||
\par C6=\'c6
|
|
||||||
\par C7=\'c7
|
|
||||||
\par C8=\'c8
|
|
||||||
\par C9=\'c9
|
|
||||||
\par CA=\'ca
|
|
||||||
\par CB=\'cb
|
|
||||||
\par CC=\'cc
|
|
||||||
\par CD=\'cd
|
|
||||||
\par CE=\'ce
|
|
||||||
\par CF=\'cf
|
|
||||||
\par D0=\'d0
|
|
||||||
\par D1=\'d1
|
|
||||||
\par D2=\'d2
|
|
||||||
\par D3=\'d3
|
|
||||||
\par D4=\'d4
|
|
||||||
\par D5=\'d5
|
|
||||||
\par D6=\'d6
|
|
||||||
\par D7=\'d7
|
|
||||||
\par D8=\'d8
|
|
||||||
\par D9=\'d9
|
|
||||||
\par DA=\'da
|
|
||||||
\par DB=\'db
|
|
||||||
\par DC=\'dc
|
|
||||||
\par DD=\'dd
|
|
||||||
\par DE=\'de
|
|
||||||
\par DF=\'df
|
|
||||||
\par E0=\'e0
|
|
||||||
\par E1=\'e1
|
|
||||||
\par E2=\'e2
|
|
||||||
\par E3=\'e3
|
|
||||||
\par E4=\'e4
|
|
||||||
\par E5=\'e5
|
|
||||||
\par E6=\'e6
|
|
||||||
\par E7=\'e7
|
|
||||||
\par E8=\'e8
|
|
||||||
\par E9=\'e9
|
|
||||||
\par EA=\'ea
|
|
||||||
\par EB=\'eb
|
|
||||||
\par EC=\'ec
|
|
||||||
\par ED=\'ed
|
|
||||||
\par EE=\'ee
|
|
||||||
\par EF=\'ef
|
|
||||||
\par F0=\'f0
|
|
||||||
\par F1=\'f1
|
|
||||||
\par F2=\'f2
|
|
||||||
\par F3=\'f3
|
|
||||||
\par F4=\'f4
|
|
||||||
\par F5=\'f5
|
|
||||||
\par F6=\'f6
|
|
||||||
\par F7=\'f7
|
|
||||||
\par F8=\'f8
|
|
||||||
\par F9=\'f9
|
|
||||||
\par FA=\'fa
|
|
||||||
\par FB=\'fb
|
|
||||||
\par FC=\'fc
|
|
||||||
\par FD=\'fd
|
|
||||||
\par FE=\'fe
|
|
||||||
\par FF=\'ff
|
|
||||||
\par }
|
|
||||||
|