Compare commits
146 Commits
0.1-alpha
...
0.6.3-beta
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ebb389b6d7 | ||
|
|
55814f1d6d | ||
|
|
7dc192e59b | ||
|
|
37f7be7db1 | ||
|
|
680130f26e | ||
|
|
cbe0ad14e5 | ||
|
|
e3a8f43b52 | ||
|
|
baa20d69b7 | ||
|
|
bd49ba9e47 | ||
|
|
16f6d353ed | ||
|
|
42d19b6e4b | ||
|
|
9759321566 | ||
|
|
426052f573 | ||
|
|
3048e4dbc6 | ||
|
|
a4680326f0 | ||
|
|
a28e56797b | ||
|
|
e56fc40f8f | ||
|
|
24cafa00d4 | ||
|
|
8e14d144fd | ||
|
|
e001aa7896 | ||
|
|
22703e4bfb | ||
|
|
6d79329442 | ||
|
|
fe69e0cd0f | ||
|
|
fe53cabee1 | ||
|
|
86e2856d50 | ||
|
|
116b02d68f | ||
|
|
c9a0d0bf84 | ||
|
|
f36143e10f | ||
|
|
6185ca7110 | ||
|
|
3729b293f0 | ||
|
|
b71c178840 | ||
|
|
ba2d1241d0 | ||
|
|
07e0d757bb | ||
|
|
88ea066481 | ||
|
|
7b59c78cee | ||
|
|
c8f61d4d2c | ||
|
|
bd894a523e | ||
|
|
27ae092b5e | ||
|
|
43dcccb00c | ||
|
|
0d0db2f8b6 | ||
|
|
61c52fe7cc | ||
|
|
4b6ea4398d | ||
|
|
ed821bde45 | ||
|
|
955a1262e9 | ||
|
|
021fbfa60f | ||
|
|
801efa570c | ||
|
|
0b50b064db | ||
|
|
c5e3fd5668 | ||
|
|
3125762133 | ||
|
|
c0670c8281 | ||
|
|
7493c8f527 | ||
|
|
9bd9f180c2 | ||
|
|
c64be6a4c9 | ||
|
|
eb752835fd | ||
|
|
72a2ef9660 | ||
|
|
213eace512 | ||
|
|
06e0b401ca | ||
|
|
bc8780608c | ||
|
|
536c231941 | ||
|
|
4faed597c0 | ||
|
|
75816e7750 | ||
|
|
8b655afb2d | ||
|
|
abf649d2ab | ||
|
|
a8f2152c92 | ||
|
|
9d05126e8f | ||
|
|
4d4bcc6151 | ||
|
|
5b6fd56a37 | ||
|
|
bcc0d0d8e7 | ||
|
|
ec4bdd6f7f | ||
|
|
a8798260dc | ||
|
|
50582a95a4 | ||
|
|
1302205911 | ||
|
|
73c1b482e2 | ||
|
|
d5487140cf | ||
|
|
45bfa80340 | ||
|
|
661a85066b | ||
|
|
e16bb15310 | ||
|
|
74ebc91ee6 | ||
|
|
cda458b5bf | ||
|
|
9d27ae24e7 | ||
|
|
1eeeafb51e | ||
|
|
4aa77395c0 | ||
|
|
ae1aef63f9 | ||
|
|
406ab6c6ed | ||
|
|
2da8da8c8e | ||
|
|
c2057e979d | ||
|
|
f5402d216a | ||
|
|
43e3155a08 | ||
|
|
fa29f25c91 | ||
|
|
dab124cf80 | ||
|
|
2f23c83d49 | ||
|
|
3d68e384a5 | ||
|
|
dc2f1cffe6 | ||
|
|
f61dfbf654 | ||
|
|
e6175dc7f4 | ||
|
|
0ae8a93c21 | ||
|
|
b921d14dbf | ||
|
|
64943f01c5 | ||
|
|
279eaaad3e | ||
|
|
d8afb84861 | ||
|
|
43691750ef | ||
|
|
1104bc3c18 | ||
|
|
3e9f728cda | ||
|
|
2c16e541d5 | ||
|
|
52f5bc5ebd | ||
|
|
e466ef2afe | ||
|
|
1e943c3f26 | ||
|
|
798630bd50 | ||
|
|
c356d0f5ae | ||
|
|
c2ae42701c | ||
|
|
9edaa2fda7 | ||
|
|
2ec4039393 | ||
|
|
69b6ad1976 | ||
|
|
d5a9469df1 | ||
|
|
e7f6a54d8d | ||
|
|
e1a1bcf070 | ||
|
|
cd531bd0a5 | ||
|
|
a54385b8f5 | ||
|
|
38746c7f38 | ||
|
|
f256e4ed28 | ||
|
|
06970fd448 | ||
|
|
a3d4783796 | ||
|
|
c8e57150eb | ||
|
|
23c2623bcf | ||
|
|
72dbe94dc2 | ||
|
|
8d415f1a0f | ||
|
|
2c04c898ab | ||
|
|
a66474b7cf | ||
|
|
3906bfde80 | ||
|
|
19f74a1761 | ||
|
|
8903703064 | ||
|
|
56e7b0500c | ||
|
|
9fa86f8c4c | ||
|
|
551172aed3 | ||
|
|
7a0dbcca2b | ||
|
|
1a7e5aff5f | ||
|
|
c24a6b1fa1 | ||
|
|
4387e9ffbd | ||
|
|
f258c6e095 | ||
|
|
5739474957 | ||
|
|
8bc0c2c886 | ||
|
|
4584ba668d | ||
|
|
d3607f5b00 | ||
|
|
2ad5d45153 | ||
|
|
690f340922 | ||
|
|
dc93fe856d |
33
README.md
33
README.md
@@ -1,34 +1,39 @@
|
|||||||
# QTeletextMaker
|
# QTeletextMaker
|
||||||
QTeletextMaker is a teletext page editor in development. It is written in C++ using the Qt 5 widget libraries, and released under the GNU General Public License v3.0
|
QTeletextMaker is a teletext page editor with an emphasis on Level 2.5 enhancement editing, released under the GNU General Public License v3.0
|
||||||
|
|
||||||
|
It is written in C++ using the Qt 5 widget libraries but should also compile with Qt 6.
|
||||||
|
|
||||||
Features
|
Features
|
||||||
- Load and save teletext pages in .tti format.
|
- Load and save teletext pages in .tti format.
|
||||||
- Rendering of teletext pages in Levels 1, 1.5, 2.5 and 3.5
|
- Rendering of teletext pages in Levels 1, 1.5, 2.5 and 3.5
|
||||||
- Rendering of Local Objects and side panels.
|
- Rendering of Local Objects and side panels.
|
||||||
|
- Import and export of single pages in .t42 format.
|
||||||
|
- Export PNG images of teletext pages.
|
||||||
|
- Undo and redo of editing actions.
|
||||||
- Interactive X/26 Local Enhancement Data triplet editor.
|
- Interactive X/26 Local Enhancement Data triplet editor.
|
||||||
- Editing of X/27/4 and X/27/5 compositional links to enhancement data pages.
|
- Editing of X/27/4 and X/27/5 compositional links to enhancement data pages.
|
||||||
- Palette editor.
|
- Palette editor.
|
||||||
- Configurable zoom.
|
- Configurable zoom.
|
||||||
- View teletext pages in 4:3, 16:9 pillar-box and 16:9 stretch aspect ratios.
|
- View teletext pages in 4:3, 16:9 pillar-box and 16:9 stretch aspect ratios.
|
||||||
|
|
||||||
Although designed on and developed for Linux, the Qt 5 libraries are cross platform so a Windows executable can be built. A Windows executable can be found within the "Releases" link, compiled on a Linux host using [MXE](https://github.com/mxe/mxe) based on [these instructions](https://blog.8bitbuddhism.com/2018/08/22/cross-compiling-windows-applications-with-mxe/). After MXE is installed `make qtbase` will be enough to build QTeletextMaker.
|
Although designed on and developed for Linux, the Qt libraries are cross platform so a Windows executable can be built. A Windows executable can be found within the "Releases" link, compiled on a Linux host using [MXE](https://github.com/mxe/mxe) based on [these instructions](https://blog.8bitbuddhism.com/2018/08/22/cross-compiling-windows-applications-with-mxe/). After MXE is installed `make qtbase` should build and install the required dependencies to build QTeletextMaker.
|
||||||
|
|
||||||
## Building
|
## Building
|
||||||
### Linux
|
### Linux
|
||||||
Install the QtCore, QtGui and QtWidgets libraries and build headers, along with the qmake tool. Then type `qmake && make -j3` in a terminal, you can replace -j3 with the number of processor cores used for the compile process.
|
Install version 5 or 6 of the QtCore, QtGui and QtWidgets libraries and build headers, along with the qmake tool. Depending on how qmake is installed, type `qmake && make -j3` or `qmake5 && make -j3` or `qmake6 && make -j3` in a terminal to build, you can replace -j3 with the number of processor cores used for the compile process.
|
||||||
|
|
||||||
The above will place the qteletextmaker executable in the same directory as the source, type `./qteletextmaker` in the terminal to launch. Optionally, type `make install` afterwards to place the executable into /usr/local/bin.
|
The above should place the qteletextmaker executable in the same directory as the source, type `./qteletextmaker` in the terminal to launch. Some Qt installs may place the executable into a "release" directory.
|
||||||
|
|
||||||
|
Optionally, type `make install` afterwards to place the executable into /usr/local/bin.
|
||||||
|
|
||||||
## Current limitations
|
## Current limitations
|
||||||
The following X/26 enhancement triplets are not rendered by the editor, although the list is fully aware of them.
|
The following X/26 enhancement triplets are not rendered by the editor, although the list is fully aware of them.
|
||||||
- Invocation of Objects from POP and GPOP pages.
|
- Invocation of Objects from POP and GPOP pages.
|
||||||
- DRCS characters.
|
- DRCS characters.
|
||||||
- Modified G0 and G2 character set designation using X/26 triplets with mode 01000.
|
- Proportional font spacing on Level 3.5
|
||||||
- Full screen and full row colours set by Active Objects.
|
|
||||||
- Level 3.5 font style: bold, italic and proportional spacing.
|
|
||||||
|
|
||||||
## Using the X/26 triplet editor
|
## Using the X/26 triplet editor
|
||||||
The X/26 triplet editor sorts all the triplet modes available into categories selected by the triplet *type* dropdown on the left. The categories are:
|
The X/26 triplet editor sorts all the triplet modes available into categories, which are:
|
||||||
- Set Active Position
|
- Set Active Position
|
||||||
- Row triplet - full screen and full row colours, address row 0 and DRCS mode
|
- Row triplet - full screen and full row colours, address row 0 and DRCS mode
|
||||||
- Column triplet - non-spacing attributes and overwriting characters
|
- Column triplet - non-spacing attributes and overwriting characters
|
||||||
@@ -36,24 +41,20 @@ The X/26 triplet editor sorts all the triplet modes available into categories se
|
|||||||
- Terminator
|
- Terminator
|
||||||
- PDC/reserved
|
- PDC/reserved
|
||||||
|
|
||||||
Selecting "Set Active Position" or "Terminator" will change the triplet mode immediately, other selections will activate the triplet *mode* dropdown on the right which can be used to then change the triplet mode. Most triplet modes will present varying widgets below which can be used to alter the parameters of the currently selected triplet (e.g. colour or character).
|
After selecting the triplet mode the Row and Column spinboxes can then be used to place the Active Position of the selected triplet, whether or not each spinbox can be altered depends on the mode of the selected triplet. As well as the explicit "Set Active Position" triplet that can set both the row and the column, all column triplets can simultaneously set the column of the Active Position. Additionally the Full Row Colour triplet can set the row of the Active Position, with the column always set to 0.
|
||||||
|
|
||||||
Between the two dropdowns are the Row and Column spinboxes that are used to place the Active Position of the selected triplet, whether or not each spinbox can be altered depends on the mode of the selected triplet. As well as the explicit "Set Active Position" triplet that can set both the row and the column, all column triplets can simultaneously set the column of the Active Position. Additionally the Full Row Colour triplet can set the row of the Active Position, with the column always set to 0.
|
Most triplet modes will present varying widgets below which can be used to alter the parameters of the currently selected triplet e.g. colour or character.
|
||||||
|
|
||||||
By checking "raw values" it is also possible to view and edit the raw Address, Mode and Data numbers of the triplets. When editing triplets this way, remember that address values 0-39 select a column triplet which has one set of modes and address values 40-63 select a row triplet which has a different set of modes.
|
By checking "raw values" it is also possible to view and edit the raw Address, Mode and Data numbers of the triplets. When editing triplets this way, remember that address values 0-39 select a column triplet which has one set of modes and address values 40-63 select a row triplet which has a different set of modes.
|
||||||
|
|
||||||
The full behaviour of X/26 enhancement triplets can be found in section 12.3 of the [Enhanced Teletext specification ETS 300 706](https://web.archive.org/web/20160326062859/https://www.phecap.nl/download/enhenced-teletext-specs.pdf).
|
The full behaviour of X/26 enhancement triplets can be found in section 12.3 of the [Enhanced Teletext specification ETS 300 706](https://www.etsi.org/deliver/etsi_en/300700_300799/300706/01.02.01_60/en_300706v010201p.pdf).
|
||||||
|
|
||||||
### Setting the Active Position
|
### Setting the Active Position
|
||||||
The Active Position, whether set explicitly by a "Set Active Position" triplet or by a row or column triplet, can only be moved in screen address order i.e. from left to right within a row and then from top to bottom across rows. In other words:
|
The Active Position, whether set explicitly by a "Set Active Position" triplet or by a row or column triplet, can only be moved in screen address order i.e. from left to right within a row and then from top to bottom across rows. In other words:
|
||||||
- The Active Position can never be moved up to a lesser numbered row.
|
- The Active Position can never be moved up to a lesser numbered row.
|
||||||
- The Active Position can never be moved left *within the same row* to a lesser numbered column, but it can be moved left at the same time as it is moved down to a greater numbered row.
|
- The Active Position can never be moved left *within the same row* to a lesser numbered column, but it can be moved left at the same time as it is moved down to a greater numbered row.
|
||||||
|
|
||||||
If this rule is not followed then triplets in earlier screen addresses will be ignored.
|
If this rule is not followed then triplets in earlier screen addresses will be ignored. Triplets that break this rule will be highlighted red in the X/26 triplet editor.
|
||||||
|
|
||||||
### Objects
|
### Objects
|
||||||
"Define ... Object" triplets need to declare that they are in the correct place in the triplet list e.g. if the Define Object triplet is at `d1 t3` in the list then the data field must show `Local: d1 t3`, otherwise the Object won't appear.
|
|
||||||
|
|
||||||
Insert and deleting triplets from the list will upset the Object pointers on both "Define" and "Invoke" triplets and will need to be corrected afterwards. A future version of the editor may adjust these pointers automatically.
|
|
||||||
|
|
||||||
"Invoke ... Object" triplets must point to a "Define ... Object" of the same type e.g. "Invoke *Active* Object" must point to a "Define *Active* Object", otherwise the Object won't appear.
|
"Invoke ... Object" triplets must point to a "Define ... Object" of the same type e.g. "Invoke *Active* Object" must point to a "Define *Active* Object", otherwise the Object won't appear.
|
||||||
|
|||||||
1165
decode.cpp
Normal file
1165
decode.cpp
Normal file
File diff suppressed because it is too large
Load Diff
279
decode.h
Normal file
279
decode.h
Normal file
@@ -0,0 +1,279 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
|
*
|
||||||
|
* This file is part of QTeletextMaker.
|
||||||
|
*
|
||||||
|
* QTeletextMaker 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.
|
||||||
|
*
|
||||||
|
* QTeletextMaker 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 QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef DECODE_H
|
||||||
|
#define DECODE_H
|
||||||
|
|
||||||
|
#include <QList>
|
||||||
|
#include <QMap>
|
||||||
|
#include <QMultiMap>
|
||||||
|
|
||||||
|
#include "levelonepage.h"
|
||||||
|
|
||||||
|
class TeletextPageDecode : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
enum CharacterFragment { NormalSize, DoubleHeightTopHalf, DoubleHeightBottomHalf, DoubleWidthLeftHalf, DoubleWidthRightHalf, DoubleSizeTopLeftQuarter, DoubleSizeTopRightQuarter, DoubleSizeBottomLeftQuarter, DoubleSizeBottomRightQuarter };
|
||||||
|
enum RowHeight { NormalHeight, TopHalf, BottomHalf };
|
||||||
|
|
||||||
|
TeletextPageDecode();
|
||||||
|
~TeletextPageDecode();
|
||||||
|
bool refresh(int r, int c) const { return m_refresh[r][c]; }
|
||||||
|
void setRefresh(int r, int c, bool refresh);
|
||||||
|
void decodePage();
|
||||||
|
LevelOnePage *teletextPage() const { return m_levelOnePage; };
|
||||||
|
void setTeletextPage(LevelOnePage *newCurrentPage);
|
||||||
|
void updateSidePanels();
|
||||||
|
|
||||||
|
unsigned char cellCharacterCode(int r, int c) const { return m_cell[r][c].character.code; };
|
||||||
|
int cellCharacterSet(int r, int c) const { return m_cell[r][c].character.set; };
|
||||||
|
int cellCharacterDiacritical(int r, int c) const { return m_cell[r][c].character.diacritical; };
|
||||||
|
int cellG0CharacterSet(int r, int c) const { return m_cell[r][c].g0Set; };
|
||||||
|
int cellG2CharacterSet(int r, int c) const { return m_cell[r][c].g2Set; };
|
||||||
|
int cellForegroundCLUT(int r, int c) const { return m_cell[r][c].attribute.foregroundCLUT; };
|
||||||
|
int cellBackgroundCLUT(int r, int c) const { return m_cell[r][c].attribute.backgroundCLUT; };
|
||||||
|
QColor cellForegroundQColor(int r, int c);
|
||||||
|
QColor cellBackgroundQColor(int r, int c);
|
||||||
|
QColor cellFlashForegroundQColor(int r, int c);
|
||||||
|
int cellFlashMode(int r, int c) const { return m_cell[r][c].attribute.flash.mode; };
|
||||||
|
int cellFlashRatePhase(int r, int c) const { return m_cell[r][c].attribute.flash.ratePhase; };
|
||||||
|
int cellFlash2HzPhaseNumber(int r, int c) const { return m_cell[r][c].attribute.flash.phase2HzShown; };
|
||||||
|
CharacterFragment cellCharacterFragment(int r, int c) const { return m_cell[r][c].fragment; };
|
||||||
|
bool cellBoxed(int r, int c) const { return m_cell[r][c].attribute.display.boxingWindow; };
|
||||||
|
bool cellConceal(int r, int c) const { return m_cell[r][c].attribute.display.conceal; };
|
||||||
|
bool cellUnderlined(int r, int c) const { return cellCharacterSet(r, c) < 24 ? m_cell[r][c].attribute.display.underlineSeparated : false; };
|
||||||
|
bool cellBold(int r, int c) const { return m_cell[r][c].attribute.style.bold; };
|
||||||
|
bool cellItalic(int r, int c) const { return m_cell[r][c].attribute.style.italic; };
|
||||||
|
bool cellProportional(int r, int c) const { return m_cell[r][c].attribute.style.proportional; };
|
||||||
|
|
||||||
|
bool level1MosaicAttribute(int r, int c) const { return m_cellLevel1Mosaic[r][c]; };
|
||||||
|
int level1CharSet(int r, int c) const { return m_cellLevel1CharSet[r][c]; };
|
||||||
|
|
||||||
|
RowHeight rowHeight(int r) const { return m_rowHeight[r]; };
|
||||||
|
|
||||||
|
QColor fullScreenQColor() const { return m_finalFullScreenQColor; };
|
||||||
|
QColor fullRowQColor(int r) const { return m_fullRowQColor[r]; };
|
||||||
|
int leftSidePanelColumns() const { return m_leftSidePanelColumns; };
|
||||||
|
int rightSidePanelColumns() const { return m_rightSidePanelColumns; };
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void setLevel(int level);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void fullScreenColourChanged(QColor newColour);
|
||||||
|
void fullRowColourChanged(int r, QColor newColour);
|
||||||
|
void sidePanelsChanged();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
inline void setFullScreenColour(int newColour);
|
||||||
|
inline void setFullRowColour(int row, int newColour);
|
||||||
|
|
||||||
|
int m_finalFullScreenColour, m_level;
|
||||||
|
QColor m_finalFullScreenQColor;
|
||||||
|
int m_leftSidePanelColumns, m_rightSidePanelColumns;
|
||||||
|
const int m_foregroundRemap[8] = { 0, 0, 0, 8, 8, 16, 16, 16 };
|
||||||
|
const int m_backgroundRemap[8] = { 0, 8, 16, 8, 16, 8, 16, 24 };
|
||||||
|
|
||||||
|
private:
|
||||||
|
class Invocation;
|
||||||
|
|
||||||
|
enum ColourPart { Foreground, Background, FlashForeground };
|
||||||
|
|
||||||
|
struct textCharacter {
|
||||||
|
unsigned char code=0x20;
|
||||||
|
int set=0;
|
||||||
|
int diacritical=0;
|
||||||
|
};
|
||||||
|
|
||||||
|
friend inline bool operator!=(const textCharacter &lhs, const textCharacter &rhs)
|
||||||
|
{
|
||||||
|
return lhs.code != rhs.code ||
|
||||||
|
lhs.set != rhs.set ||
|
||||||
|
lhs.diacritical != rhs.diacritical;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct flashFunctions {
|
||||||
|
int mode=0;
|
||||||
|
int ratePhase=0;
|
||||||
|
int phase2HzShown=0;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct displayAttributes {
|
||||||
|
bool doubleHeight=false;
|
||||||
|
bool doubleWidth=false;
|
||||||
|
bool boxingWindow=false;
|
||||||
|
bool conceal=false;
|
||||||
|
bool invert=false;
|
||||||
|
bool underlineSeparated=false;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct fontStyle {
|
||||||
|
bool proportional=false;
|
||||||
|
bool bold=false;
|
||||||
|
bool italic=false;
|
||||||
|
};
|
||||||
|
|
||||||
|
friend inline bool operator!=(const displayAttributes &lhs, const displayAttributes &rhs)
|
||||||
|
{
|
||||||
|
return lhs.doubleHeight != rhs.doubleHeight ||
|
||||||
|
lhs.doubleWidth != rhs.doubleWidth ||
|
||||||
|
lhs.boxingWindow != rhs.boxingWindow ||
|
||||||
|
lhs.conceal != rhs.conceal ||
|
||||||
|
lhs.invert != rhs.invert ||
|
||||||
|
lhs.underlineSeparated != rhs.underlineSeparated;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct textAttributes {
|
||||||
|
int foregroundCLUT=7;
|
||||||
|
int backgroundCLUT=0;
|
||||||
|
flashFunctions flash;
|
||||||
|
displayAttributes display;
|
||||||
|
fontStyle style;
|
||||||
|
};
|
||||||
|
|
||||||
|
friend inline bool operator!=(const textAttributes &lhs, const textAttributes &rhs)
|
||||||
|
{
|
||||||
|
return lhs.foregroundCLUT != rhs.foregroundCLUT ||
|
||||||
|
lhs.backgroundCLUT != rhs.backgroundCLUT ||
|
||||||
|
lhs.flash.mode != rhs.flash.mode ||
|
||||||
|
lhs.flash.ratePhase != rhs.flash.ratePhase ||
|
||||||
|
lhs.flash.phase2HzShown != rhs.flash.phase2HzShown ||
|
||||||
|
lhs.display != rhs.display ||
|
||||||
|
lhs.style.proportional != rhs.style.proportional ||
|
||||||
|
lhs.style.bold != rhs.style.bold ||
|
||||||
|
lhs.style.italic != rhs.style.italic;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct textCell {
|
||||||
|
textCharacter character;
|
||||||
|
textAttributes attribute;
|
||||||
|
CharacterFragment fragment=NormalSize;
|
||||||
|
int g0Set=0;
|
||||||
|
int g2Set=7;
|
||||||
|
};
|
||||||
|
|
||||||
|
friend inline bool operator!=(const textCell &lhs, const textCell &rhs)
|
||||||
|
{
|
||||||
|
return lhs.character != rhs.character ||
|
||||||
|
lhs.attribute != rhs.attribute ||
|
||||||
|
lhs.fragment != rhs.fragment;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct textPainter {
|
||||||
|
textAttributes attribute;
|
||||||
|
textCell result;
|
||||||
|
textCell rightHalfCell;
|
||||||
|
textCell bottomHalfCell[72];
|
||||||
|
|
||||||
|
int styleSpreadRows=0;
|
||||||
|
int setProportionalRows[72], clearProportionalRows[72];
|
||||||
|
int setBoldRows[72], clearBoldRows[72];
|
||||||
|
int setItalicRows[72], clearItalicRows[72];
|
||||||
|
};
|
||||||
|
|
||||||
|
const QMap<int, int> m_level1CharacterMap {
|
||||||
|
{ 0x00, 12 }, { 0x01, 15 }, { 0x02, 22 }, { 0x03, 16 }, { 0x04, 14 }, { 0x05, 19 }, { 0x06, 11 },
|
||||||
|
{ 0x08, 18 }, { 0x09, 15 }, { 0x0a, 22 }, { 0x0b, 16 }, { 0x0c, 14 }, { 0x0e, 11 },
|
||||||
|
{ 0x10, 12 }, { 0x11, 15 }, { 0x12, 22 }, { 0x13, 16 }, { 0x14, 14 }, { 0x15, 19 }, { 0x16, 23 },
|
||||||
|
{ 0x1d, 21 }, { 0x1f, 20 },
|
||||||
|
{ 0x20, 1 }, { 0x21, 15 }, { 0x22, 13 }, { 0x23, 17 }, { 0x24, 2 }, { 0x25, 3 }, { 0x26, 11 },
|
||||||
|
{ 0x36, 23 }, { 0x37, 4 },
|
||||||
|
{ 0x40, 12 }, { 0x44, 14 }, { 0x47, 5 },
|
||||||
|
{ 0x55, 6 }, { 0x57, 5 }
|
||||||
|
};
|
||||||
|
const QMap<int, int> m_g0CharacterMap {
|
||||||
|
{ 0x20, 1 }, { 0x24, 2 }, { 0x25, 3 },
|
||||||
|
{ 0x37, 4 },
|
||||||
|
{ 0x47, 5 },
|
||||||
|
{ 0x55, 6 }, { 0x57, 5 }
|
||||||
|
};
|
||||||
|
const QMap<int, int> m_g2CharacterMap {
|
||||||
|
{ 0x20, 8 }, { 0x24, 8 }, { 0x25, 8 },
|
||||||
|
{ 0x37, 9 },
|
||||||
|
{ 0x40, 10 }, { 0x44, 10 }, { 0x47, 10 },
|
||||||
|
{ 0x55, 10 }, { 0x57, 10 }
|
||||||
|
};
|
||||||
|
|
||||||
|
class Invocation
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Invocation();
|
||||||
|
|
||||||
|
X26TripletList *tripletList() const { return m_tripletList; };
|
||||||
|
void clear();
|
||||||
|
void setTripletList(X26TripletList *tripletList);
|
||||||
|
int startTripletNumber() const { return m_startTripletNumber; };
|
||||||
|
void setStartTripletNumber(int n);
|
||||||
|
int endTripletNumber() const { return m_endTripletNumber; };
|
||||||
|
void setEndTripletNumber(int n);
|
||||||
|
int originRow() const { return m_originRow; };
|
||||||
|
int originColumn() const { return m_originColumn; };
|
||||||
|
void setOrigin(int row, int column);
|
||||||
|
void buildMap(int level);
|
||||||
|
|
||||||
|
QList<QPair<int, int>> charPositions() const { return m_characterMap.uniqueKeys(); };
|
||||||
|
QList<QPair<int, int>> attrPositions() const { return m_attributeMap.uniqueKeys(); };
|
||||||
|
QList<X26Triplet> charactersMappedAt(int r, int c) const { return m_characterMap.values(qMakePair(r, c)); };
|
||||||
|
QList<X26Triplet> attributesMappedAt(int r, int c) const { return m_attributeMap.values(qMakePair(r, c)); };
|
||||||
|
int rightMostColumn(int r) const { return m_rightMostColumn.value(r, -1); };
|
||||||
|
int fullScreenColour() const { return m_fullScreenCLUT; };
|
||||||
|
QList<X26Triplet> fullRowColoursMappedAt(int r) const { return m_fullRowCLUTMap.values(r); };
|
||||||
|
|
||||||
|
private:
|
||||||
|
X26TripletList *m_tripletList;
|
||||||
|
int m_startTripletNumber, m_endTripletNumber;
|
||||||
|
int m_originRow, m_originColumn;
|
||||||
|
// QPair is row and column
|
||||||
|
QMultiMap<QPair<int, int>, X26Triplet> m_characterMap;
|
||||||
|
QMultiMap<QPair<int, int>, X26Triplet> m_attributeMap;
|
||||||
|
QMap<int, int> m_rightMostColumn;
|
||||||
|
int m_fullScreenCLUT;
|
||||||
|
QMultiMap<int, X26Triplet> m_fullRowCLUTMap;
|
||||||
|
};
|
||||||
|
|
||||||
|
static int s_instances;
|
||||||
|
static textPainter s_blankPainter;
|
||||||
|
|
||||||
|
void decodeRow(int r);
|
||||||
|
QColor cellQColor(int r, int c, ColourPart colourPart);
|
||||||
|
textCell& cellAtCharacterOrigin(int r, int c);
|
||||||
|
void buildInvocationList(Invocation &invocation, int objectType);
|
||||||
|
textCharacter characterFromTriplets(const QList<X26Triplet> triplets);
|
||||||
|
inline void rotateFlashMovement(flashFunctions &flash);
|
||||||
|
|
||||||
|
bool m_refresh[25][72];
|
||||||
|
textCell m_cell[25][72];
|
||||||
|
bool m_cellLevel1Mosaic[25][40];
|
||||||
|
int m_cellLevel1CharSet[25][40];
|
||||||
|
LevelOnePage* m_levelOnePage;
|
||||||
|
int m_fullRowColour[25];
|
||||||
|
QColor m_fullRowQColor[25];
|
||||||
|
QList<Invocation> m_invocations[3];
|
||||||
|
Invocation m_localEnhancements;
|
||||||
|
textPainter m_level1ActivePainter;
|
||||||
|
QList<textPainter> m_adapPassPainter[2];
|
||||||
|
int m_level1DefaultCharSet, m_level1SecondCharSet;
|
||||||
|
int m_defaultG0andG2, m_secondG0andG2;
|
||||||
|
|
||||||
|
RowHeight m_rowHeight[25];
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
183
document.cpp
183
document.cpp
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -17,12 +17,46 @@
|
|||||||
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "document.h"
|
#include "document.h"
|
||||||
|
|
||||||
#include "levelonepage.h"
|
#include "levelonepage.h"
|
||||||
|
|
||||||
|
ClutModel::ClutModel(QObject *parent): QAbstractListModel(parent)
|
||||||
|
{
|
||||||
|
m_subPage = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ClutModel::rowCount(const QModelIndex & /*parent*/) const
|
||||||
|
{
|
||||||
|
return 32;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant ClutModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (!index.isValid())
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
if (role == Qt::DisplayRole)
|
||||||
|
return QString("CLUT %1:%2").arg(index.row() >> 3).arg(index.row() & 0x07);
|
||||||
|
|
||||||
|
if (role == Qt::DecorationRole && m_subPage != nullptr)
|
||||||
|
return m_subPage->CLUTtoQColor(index.row());
|
||||||
|
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ClutModel::setSubPage(LevelOnePage *subPage)
|
||||||
|
{
|
||||||
|
if (subPage != m_subPage) {
|
||||||
|
m_subPage = subPage;
|
||||||
|
emit dataChanged(createIndex(0, 0), createIndex(31, 0), QVector<int>(Qt::DecorationRole));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
TeletextDocument::TeletextDocument()
|
TeletextDocument::TeletextDocument()
|
||||||
{
|
{
|
||||||
m_pageNumber = 0x199;
|
m_pageNumber = 0x199;
|
||||||
@@ -34,11 +68,17 @@ TeletextDocument::TeletextDocument()
|
|||||||
m_undoStack = new QUndoStack(this);
|
m_undoStack = new QUndoStack(this);
|
||||||
m_cursorRow = 1;
|
m_cursorRow = 1;
|
||||||
m_cursorColumn = 0;
|
m_cursorColumn = 0;
|
||||||
|
m_selectionCornerRow = m_selectionCornerColumn = -1;
|
||||||
m_selectionSubPage = nullptr;
|
m_selectionSubPage = nullptr;
|
||||||
|
|
||||||
|
m_clutModel = new ClutModel;
|
||||||
|
m_clutModel->setSubPage(m_subPages[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
TeletextDocument::~TeletextDocument()
|
TeletextDocument::~TeletextDocument()
|
||||||
{
|
{
|
||||||
|
delete m_clutModel;
|
||||||
|
|
||||||
for (auto &subPage : m_subPages)
|
for (auto &subPage : m_subPages)
|
||||||
delete(subPage);
|
delete(subPage);
|
||||||
for (auto &recycleSubPage : m_recycleSubPages)
|
for (auto &recycleSubPage : m_recycleSubPages)
|
||||||
@@ -54,6 +94,25 @@ bool TeletextDocument::isEmpty() const
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TeletextDocument::clear()
|
||||||
|
{
|
||||||
|
LevelOnePage *blankSubPage = new LevelOnePage;
|
||||||
|
|
||||||
|
m_subPages.insert(m_subPages.begin(), blankSubPage);
|
||||||
|
|
||||||
|
emit aboutToChangeSubPage();
|
||||||
|
m_currentSubPageIndex = 0;
|
||||||
|
m_clutModel->setSubPage(m_subPages[0]);
|
||||||
|
emit subPageSelected();
|
||||||
|
cancelSelection();
|
||||||
|
m_undoStack->clear();
|
||||||
|
|
||||||
|
for (int i=m_subPages.size()-1; i>0; i--) {
|
||||||
|
delete(m_subPages[i]);
|
||||||
|
m_subPages.erase(m_subPages.begin()+i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
void TeletextDocument::setPageFunction(PageFunctionEnum newPageFunction)
|
void TeletextDocument::setPageFunction(PageFunctionEnum newPageFunction)
|
||||||
{
|
{
|
||||||
@@ -71,8 +130,12 @@ void TeletextDocument::selectSubPageIndex(int newSubPageIndex, bool forceRefresh
|
|||||||
// forceRefresh overrides "beyond the last subpage" check, so inserting a subpage after the last one still shows - dangerous workaround?
|
// forceRefresh overrides "beyond the last subpage" check, so inserting a subpage after the last one still shows - dangerous workaround?
|
||||||
if (forceRefresh || (newSubPageIndex != m_currentSubPageIndex && newSubPageIndex < m_subPages.size())) {
|
if (forceRefresh || (newSubPageIndex != m_currentSubPageIndex && newSubPageIndex < m_subPages.size())) {
|
||||||
emit aboutToChangeSubPage();
|
emit aboutToChangeSubPage();
|
||||||
|
|
||||||
m_currentSubPageIndex = newSubPageIndex;
|
m_currentSubPageIndex = newSubPageIndex;
|
||||||
|
|
||||||
|
m_clutModel->setSubPage(m_subPages[m_currentSubPageIndex]);
|
||||||
emit subPageSelected();
|
emit subPageSelected();
|
||||||
|
emit selectionMoved();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -81,8 +144,12 @@ void TeletextDocument::selectSubPageNext()
|
|||||||
{
|
{
|
||||||
if (m_currentSubPageIndex < m_subPages.size()-1) {
|
if (m_currentSubPageIndex < m_subPages.size()-1) {
|
||||||
emit aboutToChangeSubPage();
|
emit aboutToChangeSubPage();
|
||||||
|
|
||||||
m_currentSubPageIndex++;
|
m_currentSubPageIndex++;
|
||||||
|
|
||||||
|
m_clutModel->setSubPage(m_subPages[m_currentSubPageIndex]);
|
||||||
emit subPageSelected();
|
emit subPageSelected();
|
||||||
|
emit selectionMoved();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -90,8 +157,12 @@ void TeletextDocument::selectSubPagePrevious()
|
|||||||
{
|
{
|
||||||
if (m_currentSubPageIndex > 0) {
|
if (m_currentSubPageIndex > 0) {
|
||||||
emit aboutToChangeSubPage();
|
emit aboutToChangeSubPage();
|
||||||
|
|
||||||
m_currentSubPageIndex--;
|
m_currentSubPageIndex--;
|
||||||
|
|
||||||
|
m_clutModel->setSubPage(m_subPages[m_currentSubPageIndex]);
|
||||||
emit subPageSelected();
|
emit subPageSelected();
|
||||||
|
emit selectionMoved();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -112,6 +183,8 @@ void TeletextDocument::insertSubPage(int beforeSubPageIndex, bool copySubPage)
|
|||||||
|
|
||||||
void TeletextDocument::deleteSubPage(int subPageToDelete)
|
void TeletextDocument::deleteSubPage(int subPageToDelete)
|
||||||
{
|
{
|
||||||
|
m_clutModel->setSubPage(nullptr);
|
||||||
|
|
||||||
delete(m_subPages[subPageToDelete]);
|
delete(m_subPages[subPageToDelete]);
|
||||||
m_subPages.erase(m_subPages.begin()+subPageToDelete);
|
m_subPages.erase(m_subPages.begin()+subPageToDelete);
|
||||||
}
|
}
|
||||||
@@ -128,17 +201,12 @@ void TeletextDocument::unDeleteSubPageFromRecycle(int subPage)
|
|||||||
m_recycleSubPages.pop_back();
|
m_recycleSubPages.pop_back();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextDocument::setPageNumber(QString pageNumberString)
|
void TeletextDocument::setPageNumber(int pageNumber)
|
||||||
{
|
{
|
||||||
bool pageNumberOk;
|
|
||||||
int pageNumberRead = pageNumberString.toInt(&pageNumberOk, 16);
|
|
||||||
if ((!pageNumberOk) || pageNumberRead < 0x100 || pageNumberRead > 0x8ff)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// If the magazine number was changed, we need to update the relative magazine numbers in FastText
|
// If the magazine number was changed, we need to update the relative magazine numbers in FastText
|
||||||
// and page enhancement links
|
// and page enhancement links
|
||||||
int oldMagazine = (m_pageNumber & 0xf00);
|
int oldMagazine = (m_pageNumber & 0xf00);
|
||||||
int newMagazine = (pageNumberRead & 0xf00);
|
int newMagazine = (pageNumber & 0xf00);
|
||||||
// Fix magazine 0 to 8
|
// Fix magazine 0 to 8
|
||||||
if (oldMagazine == 0x800)
|
if (oldMagazine == 0x800)
|
||||||
oldMagazine = 0x000;
|
oldMagazine = 0x000;
|
||||||
@@ -146,7 +214,7 @@ void TeletextDocument::setPageNumber(QString pageNumberString)
|
|||||||
newMagazine = 0x000;
|
newMagazine = 0x000;
|
||||||
int magazineFlip = oldMagazine ^ newMagazine;
|
int magazineFlip = oldMagazine ^ newMagazine;
|
||||||
|
|
||||||
m_pageNumber = pageNumberRead;
|
m_pageNumber = pageNumber;
|
||||||
|
|
||||||
for (auto &subPage : m_subPages)
|
for (auto &subPage : m_subPages)
|
||||||
if (magazineFlip) {
|
if (magazineFlip) {
|
||||||
@@ -157,6 +225,17 @@ void TeletextDocument::setPageNumber(QString pageNumberString)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TeletextDocument::setPageNumberFromString(QString pageNumberString)
|
||||||
|
{
|
||||||
|
bool pageNumberOk;
|
||||||
|
int pageNumberRead = pageNumberString.toInt(&pageNumberOk, 16);
|
||||||
|
|
||||||
|
if ((!pageNumberOk) || pageNumberRead < 0x100 || pageNumberRead > 0x8ff)
|
||||||
|
return;
|
||||||
|
|
||||||
|
setPageNumber(pageNumberRead);
|
||||||
|
}
|
||||||
|
|
||||||
void TeletextDocument::setDescription(QString newDescription)
|
void TeletextDocument::setDescription(QString newDescription)
|
||||||
{
|
{
|
||||||
m_description = newDescription;
|
m_description = newDescription;
|
||||||
@@ -168,62 +247,124 @@ void TeletextDocument::setFastTextLinkPageNumberOnAllSubPages(int linkNumber, in
|
|||||||
subPage->setFastTextLinkPageNumber(linkNumber, pageNumber);
|
subPage->setFastTextLinkPageNumber(linkNumber, pageNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextDocument::cursorUp()
|
void TeletextDocument::cursorUp(bool shiftKey)
|
||||||
{
|
{
|
||||||
|
if (shiftKey && !selectionActive())
|
||||||
|
setSelectionCorner(m_cursorRow, m_cursorColumn);
|
||||||
|
|
||||||
if (--m_cursorRow == 0)
|
if (--m_cursorRow == 0)
|
||||||
m_cursorRow = 24;
|
m_cursorRow = 24;
|
||||||
|
|
||||||
|
if (shiftKey)
|
||||||
|
emit selectionMoved();
|
||||||
|
else
|
||||||
|
cancelSelection();
|
||||||
|
|
||||||
emit cursorMoved();
|
emit cursorMoved();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextDocument::cursorDown()
|
void TeletextDocument::cursorDown(bool shiftKey)
|
||||||
{
|
{
|
||||||
|
if (shiftKey && !selectionActive())
|
||||||
|
setSelectionCorner(m_cursorRow, m_cursorColumn);
|
||||||
|
|
||||||
if (++m_cursorRow == 25)
|
if (++m_cursorRow == 25)
|
||||||
m_cursorRow = 1;
|
m_cursorRow = 1;
|
||||||
|
|
||||||
|
if (shiftKey)
|
||||||
|
emit selectionMoved();
|
||||||
|
else
|
||||||
|
cancelSelection();
|
||||||
|
|
||||||
emit cursorMoved();
|
emit cursorMoved();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextDocument::cursorLeft()
|
void TeletextDocument::cursorLeft(bool shiftKey)
|
||||||
{
|
{
|
||||||
|
if (shiftKey && !selectionActive())
|
||||||
|
setSelectionCorner(m_cursorRow, m_cursorColumn);
|
||||||
|
|
||||||
if (--m_cursorColumn == -1) {
|
if (--m_cursorColumn == -1) {
|
||||||
m_cursorColumn = 39;
|
m_cursorColumn = 39;
|
||||||
cursorUp();
|
cursorUp(shiftKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (shiftKey)
|
||||||
|
emit selectionMoved();
|
||||||
|
else
|
||||||
|
cancelSelection();
|
||||||
|
|
||||||
emit cursorMoved();
|
emit cursorMoved();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextDocument::cursorRight()
|
void TeletextDocument::cursorRight(bool shiftKey)
|
||||||
{
|
{
|
||||||
|
if (shiftKey && !selectionActive())
|
||||||
|
setSelectionCorner(m_cursorRow, m_cursorColumn);
|
||||||
|
|
||||||
if (++m_cursorColumn == 40) {
|
if (++m_cursorColumn == 40) {
|
||||||
m_cursorColumn = 0;
|
m_cursorColumn = 0;
|
||||||
cursorDown();
|
cursorDown(shiftKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (shiftKey)
|
||||||
|
emit selectionMoved();
|
||||||
|
else
|
||||||
|
cancelSelection();
|
||||||
|
|
||||||
emit cursorMoved();
|
emit cursorMoved();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextDocument::moveCursor(int cursorRow, int cursorColumn)
|
void TeletextDocument::moveCursor(int cursorRow, int cursorColumn, bool selectionInProgress)
|
||||||
{
|
{
|
||||||
|
if (selectionInProgress && !selectionActive())
|
||||||
|
setSelectionCorner(m_cursorRow, m_cursorColumn);
|
||||||
|
|
||||||
if (cursorRow != -1)
|
if (cursorRow != -1)
|
||||||
m_cursorRow = cursorRow;
|
m_cursorRow = cursorRow;
|
||||||
if (cursorColumn != -1)
|
if (cursorColumn != -1)
|
||||||
m_cursorColumn = cursorColumn;
|
m_cursorColumn = cursorColumn;
|
||||||
|
|
||||||
|
if (selectionInProgress)
|
||||||
|
emit selectionMoved();
|
||||||
|
else
|
||||||
|
cancelSelection();
|
||||||
|
|
||||||
emit cursorMoved();
|
emit cursorMoved();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TeletextDocument::setSelectionCorner(int row, int column)
|
||||||
|
{
|
||||||
|
if (m_selectionCornerRow != row || m_selectionCornerColumn != column) {
|
||||||
|
m_selectionSubPage = currentSubPage();
|
||||||
|
m_selectionCornerRow = row;
|
||||||
|
m_selectionCornerColumn = column;
|
||||||
|
|
||||||
|
// emit selectionMoved();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void TeletextDocument::setSelection(int topRow, int leftColumn, int bottomRow, int rightColumn)
|
void TeletextDocument::setSelection(int topRow, int leftColumn, int bottomRow, int rightColumn)
|
||||||
{
|
{
|
||||||
if (m_selectionTopRow != topRow || m_selectionBottomRow != bottomRow || m_selectionLeftColumn != leftColumn || m_selectionRightColumn != rightColumn) {
|
if (selectionTopRow() != topRow || selectionBottomRow() != bottomRow || selectionLeftColumn() != leftColumn || selectionRightColumn() != rightColumn) {
|
||||||
m_selectionSubPage = currentSubPage();
|
m_selectionSubPage = currentSubPage();
|
||||||
m_selectionTopRow = topRow;
|
m_selectionCornerRow = topRow;
|
||||||
m_selectionBottomRow = bottomRow;
|
m_cursorRow = bottomRow;
|
||||||
m_selectionLeftColumn = leftColumn;
|
m_selectionCornerColumn = leftColumn;
|
||||||
m_selectionRightColumn = rightColumn;
|
m_cursorColumn = rightColumn;
|
||||||
|
|
||||||
emit selectionMoved();
|
emit selectionMoved();
|
||||||
|
emit cursorMoved();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextDocument::cancelSelection()
|
void TeletextDocument::cancelSelection()
|
||||||
{
|
{
|
||||||
|
if (m_selectionSubPage != nullptr) {
|
||||||
m_selectionSubPage = nullptr;
|
m_selectionSubPage = nullptr;
|
||||||
|
emit selectionMoved();
|
||||||
|
m_selectionCornerRow = m_selectionCornerColumn = -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int TeletextDocument::levelRequired() const
|
int TeletextDocument::levelRequired() const
|
||||||
|
|||||||
76
document.h
76
document.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -20,11 +20,28 @@
|
|||||||
#ifndef DOCUMENT_H
|
#ifndef DOCUMENT_H
|
||||||
#define DOCUMENT_H
|
#define DOCUMENT_H
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QUndoStack>
|
#include <QUndoStack>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "levelonepage.h"
|
#include "levelonepage.h"
|
||||||
|
|
||||||
|
class ClutModel : public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
ClutModel(QObject *parent = 0);
|
||||||
|
|
||||||
|
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
|
void setSubPage(LevelOnePage *page);
|
||||||
|
|
||||||
|
private:
|
||||||
|
LevelOnePage *m_subPage;
|
||||||
|
};
|
||||||
|
|
||||||
class TeletextDocument : public QObject
|
class TeletextDocument : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@@ -39,6 +56,7 @@ public:
|
|||||||
~TeletextDocument();
|
~TeletextDocument();
|
||||||
|
|
||||||
bool isEmpty() const;
|
bool isEmpty() const;
|
||||||
|
void clear();
|
||||||
|
|
||||||
PageFunctionEnum pageFunction() const { return m_pageFunction; }
|
PageFunctionEnum pageFunction() const { return m_pageFunction; }
|
||||||
// void setPageFunction(PageFunctionEnum);
|
// void setPageFunction(PageFunctionEnum);
|
||||||
@@ -49,47 +67,52 @@ public:
|
|||||||
LevelOnePage* subPage(int p) const { return m_subPages[p]; }
|
LevelOnePage* subPage(int p) const { return m_subPages[p]; }
|
||||||
LevelOnePage* currentSubPage() const { return m_subPages[m_currentSubPageIndex]; }
|
LevelOnePage* currentSubPage() const { return m_subPages[m_currentSubPageIndex]; }
|
||||||
int currentSubPageIndex() const { return m_currentSubPageIndex; }
|
int currentSubPageIndex() const { return m_currentSubPageIndex; }
|
||||||
void selectSubPageIndex(int, bool=false);
|
void selectSubPageIndex(int newSubPageIndex, bool refresh=false);
|
||||||
void selectSubPageNext();
|
void selectSubPageNext();
|
||||||
void selectSubPagePrevious();
|
void selectSubPagePrevious();
|
||||||
void insertSubPage(int, bool);
|
void insertSubPage(int beforeSubPageIndex, bool copySubPage);
|
||||||
void deleteSubPage(int);
|
void deleteSubPage(int subPageToDelete);
|
||||||
void deleteSubPageToRecycle(int);
|
void deleteSubPageToRecycle(int subPageToRecycle);
|
||||||
void unDeleteSubPageFromRecycle(int);
|
void unDeleteSubPageFromRecycle(int subPage);
|
||||||
int pageNumber() const { return m_pageNumber; }
|
int pageNumber() const { return m_pageNumber; }
|
||||||
void setPageNumber(QString);
|
void setPageNumber(int pageNumber);
|
||||||
|
void setPageNumberFromString(QString pageNumberString);
|
||||||
QString description() const { return m_description; }
|
QString description() const { return m_description; }
|
||||||
void setDescription(QString);
|
void setDescription(QString newDescription);
|
||||||
void setFastTextLinkPageNumberOnAllSubPages(int, int);
|
void setFastTextLinkPageNumberOnAllSubPages(int linkNumber, int pageNumber);
|
||||||
QUndoStack *undoStack() const { return m_undoStack; }
|
QUndoStack *undoStack() const { return m_undoStack; }
|
||||||
|
ClutModel *clutModel() const { return m_clutModel; }
|
||||||
int cursorRow() const { return m_cursorRow; }
|
int cursorRow() const { return m_cursorRow; }
|
||||||
int cursorColumn() const { return m_cursorColumn; }
|
int cursorColumn() const { return m_cursorColumn; }
|
||||||
void cursorUp();
|
void cursorUp(bool shiftKey=false);
|
||||||
void cursorDown();
|
void cursorDown(bool shiftKey=false);
|
||||||
void cursorLeft();
|
void cursorLeft(bool shiftKey=false);
|
||||||
void cursorRight();
|
void cursorRight(bool shiftKey=false);
|
||||||
void moveCursor(int, int);
|
void moveCursor(int cursorRow, int cursorColumn, bool selectionInProgress=false);
|
||||||
int selectionTopRow() const { return m_selectionTopRow; }
|
int selectionTopRow() const { return m_selectionCornerRow == -1 ? m_cursorRow : qMin(m_selectionCornerRow, m_cursorRow); }
|
||||||
int selectionBottomRow() const { return m_selectionBottomRow; }
|
int selectionBottomRow() const { return qMax(m_selectionCornerRow, m_cursorRow); }
|
||||||
int selectionLeftColumn() const { return m_selectionLeftColumn; }
|
int selectionLeftColumn() const { return m_selectionCornerColumn == -1 ? m_cursorColumn : qMin(m_selectionCornerColumn, m_cursorColumn); }
|
||||||
int selectionRightColumn() const { return m_selectionRightColumn; }
|
int selectionRightColumn() const { return qMax(m_selectionCornerColumn, m_cursorColumn); }
|
||||||
int selectionWidth() const { return m_selectionRightColumn - m_selectionLeftColumn + 1; }
|
int selectionWidth() const { return m_selectionCornerColumn == -1 ? 1 : selectionRightColumn() - selectionLeftColumn() + 1; }
|
||||||
int selectionHeight() const { return m_selectionBottomRow - m_selectionTopRow + 1; }
|
int selectionHeight() const { return m_selectionCornerRow == -1 ? 1 : selectionBottomRow() - selectionTopRow() + 1; }
|
||||||
bool selectionActive() const { return m_selectionSubPage == currentSubPage(); }
|
bool selectionActive() const { return m_selectionSubPage == currentSubPage(); }
|
||||||
void setSelection(int, int, int, int);
|
int selectionCornerRow() const { return m_selectionCornerRow == -1 ? m_cursorRow : m_selectionCornerRow; }
|
||||||
|
int selectionCornerColumn() const { return m_selectionCornerColumn == -1 ? m_cursorColumn : m_selectionCornerColumn; }
|
||||||
|
void setSelectionCorner(int row, int column);
|
||||||
|
void setSelection(int topRow, int leftColumn, int bottomRow, int rightColumn);
|
||||||
void cancelSelection();
|
void cancelSelection();
|
||||||
int levelRequired() const;
|
int levelRequired() const;
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void cursorMoved();
|
void cursorMoved();
|
||||||
void selectionMoved();
|
void selectionMoved();
|
||||||
void colourChanged(int);
|
void colourChanged(int i);
|
||||||
void contentsChange(int);
|
void pageOptionsChanged();
|
||||||
void aboutToChangeSubPage();
|
void aboutToChangeSubPage();
|
||||||
void subPageSelected();
|
void subPageSelected();
|
||||||
void refreshNeeded();
|
void contentsChanged();
|
||||||
|
|
||||||
void tripletCommandHighlight(int);
|
void tripletCommandHighlight(int tripletNumber);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QString m_description;
|
QString m_description;
|
||||||
@@ -99,8 +122,9 @@ private:
|
|||||||
std::vector<LevelOnePage *> m_subPages;
|
std::vector<LevelOnePage *> m_subPages;
|
||||||
std::vector<LevelOnePage *> m_recycleSubPages;
|
std::vector<LevelOnePage *> m_recycleSubPages;
|
||||||
QUndoStack *m_undoStack;
|
QUndoStack *m_undoStack;
|
||||||
int m_cursorRow, m_cursorColumn, m_selectionTopRow, m_selectionBottomRow, m_selectionLeftColumn, m_selectionRightColumn;
|
int m_cursorRow, m_cursorColumn, m_selectionCornerRow, m_selectionCornerColumn;
|
||||||
LevelOnePage *m_selectionSubPage;
|
LevelOnePage *m_selectionSubPage;
|
||||||
|
ClutModel *m_clutModel;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
23
examples/Level1p5-Defcon.tti
Normal file
23
examples/Level1p5-Defcon.tti
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
PN,11600
|
||||||
|
SC,0000
|
||||||
|
PS,8000
|
||||||
|
CT,8,T
|
||||||
|
OL,26,@lD@Ib\UbTXb\ZbT]b\`B]mD@HB{VBsWB{[Bs\B{
|
||||||
|
OL,26,AaBRnD@`BraB]oD@HBXVBPWBX[BP\BX`B}pD@Iby
|
||||||
|
OL,26,BUbqXbyZbq]byaBruD@KBYLBSRbTVb\ZbT^b\cbY
|
||||||
|
OL,26,CdBSvD@JBYKBSLBywD@Fb\JBTKByLBsVB^ZbX[bU
|
||||||
|
OL,26,D\bS^bxxD@[bx\bu]bsyD@BbqFbyRbqVbyCC
|
||||||
|
OL,3,W <,,,,,,,,,,,,,,,,,,,,,,,,,,,,,l
|
||||||
|
OL,4,W 5t xt xt 5 j
|
||||||
|
OL,5,W 5 " !"/ !" j
|
||||||
|
OL,6,W 5 j5j
|
||||||
|
OL,7,W 5 ` 0`| 0` j
|
||||||
|
OL,8,W 5' +' +' jj
|
||||||
|
OL,9,W -,,,,,,,,,,,,,,,,,,,,,,,,,,,,,.
|
||||||
|
OL,12,V<,,,,,lR<,,,,,lS<,,,,,lQ<,,,,,lW<,,,,,l
|
||||||
|
OL,13,V5jR5 `5jS5xtjQ5xtjW5 `~5 j
|
||||||
|
OL,14,V55 jR5 ~75jS5/% jjQ5/% jjW5 #5 j
|
||||||
|
OL,15,V5tjR5z? 5jS5 j5jQ5`x~/jW5 5 j
|
||||||
|
OL,16,V5|4 jjR5jS5|4 jjQ5?# jW5 5 j
|
||||||
|
OL,17,V5+'jR5 5jS5+'jQ5jW5 5 j
|
||||||
|
OL,18,V-,,,,,.R-,,,,,.S-,,,,,.Q-,,,,,.W-,,,,,.
|
||||||
71
examples/Level3p5-CharDesignation.tti
Normal file
71
examples/Level3p5-CharDesignation.tti
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
DE,Unlimited character sets with Level 3.5
|
||||||
|
PN,19901
|
||||||
|
SC,0001
|
||||||
|
PS,8000
|
||||||
|
CT,8,T
|
||||||
|
OL,28,@@@|gpCu_@|wKpZA`UB_GQd^w}ww]_}_wM@D
|
||||||
|
OL,28,D@@|g@@pC@|p@@CO|O@pA@\p]@@wAG\gYF@D
|
||||||
|
OL,26,@laIA@Bma|mDRdo}eO~fo~gOoayA@BpawXH@Xi`
|
||||||
|
OL,26,AYIaZia[Ib]ip^Iq_iq`IrbI~do}eO~fo~gOrA|
|
||||||
|
OL,26,BXHPXi`YIaZia[Ib]ip^Iq_iq`IrbI~do}eO~fo~
|
||||||
|
OL,26,CgOtawXHRXi`YIaZia[Ib]ip^Iq_iq`IrbI~do}
|
||||||
|
OL,26,DeO~fo~gOvA|XhRXi`YIaZia[Ib]ip^Iq_iq`Ir
|
||||||
|
OL,26,EbI~do}eO~fo~gOxawXh[Xi`YIaZia[Ib]ip^Iq
|
||||||
|
OL,26,F_iq`IrbI~do}eO~fo~gOzA|XhjXi`YIaZia[Ib
|
||||||
|
OL,26,G]ip^Iq_iq`IrbI~do}eO~fo~gO|awXhkXi`YIa
|
||||||
|
OL,26,HZia[Ib]ip^Iq_iq`IrbI~do}eO~fo~gO~A|XH`
|
||||||
|
OL,26,IXi`YIaZia[Ib]ip^Iq_iq`IrbI~do}eO~fo~gO
|
||||||
|
OL,26,JhApCCCCCCCCCCCC
|
||||||
|
OL,1, Level 3.5 allows unlimited char F(1/2)
|
||||||
|
OL,2, sets using the "Modified G0 and G2
|
||||||
|
OL,3, Character Set Designation" triplet.
|
||||||
|
OL,4,F `G0 lvl 1`` -G2`
|
||||||
|
OL,5,FLevel 1 with G ABCD abcd |
|
||||||
|
OL,6,F Level 1.5 G2 chars
|
||||||
|
OL,7,F`Char set```` `Bits`` `G0 X/26``` `G2`
|
||||||
|
OL,8,FLatin B0000xxxG
|
||||||
|
OL,10,FCyrillic 1 B0100000G
|
||||||
|
OL,11,F Serbian/Croatian
|
||||||
|
OL,12,FCyrillic 2 B0100100G
|
||||||
|
OL,13,F Russian/Bulgarian
|
||||||
|
OL,14,FCyrillic 3 B0100101G
|
||||||
|
OL,15,F Ukranian
|
||||||
|
OL,16,FGreek B0110111G
|
||||||
|
OL,18,FHebrew B1010101G
|
||||||
|
OL,20,FArabic B1010111G
|
||||||
|
OL,22,FLatin with B1000000G
|
||||||
|
OL,23,F Arabic G2
|
||||||
|
PN,19902
|
||||||
|
SC,0002
|
||||||
|
PS,8000
|
||||||
|
CT,8,T
|
||||||
|
OL,28,@@@|gpCu_@|wKpZA`UB_GQd^w}ww]_}_wM@D
|
||||||
|
OL,28,D@@|g@@pC@|p@@CO|O@pA@\p]@@wAG\gYF@D
|
||||||
|
OL,26,@layA@Bma|nD@Ao|BO}Co}DO~Eo~FOoayA@Bpaw
|
||||||
|
OL,26,AhQaHH@qD@@H@rA|hQaHHPsD@@HPtawhQaHHRuD@
|
||||||
|
OL,26,B@HRvA|hQaHhRwD@@hRxawhQaHh[yD@@h[zA|hQa
|
||||||
|
OL,26,CHhj{D@@hj|awhQaHhk}D@@hk~A|hQaHH`D@@H`
|
||||||
|
OL,26,DhAp_CxUaHI`Ii`JIaKiaLIbMibNIcOicPIdQid
|
||||||
|
OL,26,ERIeSieTIfUifVIgWigXIhYihZIi[ii\Ij]ij^Ik
|
||||||
|
OL,26,F_ik`IlailbImcimdIneinfIogioiD@Ao|BO}Co}
|
||||||
|
OL,26,GDO~Eo~FOHIpIipJIqKiqLIrMirNIsOisPItQit
|
||||||
|
OL,26,HRIuSiuTIvUivVIwWiwXIxYixZIy[iy\Iz]iz^I{
|
||||||
|
OL,26,I_i{`I|ai|bI}ci}dI~ei~fIBBBBB
|
||||||
|
OL,1, Level 3.5 allows unlimited char F(2/2)
|
||||||
|
OL,2, sets using the "Modified G0 and G2
|
||||||
|
OL,3, Character Set Designation" triplet.
|
||||||
|
OL,4,F`G2``` `G0 from level 1`````````` `NOS`
|
||||||
|
OL,5,FL1.5 G@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_
|
||||||
|
OL,6, `abcdefghijklmnopqrstuvwxyz{|}~
|
||||||
|
OL,7,F`G2``` `G0 from X/26 enhancements``````
|
||||||
|
OL,8,FLatin G
|
||||||
|
OL,10,FCyr 1 CG CG C G C FGCFC F
|
||||||
|
OL,11, CG CG C G C FGCFC
|
||||||
|
OL,12,FCyr 2 CG CG C G C FGCFC F
|
||||||
|
OL,13, CG CG C G C FGCFC
|
||||||
|
OL,14,FCyr 3 CG CG C G C FGCFC F
|
||||||
|
OL,15, CG CG C G C FGCFC
|
||||||
|
OL,16,FGreek G
|
||||||
|
OL,18,FHebrewG
|
||||||
|
OL,20,FArabicG
|
||||||
|
OL,22,FL/AbG2G
|
||||||
33
examples/SidePanels-PriceChop.tti
Normal file
33
examples/SidePanels-PriceChop.tti
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
PN,13200
|
||||||
|
SC,0000
|
||||||
|
PS,8000
|
||||||
|
CT,8,T
|
||||||
|
OL,28,@@@|H@p_@|wsA@@AfUrLs_w}ww]_}_wM@p
|
||||||
|
OL,26,@rD@Ab|BBpKBxLbtsD@BB{CbqJbyKBstD@Db|EBp
|
||||||
|
OL,26,AHBxIbtuD@EB{FbqGbyHBswD@AbqLbyxD@Bb|CBp
|
||||||
|
OL,26,BJBxKbtyD@CB{DbqIbyJBszD@Eb|FBpGBxHbt{D@
|
||||||
|
OL,26,CFB{GBs}aJ@cJhPThr]hP`hre~aI@cI_Cxv]@It
|
||||||
|
OL,26,DAipBitCIwGi\iD@@@H@irAiyBIzCirDIyxve@Iy
|
||||||
|
OL,26,EAIqCiqDItEipFitGIwiD@@@HAisBIvCipDiyEis
|
||||||
|
OL,26,FFiwGi{BBBBBBBBBBB
|
||||||
|
OL,1, calls cost #5 WZ`p0ppb`p0pp `p0up`p0pp
|
||||||
|
OL,2, 0909 879 0100 WZjp55"jj 1=.(j 15jj 5uz
|
||||||
|
OL,3, GB #9.99 p&p WZ* ! ""#!## "#!!""#!%
|
||||||
|
OL,4, W Zn,h h
|
||||||
|
OL,5,C]Dstart #89 \ W Zjp"d&
|
||||||
|
OL,6,S############
|
||||||
|
OL,7, LIVE
|
||||||
|
OL,8,E]G now at \
|
||||||
|
OL,9,Ussssssssssss
|
||||||
|
OL,10,U+'
|
||||||
|
OL,11,U^"o]G#3U?\!
|
||||||
|
OL,12,U +'
|
||||||
|
OL,13,U "o?!
|
||||||
|
OL,14,Spppppppppppp
|
||||||
|
OL,15,S+]D^leftS\'
|
||||||
|
OL,16,S^"o]D24S?\!
|
||||||
|
OL,17,S +'
|
||||||
|
OL,18,S "o?!
|
||||||
|
OL,19,S
|
||||||
|
OL,21,E]G9ct gold curb chain 9ct gold curb c
|
||||||
|
OL,22,C]DNEW BUYER JOHN london JANET manch
|
||||||
36
examples/SubObjects-TeletextStoreBag.tti
Normal file
36
examples/SubObjects-TeletextStoreBag.tti
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
DE,Parody of Microsoft Store icon
|
||||||
|
PN,15400
|
||||||
|
SC,0000
|
||||||
|
PS,8000
|
||||||
|
CT,8,T
|
||||||
|
OL,28,@@@|gpCu_@|wKpZA`UB_GdN`M\LXKTJPILXwM@@
|
||||||
|
OL,26,@iD@hq]jD@hq]kD@hq]lD@hq]mD@hq]nD@hq]oD@
|
||||||
|
OL,26,Ahq]pD@ACLN@LPbZQ`OQcOXCOXbRqD@hQarD@hQa
|
||||||
|
OL,26,BsD@ACLJcO_COtD@hQdvD@hQdxD@hQdzD@hQd|D@
|
||||||
|
OL,26,CACLJBp^CO^Bx}D@hq]~D@hq]D@hq]_Cxu]A@L
|
||||||
|
OL,26,DhshXCOxUaACLhshN@LQ@LYCOxUdACL_COiD@ACL
|
||||||
|
OL,26,E_COxwhQ`LQaR@MRaS`MSaT@NTaU`NUaV@O
|
||||||
|
OL,26,FVaBBBBBBBBBBBB
|
||||||
|
OL,1,Twwf$$$
|
||||||
|
OL,2,T==9111
|
||||||
|
OL,3,Twwf$$$FWelcome to the
|
||||||
|
OL,4,T==9111
|
||||||
|
OL,5,Twwf$$$WTELETEXT STORE
|
||||||
|
OL,6,T==9111
|
||||||
|
OL,7,Twwf$$$
|
||||||
|
OL,8,T] ^xWV] tW\
|
||||||
|
OL,9,T] ^WT\wwf$$W
|
||||||
|
OL,10,T] ^WT\==911W
|
||||||
|
OL,11,T] V]W \
|
||||||
|
OL,12,T] W] \
|
||||||
|
OL,13,T] W] \
|
||||||
|
OL,14,T] W]Q RS \
|
||||||
|
OL,15,T] W]Q RS \
|
||||||
|
OL,16,T] W]Q \
|
||||||
|
OL,17,T] W]Q TUV \
|
||||||
|
OL,18,T] W]Q TUV \
|
||||||
|
OL,19,T] W] \
|
||||||
|
OL,20,T] Wo] ^?\
|
||||||
|
OL,21,Twwf$$$
|
||||||
|
OL,22,T==9111
|
||||||
|
OL,23,Twwf$$$
|
||||||
9
examples/Template-ActivePosition.tti
Normal file
9
examples/Template-ActivePosition.tti
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
PN,19900
|
||||||
|
SC,0000
|
||||||
|
PS,8000
|
||||||
|
CT,8,T
|
||||||
|
OL,26,@iD@@kfjD@@kfkD@@kflD@@kfmD@@kfnD@@kfoD@
|
||||||
|
OL,26,A@kfpD@@kfqD@@kfrD@@kfsD@@kftD@@kfuD@@kf
|
||||||
|
OL,26,BvD@@kfwD@@kfxD@@kfxD@@kfyD@@kfzD@@kf{D@
|
||||||
|
OL,26,C@kf|D@@kf}D@@kf~D@@kfD@@kfCCCC
|
||||||
|
OL,1, Active Position across every row
|
||||||
262
hamming.h
Normal file
262
hamming.h
Normal file
@@ -0,0 +1,262 @@
|
|||||||
|
#ifndef HAMMING_H
|
||||||
|
#define HAMMING_H
|
||||||
|
|
||||||
|
// Hamming 8/4 encoding table
|
||||||
|
// encoded_value = hamming_8_4_encode[value_to_encode]
|
||||||
|
const unsigned char hamming_8_4_encode[16] = {
|
||||||
|
0x15, 0x02, 0x49, 0x5e, 0x64, 0x73, 0x38, 0x2f,
|
||||||
|
0xd0, 0xc7, 0x8c, 0x9b, 0xa1, 0xb6, 0xfd, 0xea
|
||||||
|
};
|
||||||
|
|
||||||
|
// Hamming 8/4 decoding table
|
||||||
|
// decoded_value = hamming_8_4_decode[encoded_value]
|
||||||
|
// 0xff - double bit error that can't be corrected
|
||||||
|
const unsigned char hamming_8_4_decode[256] = {
|
||||||
|
0x01, 0xff, 0x01, 0x01, 0xff, 0x00, 0x01, 0xff,
|
||||||
|
0xff, 0x02, 0x01, 0xff, 0x0a, 0xff, 0xff, 0x07,
|
||||||
|
0xff, 0x00, 0x01, 0xff, 0x00, 0x00, 0xff, 0x00,
|
||||||
|
0x06, 0xff, 0xff, 0x0b, 0xff, 0x00, 0x03, 0xff,
|
||||||
|
0xff, 0x0c, 0x01, 0xff, 0x04, 0xff, 0xff, 0x07,
|
||||||
|
0x06, 0xff, 0xff, 0x07, 0xff, 0x07, 0x07, 0x07,
|
||||||
|
0x06, 0xff, 0xff, 0x05, 0xff, 0x00, 0x0d, 0xff,
|
||||||
|
0x06, 0x06, 0x06, 0xff, 0x06, 0xff, 0xff, 0x07,
|
||||||
|
0xff, 0x02, 0x01, 0xff, 0x04, 0xff, 0xff, 0x09,
|
||||||
|
0x02, 0x02, 0xff, 0x02, 0xff, 0x02, 0x03, 0xff,
|
||||||
|
0x08, 0xff, 0xff, 0x05, 0xff, 0x00, 0x03, 0xff,
|
||||||
|
0xff, 0x02, 0x03, 0xff, 0x03, 0xff, 0x03, 0x03,
|
||||||
|
0x04, 0xff, 0xff, 0x05, 0x04, 0x04, 0x04, 0xff,
|
||||||
|
0xff, 0x02, 0x0f, 0xff, 0x04, 0xff, 0xff, 0x07,
|
||||||
|
0xff, 0x05, 0x05, 0x05, 0x04, 0xff, 0xff, 0x05,
|
||||||
|
0x06, 0xff, 0xff, 0x05, 0xff, 0x0e, 0x03, 0xff,
|
||||||
|
0xff, 0x0c, 0x01, 0xff, 0x0a, 0xff, 0xff, 0x09,
|
||||||
|
0x0a, 0xff, 0xff, 0x0b, 0x0a, 0x0a, 0x0a, 0xff,
|
||||||
|
0x08, 0xff, 0xff, 0x0b, 0xff, 0x00, 0x0d, 0xff,
|
||||||
|
0xff, 0x0b, 0x0b, 0x0b, 0x0a, 0xff, 0xff, 0x0b,
|
||||||
|
0x0c, 0x0c, 0xff, 0x0c, 0xff, 0x0c, 0x0d, 0xff,
|
||||||
|
0xff, 0x0c, 0x0f, 0xff, 0x0a, 0xff, 0xff, 0x07,
|
||||||
|
0xff, 0x0c, 0x0d, 0xff, 0x0d, 0xff, 0x0d, 0x0d,
|
||||||
|
0x06, 0xff, 0xff, 0x0b, 0xff, 0x0e, 0x0d, 0xff,
|
||||||
|
0x08, 0xff, 0xff, 0x09, 0xff, 0x09, 0x09, 0x09,
|
||||||
|
0xff, 0x02, 0x0f, 0xff, 0x0a, 0xff, 0xff, 0x09,
|
||||||
|
0x08, 0x08, 0x08, 0xff, 0x08, 0xff, 0xff, 0x09,
|
||||||
|
0x08, 0xff, 0xff, 0x0b, 0xff, 0x0e, 0x03, 0xff,
|
||||||
|
0xff, 0x0c, 0x0f, 0xff, 0x04, 0xff, 0xff, 0x09,
|
||||||
|
0x0f, 0xff, 0x0f, 0x0f, 0xff, 0x0e, 0x0f, 0xff,
|
||||||
|
0x08, 0xff, 0xff, 0x05, 0xff, 0x0e, 0x0d, 0xff,
|
||||||
|
0xff, 0x0e, 0x0f, 0xff, 0x0e, 0x0e, 0xff, 0x0e
|
||||||
|
};
|
||||||
|
|
||||||
|
const unsigned char hamming_24_18_forward[2][256] = {
|
||||||
|
{
|
||||||
|
0x8b, 0x8c, 0x92, 0x95, 0xa1, 0xa6, 0xb8, 0xbf,
|
||||||
|
0xc0, 0xc7, 0xd9, 0xde, 0xea, 0xed, 0xf3, 0xf4,
|
||||||
|
0x0a, 0x0d, 0x13, 0x14, 0x20, 0x27, 0x39, 0x3e,
|
||||||
|
0x41, 0x46, 0x58, 0x5f, 0x6b, 0x6c, 0x72, 0x75,
|
||||||
|
0x09, 0x0e, 0x10, 0x17, 0x23, 0x24, 0x3a, 0x3d,
|
||||||
|
0x42, 0x45, 0x5b, 0x5c, 0x68, 0x6f, 0x71, 0x76,
|
||||||
|
0x88, 0x8f, 0x91, 0x96, 0xa2, 0xa5, 0xbb, 0xbc,
|
||||||
|
0xc3, 0xc4, 0xda, 0xdd, 0xe9, 0xee, 0xf0, 0xf7,
|
||||||
|
0x08, 0x0f, 0x11, 0x16, 0x22, 0x25, 0x3b, 0x3c,
|
||||||
|
0x43, 0x44, 0x5a, 0x5d, 0x69, 0x6e, 0x70, 0x77,
|
||||||
|
0x89, 0x8e, 0x90, 0x97, 0xa3, 0xa4, 0xba, 0xbd,
|
||||||
|
0xc2, 0xc5, 0xdb, 0xdc, 0xe8, 0xef, 0xf1, 0xf6,
|
||||||
|
0x8a, 0x8d, 0x93, 0x94, 0xa0, 0xa7, 0xb9, 0xbe,
|
||||||
|
0xc1, 0xc6, 0xd8, 0xdf, 0xeb, 0xec, 0xf2, 0xf5,
|
||||||
|
0x0b, 0x0c, 0x12, 0x15, 0x21, 0x26, 0x38, 0x3f,
|
||||||
|
0x40, 0x47, 0x59, 0x5e, 0x6a, 0x6d, 0x73, 0x74,
|
||||||
|
0x03, 0x04, 0x1a, 0x1d, 0x29, 0x2e, 0x30, 0x37,
|
||||||
|
0x48, 0x4f, 0x51, 0x56, 0x62, 0x65, 0x7b, 0x7c,
|
||||||
|
0x82, 0x85, 0x9b, 0x9c, 0xa8, 0xaf, 0xb1, 0xb6,
|
||||||
|
0xc9, 0xce, 0xd0, 0xd7, 0xe3, 0xe4, 0xfa, 0xfd,
|
||||||
|
0x81, 0x86, 0x98, 0x9f, 0xab, 0xac, 0xb2, 0xb5,
|
||||||
|
0xca, 0xcd, 0xd3, 0xd4, 0xe0, 0xe7, 0xf9, 0xfe,
|
||||||
|
0x00, 0x07, 0x19, 0x1e, 0x2a, 0x2d, 0x33, 0x34,
|
||||||
|
0x4b, 0x4c, 0x52, 0x55, 0x61, 0x66, 0x78, 0x7f,
|
||||||
|
0x80, 0x87, 0x99, 0x9e, 0xaa, 0xad, 0xb3, 0xb4,
|
||||||
|
0xcb, 0xcc, 0xd2, 0xd5, 0xe1, 0xe6, 0xf8, 0xff,
|
||||||
|
0x01, 0x06, 0x18, 0x1f, 0x2b, 0x2c, 0x32, 0x35,
|
||||||
|
0x4a, 0x4d, 0x53, 0x54, 0x60, 0x67, 0x79, 0x7e,
|
||||||
|
0x02, 0x05, 0x1b, 0x1c, 0x28, 0x2f, 0x31, 0x36,
|
||||||
|
0x49, 0x4e, 0x50, 0x57, 0x63, 0x64, 0x7a, 0x7d,
|
||||||
|
0x83, 0x84, 0x9a, 0x9d, 0xa9, 0xae, 0xb0, 0xb7,
|
||||||
|
0xc8, 0xcf, 0xd1, 0xd6, 0xe2, 0xe5, 0xfb, 0xfc
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x00, 0x89, 0x8a, 0x03, 0x8b, 0x02, 0x01, 0x88,
|
||||||
|
0x01, 0x88, 0x8b, 0x02, 0x8a, 0x03, 0x00, 0x89,
|
||||||
|
0x02, 0x8b, 0x88, 0x01, 0x89, 0x00, 0x03, 0x8a,
|
||||||
|
0x03, 0x8a, 0x89, 0x00, 0x88, 0x01, 0x02, 0x8b,
|
||||||
|
0x03, 0x8a, 0x89, 0x00, 0x88, 0x01, 0x02, 0x8b,
|
||||||
|
0x02, 0x8b, 0x88, 0x01, 0x89, 0x00, 0x03, 0x8a,
|
||||||
|
0x01, 0x88, 0x8b, 0x02, 0x8a, 0x03, 0x00, 0x89,
|
||||||
|
0x00, 0x89, 0x8a, 0x03, 0x8b, 0x02, 0x01, 0x88,
|
||||||
|
0x08, 0x81, 0x82, 0x0b, 0x83, 0x0a, 0x09, 0x80,
|
||||||
|
0x09, 0x80, 0x83, 0x0a, 0x82, 0x0b, 0x08, 0x81,
|
||||||
|
0x0a, 0x83, 0x80, 0x09, 0x81, 0x08, 0x0b, 0x82,
|
||||||
|
0x0b, 0x82, 0x81, 0x08, 0x80, 0x09, 0x0a, 0x83,
|
||||||
|
0x0b, 0x82, 0x81, 0x08, 0x80, 0x09, 0x0a, 0x83,
|
||||||
|
0x0a, 0x83, 0x80, 0x09, 0x81, 0x08, 0x0b, 0x82,
|
||||||
|
0x09, 0x80, 0x83, 0x0a, 0x82, 0x0b, 0x08, 0x81,
|
||||||
|
0x08, 0x81, 0x82, 0x0b, 0x83, 0x0a, 0x09, 0x80,
|
||||||
|
0x09, 0x80, 0x83, 0x0a, 0x82, 0x0b, 0x08, 0x81,
|
||||||
|
0x08, 0x81, 0x82, 0x0b, 0x83, 0x0a, 0x09, 0x80,
|
||||||
|
0x0b, 0x82, 0x81, 0x08, 0x80, 0x09, 0x0a, 0x83,
|
||||||
|
0x0a, 0x83, 0x80, 0x09, 0x81, 0x08, 0x0b, 0x82,
|
||||||
|
0x0a, 0x83, 0x80, 0x09, 0x81, 0x08, 0x0b, 0x82,
|
||||||
|
0x0b, 0x82, 0x81, 0x08, 0x80, 0x09, 0x0a, 0x83,
|
||||||
|
0x08, 0x81, 0x82, 0x0b, 0x83, 0x0a, 0x09, 0x80,
|
||||||
|
0x09, 0x80, 0x83, 0x0a, 0x82, 0x0b, 0x08, 0x81,
|
||||||
|
0x01, 0x88, 0x8b, 0x02, 0x8a, 0x03, 0x00, 0x89,
|
||||||
|
0x00, 0x89, 0x8a, 0x03, 0x8b, 0x02, 0x01, 0x88,
|
||||||
|
0x03, 0x8a, 0x89, 0x00, 0x88, 0x01, 0x02, 0x8b,
|
||||||
|
0x02, 0x8b, 0x88, 0x01, 0x89, 0x00, 0x03, 0x8a,
|
||||||
|
0x02, 0x8b, 0x88, 0x01, 0x89, 0x00, 0x03, 0x8a,
|
||||||
|
0x03, 0x8a, 0x89, 0x00, 0x88, 0x01, 0x02, 0x8b,
|
||||||
|
0x00, 0x89, 0x8a, 0x03, 0x8b, 0x02, 0x01, 0x88,
|
||||||
|
0x01, 0x88, 0x8b, 0x02, 0x8a, 0x03, 0x00, 0x89
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const unsigned char hamming_24_18_forward_2[4] = {
|
||||||
|
0x00, 0x0a, 0x0b, 0x01
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const unsigned char hamming_24_18_parities[3][256] = {
|
||||||
|
{ // Parities of first byte
|
||||||
|
0x00, 0x21, 0x22, 0x03, 0x23, 0x02, 0x01, 0x20,
|
||||||
|
0x24, 0x05, 0x06, 0x27, 0x07, 0x26, 0x25, 0x04,
|
||||||
|
0x25, 0x04, 0x07, 0x26, 0x06, 0x27, 0x24, 0x05,
|
||||||
|
0x01, 0x20, 0x23, 0x02, 0x22, 0x03, 0x00, 0x21,
|
||||||
|
0x26, 0x07, 0x04, 0x25, 0x05, 0x24, 0x27, 0x06,
|
||||||
|
0x02, 0x23, 0x20, 0x01, 0x21, 0x00, 0x03, 0x22,
|
||||||
|
0x03, 0x22, 0x21, 0x00, 0x20, 0x01, 0x02, 0x23,
|
||||||
|
0x27, 0x06, 0x05, 0x24, 0x04, 0x25, 0x26, 0x07,
|
||||||
|
0x27, 0x06, 0x05, 0x24, 0x04, 0x25, 0x26, 0x07,
|
||||||
|
0x03, 0x22, 0x21, 0x00, 0x20, 0x01, 0x02, 0x23,
|
||||||
|
0x02, 0x23, 0x20, 0x01, 0x21, 0x00, 0x03, 0x22,
|
||||||
|
0x26, 0x07, 0x04, 0x25, 0x05, 0x24, 0x27, 0x06,
|
||||||
|
0x01, 0x20, 0x23, 0x02, 0x22, 0x03, 0x00, 0x21,
|
||||||
|
0x25, 0x04, 0x07, 0x26, 0x06, 0x27, 0x24, 0x05,
|
||||||
|
0x24, 0x05, 0x06, 0x27, 0x07, 0x26, 0x25, 0x04,
|
||||||
|
0x00, 0x21, 0x22, 0x03, 0x23, 0x02, 0x01, 0x20,
|
||||||
|
0x28, 0x09, 0x0a, 0x2b, 0x0b, 0x2a, 0x29, 0x08,
|
||||||
|
0x0c, 0x2d, 0x2e, 0x0f, 0x2f, 0x0e, 0x0d, 0x2c,
|
||||||
|
0x0d, 0x2c, 0x2f, 0x0e, 0x2e, 0x0f, 0x0c, 0x2d,
|
||||||
|
0x29, 0x08, 0x0b, 0x2a, 0x0a, 0x2b, 0x28, 0x09,
|
||||||
|
0x0e, 0x2f, 0x2c, 0x0d, 0x2d, 0x0c, 0x0f, 0x2e,
|
||||||
|
0x2a, 0x0b, 0x08, 0x29, 0x09, 0x28, 0x2b, 0x0a,
|
||||||
|
0x2b, 0x0a, 0x09, 0x28, 0x08, 0x29, 0x2a, 0x0b,
|
||||||
|
0x0f, 0x2e, 0x2d, 0x0c, 0x2c, 0x0d, 0x0e, 0x2f,
|
||||||
|
0x0f, 0x2e, 0x2d, 0x0c, 0x2c, 0x0d, 0x0e, 0x2f,
|
||||||
|
0x2b, 0x0a, 0x09, 0x28, 0x08, 0x29, 0x2a, 0x0b,
|
||||||
|
0x2a, 0x0b, 0x08, 0x29, 0x09, 0x28, 0x2b, 0x0a,
|
||||||
|
0x0e, 0x2f, 0x2c, 0x0d, 0x2d, 0x0c, 0x0f, 0x2e,
|
||||||
|
0x29, 0x08, 0x0b, 0x2a, 0x0a, 0x2b, 0x28, 0x09,
|
||||||
|
0x0d, 0x2c, 0x2f, 0x0e, 0x2e, 0x0f, 0x0c, 0x2d,
|
||||||
|
0x0c, 0x2d, 0x2e, 0x0f, 0x2f, 0x0e, 0x0d, 0x2c,
|
||||||
|
0x28, 0x09, 0x0a, 0x2b, 0x0b, 0x2a, 0x29, 0x08
|
||||||
|
},
|
||||||
|
{ // Parities of second byte
|
||||||
|
0x00, 0x29, 0x2a, 0x03, 0x2b, 0x02, 0x01, 0x28,
|
||||||
|
0x2c, 0x05, 0x06, 0x2f, 0x07, 0x2e, 0x2d, 0x04,
|
||||||
|
0x2d, 0x04, 0x07, 0x2e, 0x06, 0x2f, 0x2c, 0x05,
|
||||||
|
0x01, 0x28, 0x2b, 0x02, 0x2a, 0x03, 0x00, 0x29,
|
||||||
|
0x2e, 0x07, 0x04, 0x2d, 0x05, 0x2c, 0x2f, 0x06,
|
||||||
|
0x02, 0x2b, 0x28, 0x01, 0x29, 0x00, 0x03, 0x2a,
|
||||||
|
0x03, 0x2a, 0x29, 0x00, 0x28, 0x01, 0x02, 0x2b,
|
||||||
|
0x2f, 0x06, 0x05, 0x2c, 0x04, 0x2d, 0x2e, 0x07,
|
||||||
|
0x2f, 0x06, 0x05, 0x2c, 0x04, 0x2d, 0x2e, 0x07,
|
||||||
|
0x03, 0x2a, 0x29, 0x00, 0x28, 0x01, 0x02, 0x2b,
|
||||||
|
0x02, 0x2b, 0x28, 0x01, 0x29, 0x00, 0x03, 0x2a,
|
||||||
|
0x2e, 0x07, 0x04, 0x2d, 0x05, 0x2c, 0x2f, 0x06,
|
||||||
|
0x01, 0x28, 0x2b, 0x02, 0x2a, 0x03, 0x00, 0x29,
|
||||||
|
0x2d, 0x04, 0x07, 0x2e, 0x06, 0x2f, 0x2c, 0x05,
|
||||||
|
0x2c, 0x05, 0x06, 0x2f, 0x07, 0x2e, 0x2d, 0x04,
|
||||||
|
0x00, 0x29, 0x2a, 0x03, 0x2b, 0x02, 0x01, 0x28,
|
||||||
|
0x30, 0x19, 0x1a, 0x33, 0x1b, 0x32, 0x31, 0x18,
|
||||||
|
0x1c, 0x35, 0x36, 0x1f, 0x37, 0x1e, 0x1d, 0x34,
|
||||||
|
0x1d, 0x34, 0x37, 0x1e, 0x36, 0x1f, 0x1c, 0x35,
|
||||||
|
0x31, 0x18, 0x1b, 0x32, 0x1a, 0x33, 0x30, 0x19,
|
||||||
|
0x1e, 0x37, 0x34, 0x1d, 0x35, 0x1c, 0x1f, 0x36,
|
||||||
|
0x32, 0x1b, 0x18, 0x31, 0x19, 0x30, 0x33, 0x1a,
|
||||||
|
0x33, 0x1a, 0x19, 0x30, 0x18, 0x31, 0x32, 0x1b,
|
||||||
|
0x1f, 0x36, 0x35, 0x1c, 0x34, 0x1d, 0x1e, 0x37,
|
||||||
|
0x1f, 0x36, 0x35, 0x1c, 0x34, 0x1d, 0x1e, 0x37,
|
||||||
|
0x33, 0x1a, 0x19, 0x30, 0x18, 0x31, 0x32, 0x1b,
|
||||||
|
0x32, 0x1b, 0x18, 0x31, 0x19, 0x30, 0x33, 0x1a,
|
||||||
|
0x1e, 0x37, 0x34, 0x1d, 0x35, 0x1c, 0x1f, 0x36,
|
||||||
|
0x31, 0x18, 0x1b, 0x32, 0x1a, 0x33, 0x30, 0x19,
|
||||||
|
0x1d, 0x34, 0x37, 0x1e, 0x36, 0x1f, 0x1c, 0x35,
|
||||||
|
0x1c, 0x35, 0x36, 0x1f, 0x37, 0x1e, 0x1d, 0x34,
|
||||||
|
0x30, 0x19, 0x1a, 0x33, 0x1b, 0x32, 0x31, 0x18
|
||||||
|
},
|
||||||
|
{ // Parities of third byte
|
||||||
|
0x3f, 0x0e, 0x0d, 0x3c, 0x0c, 0x3d, 0x3e, 0x0f,
|
||||||
|
0x0b, 0x3a, 0x39, 0x08, 0x38, 0x09, 0x0a, 0x3b,
|
||||||
|
0x0a, 0x3b, 0x38, 0x09, 0x39, 0x08, 0x0b, 0x3a,
|
||||||
|
0x3e, 0x0f, 0x0c, 0x3d, 0x0d, 0x3c, 0x3f, 0x0e,
|
||||||
|
0x09, 0x38, 0x3b, 0x0a, 0x3a, 0x0b, 0x08, 0x39,
|
||||||
|
0x3d, 0x0c, 0x0f, 0x3e, 0x0e, 0x3f, 0x3c, 0x0d,
|
||||||
|
0x3c, 0x0d, 0x0e, 0x3f, 0x0f, 0x3e, 0x3d, 0x0c,
|
||||||
|
0x08, 0x39, 0x3a, 0x0b, 0x3b, 0x0a, 0x09, 0x38,
|
||||||
|
0x08, 0x39, 0x3a, 0x0b, 0x3b, 0x0a, 0x09, 0x38,
|
||||||
|
0x3c, 0x0d, 0x0e, 0x3f, 0x0f, 0x3e, 0x3d, 0x0c,
|
||||||
|
0x3d, 0x0c, 0x0f, 0x3e, 0x0e, 0x3f, 0x3c, 0x0d,
|
||||||
|
0x09, 0x38, 0x3b, 0x0a, 0x3a, 0x0b, 0x08, 0x39,
|
||||||
|
0x3e, 0x0f, 0x0c, 0x3d, 0x0d, 0x3c, 0x3f, 0x0e,
|
||||||
|
0x0a, 0x3b, 0x38, 0x09, 0x39, 0x08, 0x0b, 0x3a,
|
||||||
|
0x0b, 0x3a, 0x39, 0x08, 0x38, 0x09, 0x0a, 0x3b,
|
||||||
|
0x3f, 0x0e, 0x0d, 0x3c, 0x0c, 0x3d, 0x3e, 0x0f,
|
||||||
|
0x1f, 0x2e, 0x2d, 0x1c, 0x2c, 0x1d, 0x1e, 0x2f,
|
||||||
|
0x2b, 0x1a, 0x19, 0x28, 0x18, 0x29, 0x2a, 0x1b,
|
||||||
|
0x2a, 0x1b, 0x18, 0x29, 0x19, 0x28, 0x2b, 0x1a,
|
||||||
|
0x1e, 0x2f, 0x2c, 0x1d, 0x2d, 0x1c, 0x1f, 0x2e,
|
||||||
|
0x29, 0x18, 0x1b, 0x2a, 0x1a, 0x2b, 0x28, 0x19,
|
||||||
|
0x1d, 0x2c, 0x2f, 0x1e, 0x2e, 0x1f, 0x1c, 0x2d,
|
||||||
|
0x1c, 0x2d, 0x2e, 0x1f, 0x2f, 0x1e, 0x1d, 0x2c,
|
||||||
|
0x28, 0x19, 0x1a, 0x2b, 0x1b, 0x2a, 0x29, 0x18,
|
||||||
|
0x28, 0x19, 0x1a, 0x2b, 0x1b, 0x2a, 0x29, 0x18,
|
||||||
|
0x1c, 0x2d, 0x2e, 0x1f, 0x2f, 0x1e, 0x1d, 0x2c,
|
||||||
|
0x1d, 0x2c, 0x2f, 0x1e, 0x2e, 0x1f, 0x1c, 0x2d,
|
||||||
|
0x29, 0x18, 0x1b, 0x2a, 0x1a, 0x2b, 0x28, 0x19,
|
||||||
|
0x1e, 0x2f, 0x2c, 0x1d, 0x2d, 0x1c, 0x1f, 0x2e,
|
||||||
|
0x2a, 0x1b, 0x18, 0x29, 0x19, 0x28, 0x2b, 0x1a,
|
||||||
|
0x2b, 0x1a, 0x19, 0x28, 0x18, 0x29, 0x2a, 0x1b,
|
||||||
|
0x1f, 0x2e, 0x2d, 0x1c, 0x2c, 0x1d, 0x1e, 0x2f
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
static const unsigned char hamming_24_18_decode_d1_d4[64] = {
|
||||||
|
0x00, 0x01, 0x00, 0x01, 0x02, 0x03, 0x02, 0x03,
|
||||||
|
0x04, 0x05, 0x04, 0x05, 0x06, 0x07, 0x06, 0x07,
|
||||||
|
0x08, 0x09, 0x08, 0x09, 0x0a, 0x0b, 0x0a, 0x0b,
|
||||||
|
0x0c, 0x0d, 0x0c, 0x0d, 0x0e, 0x0f, 0x0e, 0x0f,
|
||||||
|
0x00, 0x01, 0x00, 0x01, 0x02, 0x03, 0x02, 0x03,
|
||||||
|
0x04, 0x05, 0x04, 0x05, 0x06, 0x07, 0x06, 0x07,
|
||||||
|
0x08, 0x09, 0x08, 0x09, 0x0a, 0x0b, 0x0a, 0x0b,
|
||||||
|
0x0c, 0x0d, 0x0c, 0x0d, 0x0e, 0x0f, 0x0e, 0x0f
|
||||||
|
};
|
||||||
|
|
||||||
|
// Mapping from parity checks in hamming_24_18_parities to incorrect bit
|
||||||
|
// 0x80000000 - double bit error that can't be corrected
|
||||||
|
static const unsigned int hamming_24_18_decode_correct[64] = {
|
||||||
|
0x00000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x00000000, 0x00000000, 0x00000000, 0x00000001,
|
||||||
|
0x00000000, 0x00000002, 0x00000004, 0x00000008,
|
||||||
|
0x00000000, 0x00000010, 0x00000020, 0x00000040,
|
||||||
|
0x00000080, 0x00000100, 0x00000200, 0x00000400,
|
||||||
|
0x00000000, 0x00000800, 0x00001000, 0x00002000,
|
||||||
|
0x00004000, 0x00008000, 0x00010000, 0x00020000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000,
|
||||||
|
0x80000000, 0x80000000, 0x80000000, 0x80000000
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
870
keymap.h
870
keymap.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -28,304 +28,304 @@ static const QMap<QChar, char> keymapping[24] = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
{ // 1 Cyrillic G0 1 Serbian/Croatian
|
{ // 1 Cyrillic G0 1 Serbian/Croatian
|
||||||
{ 0x0427, 0x40 }, // Ч CYRILLIC CAPITAL LETTER CHE
|
{ QChar(0x0427), 0x40 }, // Ч CYRILLIC CAPITAL LETTER CHE
|
||||||
{ 0x0410, 0x41 }, // А CYRILLIC CAPITAL LETTER A
|
{ QChar(0x0410), 0x41 }, // А CYRILLIC CAPITAL LETTER A
|
||||||
{ 0x0411, 0x42 }, // Б CYRILLIC CAPITAL LETTER BE
|
{ QChar(0x0411), 0x42 }, // Б CYRILLIC CAPITAL LETTER BE
|
||||||
{ 0x0426, 0x43 }, // Ц CYRILLIC CAPITAL LETTER TSE
|
{ QChar(0x0426), 0x43 }, // Ц CYRILLIC CAPITAL LETTER TSE
|
||||||
{ 0x0414, 0x44 }, // Д CYRILLIC CAPITAL LETTER DE
|
{ QChar(0x0414), 0x44 }, // Д CYRILLIC CAPITAL LETTER DE
|
||||||
{ 0x0415, 0x45 }, // Е CYRILLIC CAPITAL LETTER IE
|
{ QChar(0x0415), 0x45 }, // Е CYRILLIC CAPITAL LETTER IE
|
||||||
{ 0x0424, 0x46 }, // Ф CYRILLIC CAPITAL LETTER EF
|
{ QChar(0x0424), 0x46 }, // Ф CYRILLIC CAPITAL LETTER EF
|
||||||
{ 0x0413, 0x47 }, // Г CYRILLIC CAPITAL LETTER GHE
|
{ QChar(0x0413), 0x47 }, // Г CYRILLIC CAPITAL LETTER GHE
|
||||||
{ 0x0425, 0x48 }, // Х CYRILLIC CAPITAL LETTER HA
|
{ QChar(0x0425), 0x48 }, // Х CYRILLIC CAPITAL LETTER HA
|
||||||
{ 0x0418, 0x49 }, // И CYRILLIC CAPITAL LETTER I
|
{ QChar(0x0418), 0x49 }, // И CYRILLIC CAPITAL LETTER I
|
||||||
{ 0x0408, 0x4a }, // Ј CYRILLIC CAPITAL LETTER JE
|
{ QChar(0x0408), 0x4a }, // Ј CYRILLIC CAPITAL LETTER JE
|
||||||
{ 0x041a, 0x4b }, // К CYRILLIC CAPITAL LETTER KA
|
{ QChar(0x041a), 0x4b }, // К CYRILLIC CAPITAL LETTER KA
|
||||||
{ 0x041b, 0x4c }, // Л CYRILLIC CAPITAL LETTER EL
|
{ QChar(0x041b), 0x4c }, // Л CYRILLIC CAPITAL LETTER EL
|
||||||
{ 0x041c, 0x4d }, // М CYRILLIC CAPITAL LETTER EM
|
{ QChar(0x041c), 0x4d }, // М CYRILLIC CAPITAL LETTER EM
|
||||||
{ 0x041d, 0x4e }, // Н CYRILLIC CAPITAL LETTER EN
|
{ QChar(0x041d), 0x4e }, // Н CYRILLIC CAPITAL LETTER EN
|
||||||
{ 0x041e, 0x4f }, // О CYRILLIC CAPITAL LETTER O
|
{ QChar(0x041e), 0x4f }, // О CYRILLIC CAPITAL LETTER O
|
||||||
{ 0x041f, 0x50 }, // П CYRILLIC CAPITAL LETTER PE
|
{ QChar(0x041f), 0x50 }, // П CYRILLIC CAPITAL LETTER PE
|
||||||
{ 0x040c, 0x51 }, // Ќ CYRILLIC CAPITAL LETTER KJE
|
{ QChar(0x040c), 0x51 }, // Ќ CYRILLIC CAPITAL LETTER KJE
|
||||||
{ 0x0420, 0x52 }, // Р CYRILLIC CAPITAL LETTER ER
|
{ QChar(0x0420), 0x52 }, // Р CYRILLIC CAPITAL LETTER ER
|
||||||
{ 0x0421, 0x53 }, // С CYRILLIC CAPITAL LETTER ES
|
{ QChar(0x0421), 0x53 }, // С CYRILLIC CAPITAL LETTER ES
|
||||||
{ 0x0422, 0x54 }, // Т CYRILLIC CAPITAL LETTER TE
|
{ QChar(0x0422), 0x54 }, // Т CYRILLIC CAPITAL LETTER TE
|
||||||
{ 0x0423, 0x55 }, // У CYRILLIC CAPITAL LETTER U
|
{ QChar(0x0423), 0x55 }, // У CYRILLIC CAPITAL LETTER U
|
||||||
{ 0x0412, 0x56 }, // В CYRILLIC CAPITAL LETTER VE
|
{ QChar(0x0412), 0x56 }, // В CYRILLIC CAPITAL LETTER VE
|
||||||
{ 0x0403, 0x57 }, // Ѓ CYRILLIC CAPITAL LETTER GJE
|
{ QChar(0x0403), 0x57 }, // Ѓ CYRILLIC CAPITAL LETTER GJE
|
||||||
{ 0x0409, 0x58 }, // Љ CYRILLIC CAPITAL LETTER LJE
|
{ QChar(0x0409), 0x58 }, // Љ CYRILLIC CAPITAL LETTER LJE
|
||||||
{ 0x040a, 0x59 }, // Њ CYRILLIC CAPITAL LETTER NJE
|
{ QChar(0x040a), 0x59 }, // Њ CYRILLIC CAPITAL LETTER NJE
|
||||||
{ 0x0417, 0x5a }, // З CYRILLIC CAPITAL LETTER ZE
|
{ QChar(0x0417), 0x5a }, // З CYRILLIC CAPITAL LETTER ZE
|
||||||
{ 0x040b, 0x5b }, // Ћ CYRILLIC CAPITAL LETTER TSHE
|
{ QChar(0x040b), 0x5b }, // Ћ CYRILLIC CAPITAL LETTER TSHE
|
||||||
{ 0x0416, 0x5c }, // Ж CYRILLIC CAPITAL LETTER ZHE
|
{ QChar(0x0416), 0x5c }, // Ж CYRILLIC CAPITAL LETTER ZHE
|
||||||
{ 0x0402, 0x5d }, // Ђ CYRILLIC CAPITAL LETTER DJE
|
{ QChar(0x0402), 0x5d }, // Ђ CYRILLIC CAPITAL LETTER DJE
|
||||||
{ 0x0428, 0x5e }, // Ш CYRILLIC CAPITAL LETTER SHA
|
{ QChar(0x0428), 0x5e }, // Ш CYRILLIC CAPITAL LETTER SHA
|
||||||
{ 0x040f, 0x5f }, // Џ CYRILLIC CAPITAL LETTER DZHE
|
{ QChar(0x040f), 0x5f }, // Џ CYRILLIC CAPITAL LETTER DZHE
|
||||||
{ 0x0447, 0x60 }, // ч CYRILLIC SMALL LETTER CHE
|
{ QChar(0x0447), 0x60 }, // ч CYRILLIC SMALL LETTER CHE
|
||||||
{ 0x0430, 0x61 }, // а CYRILLIC SMALL LETTER A
|
{ QChar(0x0430), 0x61 }, // а CYRILLIC SMALL LETTER A
|
||||||
{ 0x0431, 0x62 }, // б CYRILLIC SMALL LETTER BE
|
{ QChar(0x0431), 0x62 }, // б CYRILLIC SMALL LETTER BE
|
||||||
{ 0x0446, 0x63 }, // ц CYRILLIC SMALL LETTER TSE
|
{ QChar(0x0446), 0x63 }, // ц CYRILLIC SMALL LETTER TSE
|
||||||
{ 0x0434, 0x64 }, // д CYRILLIC SMALL LETTER DE
|
{ QChar(0x0434), 0x64 }, // д CYRILLIC SMALL LETTER DE
|
||||||
{ 0x0435, 0x65 }, // е CYRILLIC SMALL LETTER IE
|
{ QChar(0x0435), 0x65 }, // е CYRILLIC SMALL LETTER IE
|
||||||
{ 0x0444, 0x66 }, // ф CYRILLIC SMALL LETTER EF
|
{ QChar(0x0444), 0x66 }, // ф CYRILLIC SMALL LETTER EF
|
||||||
{ 0x0433, 0x67 }, // г CYRILLIC SMALL LETTER GHE
|
{ QChar(0x0433), 0x67 }, // г CYRILLIC SMALL LETTER GHE
|
||||||
{ 0x0445, 0x68 }, // х CYRILLIC SMALL LETTER HA
|
{ QChar(0x0445), 0x68 }, // х CYRILLIC SMALL LETTER HA
|
||||||
{ 0x0438, 0x69 }, // и CYRILLIC SMALL LETTER I
|
{ QChar(0x0438), 0x69 }, // и CYRILLIC SMALL LETTER I
|
||||||
{ 0x0458, 0x6a }, // ј CYRILLIC SMALL LETTER JE
|
{ QChar(0x0458), 0x6a }, // ј CYRILLIC SMALL LETTER JE
|
||||||
{ 0x043a, 0x6b }, // к CYRILLIC SMALL LETTER KA
|
{ QChar(0x043a), 0x6b }, // к CYRILLIC SMALL LETTER KA
|
||||||
{ 0x043b, 0x6c }, // л CYRILLIC SMALL LETTER EL
|
{ QChar(0x043b), 0x6c }, // л CYRILLIC SMALL LETTER EL
|
||||||
{ 0x043c, 0x6d }, // м CYRILLIC SMALL LETTER EM
|
{ QChar(0x043c), 0x6d }, // м CYRILLIC SMALL LETTER EM
|
||||||
{ 0x043d, 0x6e }, // н CYRILLIC SMALL LETTER EN
|
{ QChar(0x043d), 0x6e }, // н CYRILLIC SMALL LETTER EN
|
||||||
{ 0x043e, 0x6f }, // о CYRILLIC SMALL LETTER O
|
{ QChar(0x043e), 0x6f }, // о CYRILLIC SMALL LETTER O
|
||||||
{ 0x043f, 0x70 }, // п CYRILLIC SMALL LETTER PE
|
{ QChar(0x043f), 0x70 }, // п CYRILLIC SMALL LETTER PE
|
||||||
{ 0x045c, 0x71 }, // ќ CYRILLIC SMALL LETTER KJE
|
{ QChar(0x045c), 0x71 }, // ќ CYRILLIC SMALL LETTER KJE
|
||||||
{ 0x0440, 0x72 }, // р CYRILLIC SMALL LETTER ER
|
{ QChar(0x0440), 0x72 }, // р CYRILLIC SMALL LETTER ER
|
||||||
{ 0x0441, 0x73 }, // с CYRILLIC SMALL LETTER ES
|
{ QChar(0x0441), 0x73 }, // с CYRILLIC SMALL LETTER ES
|
||||||
{ 0x0442, 0x74 }, // т CYRILLIC SMALL LETTER TE
|
{ QChar(0x0442), 0x74 }, // т CYRILLIC SMALL LETTER TE
|
||||||
{ 0x0443, 0x75 }, // у CYRILLIC SMALL LETTER U
|
{ QChar(0x0443), 0x75 }, // у CYRILLIC SMALL LETTER U
|
||||||
{ 0x0432, 0x76 }, // в CYRILLIC SMALL LETTER VE
|
{ QChar(0x0432), 0x76 }, // в CYRILLIC SMALL LETTER VE
|
||||||
{ 0x0453, 0x77 }, // ѓ CYRILLIC SMALL LETTER GJE
|
{ QChar(0x0453), 0x77 }, // ѓ CYRILLIC SMALL LETTER GJE
|
||||||
{ 0x0459, 0x78 }, // љ CYRILLIC SMALL LETTER LJE
|
{ QChar(0x0459), 0x78 }, // љ CYRILLIC SMALL LETTER LJE
|
||||||
{ 0x045a, 0x79 }, // њ CYRILLIC SMALL LETTER NJE
|
{ QChar(0x045a), 0x79 }, // њ CYRILLIC SMALL LETTER NJE
|
||||||
{ 0x0437, 0x7a }, // з CYRILLIC SMALL LETTER ZE
|
{ QChar(0x0437), 0x7a }, // з CYRILLIC SMALL LETTER ZE
|
||||||
{ 0x045b, 0x7b }, // ћ CYRILLIC SMALL LETTER TSHE
|
{ QChar(0x045b), 0x7b }, // ћ CYRILLIC SMALL LETTER TSHE
|
||||||
{ 0x0436, 0x7c }, // ж CYRILLIC SMALL LETTER ZHE
|
{ QChar(0x0436), 0x7c }, // ж CYRILLIC SMALL LETTER ZHE
|
||||||
{ 0x0452, 0x7d }, // ђ CYRILLIC SMALL LETTER DJE
|
{ QChar(0x0452), 0x7d }, // ђ CYRILLIC SMALL LETTER DJE
|
||||||
{ 0x0448, 0x7e } // ш CYRILLIC SMALL LETTER SHA
|
{ QChar(0x0448), 0x7e } // ш CYRILLIC SMALL LETTER SHA
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 2 Cyrillic G0 2 Russian/Bulgarian
|
{ // 2 Cyrillic G0 2 Russian/Bulgarian
|
||||||
{ 0x044b, 0x26 }, // ы CYRILLIC SMALL LETTER YERU
|
{ QChar(0x044b), 0x26 }, // ы CYRILLIC SMALL LETTER YERU
|
||||||
{ 0x042e, 0x40 }, // Ю CYRILLIC CAPITAL LETTER YU
|
{ QChar(0x042e), 0x40 }, // Ю CYRILLIC CAPITAL LETTER YU
|
||||||
{ 0x0410, 0x41 }, // А CYRILLIC CAPITAL LETTER A
|
{ QChar(0x0410), 0x41 }, // А CYRILLIC CAPITAL LETTER A
|
||||||
{ 0x0411, 0x42 }, // Б CYRILLIC CAPITAL LETTER BE
|
{ QChar(0x0411), 0x42 }, // Б CYRILLIC CAPITAL LETTER BE
|
||||||
{ 0x0426, 0x43 }, // Ц CYRILLIC CAPITAL LETTER TSE
|
{ QChar(0x0426), 0x43 }, // Ц CYRILLIC CAPITAL LETTER TSE
|
||||||
{ 0x0414, 0x44 }, // Д CYRILLIC CAPITAL LETTER DE
|
{ QChar(0x0414), 0x44 }, // Д CYRILLIC CAPITAL LETTER DE
|
||||||
{ 0x0415, 0x45 }, // Е CYRILLIC CAPITAL LETTER IE
|
{ QChar(0x0415), 0x45 }, // Е CYRILLIC CAPITAL LETTER IE
|
||||||
{ 0x0424, 0x46 }, // Ф CYRILLIC CAPITAL LETTER EF
|
{ QChar(0x0424), 0x46 }, // Ф CYRILLIC CAPITAL LETTER EF
|
||||||
{ 0x0413, 0x47 }, // Г CYRILLIC CAPITAL LETTER GHE
|
{ QChar(0x0413), 0x47 }, // Г CYRILLIC CAPITAL LETTER GHE
|
||||||
{ 0x0425, 0x48 }, // Х CYRILLIC CAPITAL LETTER HA
|
{ QChar(0x0425), 0x48 }, // Х CYRILLIC CAPITAL LETTER HA
|
||||||
{ 0x0418, 0x49 }, // И CYRILLIC CAPITAL LETTER I
|
{ QChar(0x0418), 0x49 }, // И CYRILLIC CAPITAL LETTER I
|
||||||
{ 0x0419, 0x4a }, // Й CYRILLIC CAPITAL LETTER SHORT I
|
{ QChar(0x0419), 0x4a }, // Й CYRILLIC CAPITAL LETTER SHORT I
|
||||||
{ 0x041a, 0x4b }, // К CYRILLIC CAPITAL LETTER KA
|
{ QChar(0x041a), 0x4b }, // К CYRILLIC CAPITAL LETTER KA
|
||||||
{ 0x041b, 0x4c }, // Л CYRILLIC CAPITAL LETTER EL
|
{ QChar(0x041b), 0x4c }, // Л CYRILLIC CAPITAL LETTER EL
|
||||||
{ 0x041c, 0x4d }, // М CYRILLIC CAPITAL LETTER EM
|
{ QChar(0x041c), 0x4d }, // М CYRILLIC CAPITAL LETTER EM
|
||||||
{ 0x041d, 0x4e }, // Н CYRILLIC CAPITAL LETTER EN
|
{ QChar(0x041d), 0x4e }, // Н CYRILLIC CAPITAL LETTER EN
|
||||||
{ 0x041e, 0x4f }, // О CYRILLIC CAPITAL LETTER O
|
{ QChar(0x041e), 0x4f }, // О CYRILLIC CAPITAL LETTER O
|
||||||
{ 0x041f, 0x50 }, // П CYRILLIC CAPITAL LETTER PE
|
{ QChar(0x041f), 0x50 }, // П CYRILLIC CAPITAL LETTER PE
|
||||||
{ 0x042f, 0x51 }, // Я CYRILLIC CAPITAL LETTER YA
|
{ QChar(0x042f), 0x51 }, // Я CYRILLIC CAPITAL LETTER YA
|
||||||
{ 0x0420, 0x52 }, // Р CYRILLIC CAPITAL LETTER ER
|
{ QChar(0x0420), 0x52 }, // Р CYRILLIC CAPITAL LETTER ER
|
||||||
{ 0x0421, 0x53 }, // С CYRILLIC CAPITAL LETTER ES
|
{ QChar(0x0421), 0x53 }, // С CYRILLIC CAPITAL LETTER ES
|
||||||
{ 0x0422, 0x54 }, // Т CYRILLIC CAPITAL LETTER TE
|
{ QChar(0x0422), 0x54 }, // Т CYRILLIC CAPITAL LETTER TE
|
||||||
{ 0x0423, 0x55 }, // У CYRILLIC CAPITAL LETTER U
|
{ QChar(0x0423), 0x55 }, // У CYRILLIC CAPITAL LETTER U
|
||||||
{ 0x0416, 0x56 }, // Ж CYRILLIC CAPITAL LETTER ZHE
|
{ QChar(0x0416), 0x56 }, // Ж CYRILLIC CAPITAL LETTER ZHE
|
||||||
{ 0x0412, 0x57 }, // В CYRILLIC CAPITAL LETTER VE
|
{ QChar(0x0412), 0x57 }, // В CYRILLIC CAPITAL LETTER VE
|
||||||
{ 0x042c, 0x58 }, // Ь CYRILLIC CAPITAL LETTER SOFT SIGN
|
{ QChar(0x042c), 0x58 }, // Ь CYRILLIC CAPITAL LETTER SOFT SIGN
|
||||||
{ 0x042a, 0x59 }, // Ъ CYRILLIC CAPITAL LETTER HARD SIGN
|
{ QChar(0x042a), 0x59 }, // Ъ CYRILLIC CAPITAL LETTER HARD SIGN
|
||||||
{ 0x0417, 0x5a }, // З CYRILLIC CAPITAL LETTER ZE
|
{ QChar(0x0417), 0x5a }, // З CYRILLIC CAPITAL LETTER ZE
|
||||||
{ 0x0428, 0x5b }, // Ш CYRILLIC CAPITAL LETTER SHA
|
{ QChar(0x0428), 0x5b }, // Ш CYRILLIC CAPITAL LETTER SHA
|
||||||
{ 0x042d, 0x5c }, // Э CYRILLIC CAPITAL LETTER E
|
{ QChar(0x042d), 0x5c }, // Э CYRILLIC CAPITAL LETTER E
|
||||||
{ 0x0429, 0x5d }, // Щ CYRILLIC CAPITAL LETTER SHCHA
|
{ QChar(0x0429), 0x5d }, // Щ CYRILLIC CAPITAL LETTER SHCHA
|
||||||
{ 0x0427, 0x5e }, // Ч CYRILLIC CAPITAL LETTER CHE
|
{ QChar(0x0427), 0x5e }, // Ч CYRILLIC CAPITAL LETTER CHE
|
||||||
{ 0x042b, 0x5f }, // Ы CYRILLIC CAPITAL LETTER YERU
|
{ QChar(0x042b), 0x5f }, // Ы CYRILLIC CAPITAL LETTER YERU
|
||||||
{ 0x044e, 0x60 }, // ю CYRILLIC SMALL LETTER YU
|
{ QChar(0x044e), 0x60 }, // ю CYRILLIC SMALL LETTER YU
|
||||||
{ 0x0430, 0x61 }, // а CYRILLIC SMALL LETTER A
|
{ QChar(0x0430), 0x61 }, // а CYRILLIC SMALL LETTER A
|
||||||
{ 0x0431, 0x62 }, // б CYRILLIC SMALL LETTER BE
|
{ QChar(0x0431), 0x62 }, // б CYRILLIC SMALL LETTER BE
|
||||||
{ 0x0446, 0x63 }, // ц CYRILLIC SMALL LETTER TSE
|
{ QChar(0x0446), 0x63 }, // ц CYRILLIC SMALL LETTER TSE
|
||||||
{ 0x0434, 0x64 }, // д CYRILLIC SMALL LETTER DE
|
{ QChar(0x0434), 0x64 }, // д CYRILLIC SMALL LETTER DE
|
||||||
{ 0x0435, 0x65 }, // е CYRILLIC SMALL LETTER IE
|
{ QChar(0x0435), 0x65 }, // е CYRILLIC SMALL LETTER IE
|
||||||
{ 0x0444, 0x66 }, // ф CYRILLIC SMALL LETTER EF
|
{ QChar(0x0444), 0x66 }, // ф CYRILLIC SMALL LETTER EF
|
||||||
{ 0x0433, 0x67 }, // г CYRILLIC SMALL LETTER GHE
|
{ QChar(0x0433), 0x67 }, // г CYRILLIC SMALL LETTER GHE
|
||||||
{ 0x0445, 0x68 }, // х CYRILLIC SMALL LETTER HA
|
{ QChar(0x0445), 0x68 }, // х CYRILLIC SMALL LETTER HA
|
||||||
{ 0x0438, 0x69 }, // и CYRILLIC SMALL LETTER I
|
{ QChar(0x0438), 0x69 }, // и CYRILLIC SMALL LETTER I
|
||||||
{ 0x0439, 0x6a }, // й CYRILLIC SMALL LETTER SHORT I
|
{ QChar(0x0439), 0x6a }, // й CYRILLIC SMALL LETTER SHORT I
|
||||||
{ 0x043a, 0x6b }, // к CYRILLIC SMALL LETTER KA
|
{ QChar(0x043a), 0x6b }, // к CYRILLIC SMALL LETTER KA
|
||||||
{ 0x043b, 0x6c }, // л CYRILLIC SMALL LETTER EL
|
{ QChar(0x043b), 0x6c }, // л CYRILLIC SMALL LETTER EL
|
||||||
{ 0x043c, 0x6d }, // м CYRILLIC SMALL LETTER EM
|
{ QChar(0x043c), 0x6d }, // м CYRILLIC SMALL LETTER EM
|
||||||
{ 0x043d, 0x6e }, // н CYRILLIC SMALL LETTER EN
|
{ QChar(0x043d), 0x6e }, // н CYRILLIC SMALL LETTER EN
|
||||||
{ 0x043e, 0x6f }, // о CYRILLIC SMALL LETTER O
|
{ QChar(0x043e), 0x6f }, // о CYRILLIC SMALL LETTER O
|
||||||
{ 0x043f, 0x70 }, // п CYRILLIC SMALL LETTER PE
|
{ QChar(0x043f), 0x70 }, // п CYRILLIC SMALL LETTER PE
|
||||||
{ 0x044f, 0x71 }, // я CYRILLIC SMALL LETTER YA
|
{ QChar(0x044f), 0x71 }, // я CYRILLIC SMALL LETTER YA
|
||||||
{ 0x0440, 0x72 }, // р CYRILLIC SMALL LETTER ER
|
{ QChar(0x0440), 0x72 }, // р CYRILLIC SMALL LETTER ER
|
||||||
{ 0x0441, 0x73 }, // с CYRILLIC SMALL LETTER ES
|
{ QChar(0x0441), 0x73 }, // с CYRILLIC SMALL LETTER ES
|
||||||
{ 0x0442, 0x74 }, // т CYRILLIC SMALL LETTER TE
|
{ QChar(0x0442), 0x74 }, // т CYRILLIC SMALL LETTER TE
|
||||||
{ 0x0443, 0x75 }, // у CYRILLIC SMALL LETTER U
|
{ QChar(0x0443), 0x75 }, // у CYRILLIC SMALL LETTER U
|
||||||
{ 0x0436, 0x76 }, // ж CYRILLIC SMALL LETTER ZHE
|
{ QChar(0x0436), 0x76 }, // ж CYRILLIC SMALL LETTER ZHE
|
||||||
{ 0x0432, 0x77 }, // в CYRILLIC SMALL LETTER VE
|
{ QChar(0x0432), 0x77 }, // в CYRILLIC SMALL LETTER VE
|
||||||
{ 0x044c, 0x78 }, // ь CYRILLIC SMALL LETTER SOFT SIGN
|
{ QChar(0x044c), 0x78 }, // ь CYRILLIC SMALL LETTER SOFT SIGN
|
||||||
{ 0x044a, 0x79 }, // ъ CYRILLIC SMALL LETTER HARD SIGN
|
{ QChar(0x044a), 0x79 }, // ъ CYRILLIC SMALL LETTER HARD SIGN
|
||||||
{ 0x0437, 0x7a }, // з CYRILLIC SMALL LETTER ZE
|
{ QChar(0x0437), 0x7a }, // з CYRILLIC SMALL LETTER ZE
|
||||||
{ 0x0448, 0x7b }, // ш CYRILLIC SMALL LETTER SHA
|
{ QChar(0x0448), 0x7b }, // ш CYRILLIC SMALL LETTER SHA
|
||||||
{ 0x044d, 0x7c }, // э CYRILLIC SMALL LETTER E
|
{ QChar(0x044d), 0x7c }, // э CYRILLIC SMALL LETTER E
|
||||||
{ 0x0449, 0x7d }, // щ CYRILLIC SMALL LETTER SHCHA
|
{ QChar(0x0449), 0x7d }, // щ CYRILLIC SMALL LETTER SHCHA
|
||||||
{ 0x0447, 0x7e } // ч CYRILLIC SMALL LETTER CHE
|
{ QChar(0x0447), 0x7e } // ч CYRILLIC SMALL LETTER CHE
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 3 Cyrillic G0 3 Ukranian
|
{ // 3 Cyrillic G0 3 Ukranian
|
||||||
{ 0x0457, 0x26 }, // ї CYRILLIC SMALL LETTER YI
|
{ QChar(0x0457), 0x26 }, // ї CYRILLIC SMALL LETTER YI
|
||||||
{ 0x042e, 0x40 }, // Ю CYRILLIC CAPITAL LETTER YU
|
{ QChar(0x042e), 0x40 }, // Ю CYRILLIC CAPITAL LETTER YU
|
||||||
{ 0x0410, 0x41 }, // А CYRILLIC CAPITAL LETTER A
|
{ QChar(0x0410), 0x41 }, // А CYRILLIC CAPITAL LETTER A
|
||||||
{ 0x0411, 0x42 }, // Б CYRILLIC CAPITAL LETTER BE
|
{ QChar(0x0411), 0x42 }, // Б CYRILLIC CAPITAL LETTER BE
|
||||||
{ 0x0426, 0x43 }, // Ц CYRILLIC CAPITAL LETTER TSE
|
{ QChar(0x0426), 0x43 }, // Ц CYRILLIC CAPITAL LETTER TSE
|
||||||
{ 0x0414, 0x44 }, // Д CYRILLIC CAPITAL LETTER DE
|
{ QChar(0x0414), 0x44 }, // Д CYRILLIC CAPITAL LETTER DE
|
||||||
{ 0x0415, 0x45 }, // Е CYRILLIC CAPITAL LETTER IE
|
{ QChar(0x0415), 0x45 }, // Е CYRILLIC CAPITAL LETTER IE
|
||||||
{ 0x0424, 0x46 }, // Ф CYRILLIC CAPITAL LETTER EF
|
{ QChar(0x0424), 0x46 }, // Ф CYRILLIC CAPITAL LETTER EF
|
||||||
{ 0x0413, 0x47 }, // Г CYRILLIC CAPITAL LETTER GHE
|
{ QChar(0x0413), 0x47 }, // Г CYRILLIC CAPITAL LETTER GHE
|
||||||
{ 0x0425, 0x48 }, // Х CYRILLIC CAPITAL LETTER HA
|
{ QChar(0x0425), 0x48 }, // Х CYRILLIC CAPITAL LETTER HA
|
||||||
{ 0x0418, 0x49 }, // И CYRILLIC CAPITAL LETTER I
|
{ QChar(0x0418), 0x49 }, // И CYRILLIC CAPITAL LETTER I
|
||||||
{ 0x0419, 0x4a }, // Й CYRILLIC CAPITAL LETTER SHORT I
|
{ QChar(0x0419), 0x4a }, // Й CYRILLIC CAPITAL LETTER SHORT I
|
||||||
{ 0x041a, 0x4b }, // К CYRILLIC CAPITAL LETTER KA
|
{ QChar(0x041a), 0x4b }, // К CYRILLIC CAPITAL LETTER KA
|
||||||
{ 0x041b, 0x4c }, // Л CYRILLIC CAPITAL LETTER EL
|
{ QChar(0x041b), 0x4c }, // Л CYRILLIC CAPITAL LETTER EL
|
||||||
{ 0x041c, 0x4d }, // М CYRILLIC CAPITAL LETTER EM
|
{ QChar(0x041c), 0x4d }, // М CYRILLIC CAPITAL LETTER EM
|
||||||
{ 0x041d, 0x4e }, // Н CYRILLIC CAPITAL LETTER EN
|
{ QChar(0x041d), 0x4e }, // Н CYRILLIC CAPITAL LETTER EN
|
||||||
{ 0x041e, 0x4f }, // О CYRILLIC CAPITAL LETTER O
|
{ QChar(0x041e), 0x4f }, // О CYRILLIC CAPITAL LETTER O
|
||||||
{ 0x041f, 0x50 }, // П CYRILLIC CAPITAL LETTER PE
|
{ QChar(0x041f), 0x50 }, // П CYRILLIC CAPITAL LETTER PE
|
||||||
{ 0x042f, 0x51 }, // Я CYRILLIC CAPITAL LETTER YA
|
{ QChar(0x042f), 0x51 }, // Я CYRILLIC CAPITAL LETTER YA
|
||||||
{ 0x0420, 0x52 }, // Р CYRILLIC CAPITAL LETTER ER
|
{ QChar(0x0420), 0x52 }, // Р CYRILLIC CAPITAL LETTER ER
|
||||||
{ 0x0421, 0x53 }, // С CYRILLIC CAPITAL LETTER ES
|
{ QChar(0x0421), 0x53 }, // С CYRILLIC CAPITAL LETTER ES
|
||||||
{ 0x0422, 0x54 }, // Т CYRILLIC CAPITAL LETTER TE
|
{ QChar(0x0422), 0x54 }, // Т CYRILLIC CAPITAL LETTER TE
|
||||||
{ 0x0423, 0x55 }, // У CYRILLIC CAPITAL LETTER U
|
{ QChar(0x0423), 0x55 }, // У CYRILLIC CAPITAL LETTER U
|
||||||
{ 0x0416, 0x56 }, // Ж CYRILLIC CAPITAL LETTER ZHE
|
{ QChar(0x0416), 0x56 }, // Ж CYRILLIC CAPITAL LETTER ZHE
|
||||||
{ 0x0412, 0x57 }, // В CYRILLIC CAPITAL LETTER VE
|
{ QChar(0x0412), 0x57 }, // В CYRILLIC CAPITAL LETTER VE
|
||||||
{ 0x042c, 0x58 }, // Ь CYRILLIC CAPITAL LETTER SOFT SIGN
|
{ QChar(0x042c), 0x58 }, // Ь CYRILLIC CAPITAL LETTER SOFT SIGN
|
||||||
{ 0x0406, 0x59 }, // І CYRILLIC CAPITAL LETTER BYELORUSSIAN-UKRAINIAN I
|
{ QChar(0x0406), 0x59 }, // І CYRILLIC CAPITAL LETTER BYELORUSSIAN-UKRAINIAN I
|
||||||
{ 0x0417, 0x5a }, // З CYRILLIC CAPITAL LETTER ZE
|
{ QChar(0x0417), 0x5a }, // З CYRILLIC CAPITAL LETTER ZE
|
||||||
{ 0x0428, 0x5b }, // Ш CYRILLIC CAPITAL LETTER SHA
|
{ QChar(0x0428), 0x5b }, // Ш CYRILLIC CAPITAL LETTER SHA
|
||||||
{ 0x0404, 0x5c }, // Є CYRILLIC CAPITAL LETTER UKRAINIAN IE
|
{ QChar(0x0404), 0x5c }, // Є CYRILLIC CAPITAL LETTER UKRAINIAN IE
|
||||||
{ 0x0429, 0x5d }, // Щ CYRILLIC CAPITAL LETTER SHCHA
|
{ QChar(0x0429), 0x5d }, // Щ CYRILLIC CAPITAL LETTER SHCHA
|
||||||
{ 0x0427, 0x5e }, // Ч CYRILLIC CAPITAL LETTER CHE
|
{ QChar(0x0427), 0x5e }, // Ч CYRILLIC CAPITAL LETTER CHE
|
||||||
{ 0x0407, 0x5f }, // Ї CYRILLIC CAPITAL LETTER YI
|
{ QChar(0x0407), 0x5f }, // Ї CYRILLIC CAPITAL LETTER YI
|
||||||
{ 0x044e, 0x60 }, // ю CYRILLIC SMALL LETTER YU
|
{ QChar(0x044e), 0x60 }, // ю CYRILLIC SMALL LETTER YU
|
||||||
{ 0x0430, 0x61 }, // а CYRILLIC SMALL LETTER A
|
{ QChar(0x0430), 0x61 }, // а CYRILLIC SMALL LETTER A
|
||||||
{ 0x0431, 0x62 }, // б CYRILLIC SMALL LETTER BE
|
{ QChar(0x0431), 0x62 }, // б CYRILLIC SMALL LETTER BE
|
||||||
{ 0x0446, 0x63 }, // ц CYRILLIC SMALL LETTER TSE
|
{ QChar(0x0446), 0x63 }, // ц CYRILLIC SMALL LETTER TSE
|
||||||
{ 0x0434, 0x64 }, // д CYRILLIC SMALL LETTER DE
|
{ QChar(0x0434), 0x64 }, // д CYRILLIC SMALL LETTER DE
|
||||||
{ 0x0435, 0x65 }, // е CYRILLIC SMALL LETTER IE
|
{ QChar(0x0435), 0x65 }, // е CYRILLIC SMALL LETTER IE
|
||||||
{ 0x0444, 0x66 }, // ф CYRILLIC SMALL LETTER EF
|
{ QChar(0x0444), 0x66 }, // ф CYRILLIC SMALL LETTER EF
|
||||||
{ 0x0433, 0x67 }, // г CYRILLIC SMALL LETTER GHE
|
{ QChar(0x0433), 0x67 }, // г CYRILLIC SMALL LETTER GHE
|
||||||
{ 0x0445, 0x68 }, // х CYRILLIC SMALL LETTER HA
|
{ QChar(0x0445), 0x68 }, // х CYRILLIC SMALL LETTER HA
|
||||||
{ 0x0438, 0x69 }, // и CYRILLIC SMALL LETTER I
|
{ QChar(0x0438), 0x69 }, // и CYRILLIC SMALL LETTER I
|
||||||
{ 0x0439, 0x6a }, // й CYRILLIC SMALL LETTER SHORT I
|
{ QChar(0x0439), 0x6a }, // й CYRILLIC SMALL LETTER SHORT I
|
||||||
{ 0x043a, 0x6b }, // к CYRILLIC SMALL LETTER KA
|
{ QChar(0x043a), 0x6b }, // к CYRILLIC SMALL LETTER KA
|
||||||
{ 0x043b, 0x6c }, // л CYRILLIC SMALL LETTER EL
|
{ QChar(0x043b), 0x6c }, // л CYRILLIC SMALL LETTER EL
|
||||||
{ 0x043c, 0x6d }, // м CYRILLIC SMALL LETTER EM
|
{ QChar(0x043c), 0x6d }, // м CYRILLIC SMALL LETTER EM
|
||||||
{ 0x043d, 0x6e }, // н CYRILLIC SMALL LETTER EN
|
{ QChar(0x043d), 0x6e }, // н CYRILLIC SMALL LETTER EN
|
||||||
{ 0x043e, 0x6f }, // о CYRILLIC SMALL LETTER O
|
{ QChar(0x043e), 0x6f }, // о CYRILLIC SMALL LETTER O
|
||||||
{ 0x043f, 0x70 }, // п CYRILLIC SMALL LETTER PE
|
{ QChar(0x043f), 0x70 }, // п CYRILLIC SMALL LETTER PE
|
||||||
{ 0x044f, 0x71 }, // я CYRILLIC SMALL LETTER YA
|
{ QChar(0x044f), 0x71 }, // я CYRILLIC SMALL LETTER YA
|
||||||
{ 0x0440, 0x72 }, // р CYRILLIC SMALL LETTER ER
|
{ QChar(0x0440), 0x72 }, // р CYRILLIC SMALL LETTER ER
|
||||||
{ 0x0441, 0x73 }, // с CYRILLIC SMALL LETTER ES
|
{ QChar(0x0441), 0x73 }, // с CYRILLIC SMALL LETTER ES
|
||||||
{ 0x0442, 0x74 }, // т CYRILLIC SMALL LETTER TE
|
{ QChar(0x0442), 0x74 }, // т CYRILLIC SMALL LETTER TE
|
||||||
{ 0x0443, 0x75 }, // у CYRILLIC SMALL LETTER U
|
{ QChar(0x0443), 0x75 }, // у CYRILLIC SMALL LETTER U
|
||||||
{ 0x0436, 0x76 }, // ж CYRILLIC SMALL LETTER ZHE
|
{ QChar(0x0436), 0x76 }, // ж CYRILLIC SMALL LETTER ZHE
|
||||||
{ 0x0432, 0x77 }, // в CYRILLIC SMALL LETTER VE
|
{ QChar(0x0432), 0x77 }, // в CYRILLIC SMALL LETTER VE
|
||||||
{ 0x044c, 0x78 }, // ь CYRILLIC SMALL LETTER SOFT SIGN
|
{ QChar(0x044c), 0x78 }, // ь CYRILLIC SMALL LETTER SOFT SIGN
|
||||||
{ 0x0456, 0x79 }, // і CYRILLIC SMALL LETTER BYELORUSSIAN-UKRAINIAN I
|
{ QChar(0x0456), 0x79 }, // і CYRILLIC SMALL LETTER BYELORUSSIAN-UKRAINIAN I
|
||||||
{ 0x0437, 0x7a }, // з CYRILLIC SMALL LETTER ZE
|
{ QChar(0x0437), 0x7a }, // з CYRILLIC SMALL LETTER ZE
|
||||||
{ 0x0448, 0x7b }, // ш CYRILLIC SMALL LETTER SHA
|
{ QChar(0x0448), 0x7b }, // ш CYRILLIC SMALL LETTER SHA
|
||||||
{ 0x0454, 0x7c }, // є CYRILLIC SMALL LETTER UKRAINIAN IE
|
{ QChar(0x0454), 0x7c }, // є CYRILLIC SMALL LETTER UKRAINIAN IE
|
||||||
{ 0x0449, 0x7d }, // щ CYRILLIC SMALL LETTER SHCHA
|
{ QChar(0x0449), 0x7d }, // щ CYRILLIC SMALL LETTER SHCHA
|
||||||
{ 0x0447, 0x7e } // ч CYRILLIC SMALL LETTER CHE
|
{ QChar(0x0447), 0x7e } // ч CYRILLIC SMALL LETTER CHE
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 4 Greek
|
{ // 4 Greek
|
||||||
{ 0x0390, 0x40 }, // ΐ GREEK SMALL LETTER IOTA WITH DIALYTIKA AND TONOS
|
{ QChar(0x0390), 0x40 }, // ΐ GREEK SMALL LETTER IOTA WITH DIALYTIKA AND TONOS
|
||||||
{ 0x0391, 0x41 }, // Α GREEK CAPITAL LETTER ALPHA
|
{ QChar(0x0391), 0x41 }, // Α GREEK CAPITAL LETTER ALPHA
|
||||||
{ 0x0392, 0x42 }, // Β GREEK CAPITAL LETTER BETA
|
{ QChar(0x0392), 0x42 }, // Β GREEK CAPITAL LETTER BETA
|
||||||
{ 0x0393, 0x43 }, // Γ GREEK CAPITAL LETTER GAMMA
|
{ QChar(0x0393), 0x43 }, // Γ GREEK CAPITAL LETTER GAMMA
|
||||||
{ 0x0394, 0x44 }, // Δ GREEK CAPITAL LETTER DELTA
|
{ QChar(0x0394), 0x44 }, // Δ GREEK CAPITAL LETTER DELTA
|
||||||
{ 0x0395, 0x45 }, // Ε GREEK CAPITAL LETTER EPSILON
|
{ QChar(0x0395), 0x45 }, // Ε GREEK CAPITAL LETTER EPSILON
|
||||||
{ 0x0396, 0x46 }, // Ζ GREEK CAPITAL LETTER ZETA
|
{ QChar(0x0396), 0x46 }, // Ζ GREEK CAPITAL LETTER ZETA
|
||||||
{ 0x0397, 0x47 }, // Η GREEK CAPITAL LETTER ETA
|
{ QChar(0x0397), 0x47 }, // Η GREEK CAPITAL LETTER ETA
|
||||||
{ 0x0398, 0x48 }, // Θ GREEK CAPITAL LETTER THETA
|
{ QChar(0x0398), 0x48 }, // Θ GREEK CAPITAL LETTER THETA
|
||||||
{ 0x0399, 0x49 }, // Ι GREEK CAPITAL LETTER IOTA
|
{ QChar(0x0399), 0x49 }, // Ι GREEK CAPITAL LETTER IOTA
|
||||||
{ 0x039a, 0x4a }, // Κ GREEK CAPITAL LETTER KAPPA
|
{ QChar(0x039a), 0x4a }, // Κ GREEK CAPITAL LETTER KAPPA
|
||||||
{ 0x039b, 0x4b }, // Λ GREEK CAPITAL LETTER LAMBDA
|
{ QChar(0x039b), 0x4b }, // Λ GREEK CAPITAL LETTER LAMBDA
|
||||||
{ 0x039c, 0x4c }, // Μ GREEK CAPITAL LETTER MU
|
{ QChar(0x039c), 0x4c }, // Μ GREEK CAPITAL LETTER MU
|
||||||
{ 0x039d, 0x4d }, // Ν GREEK CAPITAL LETTER NU
|
{ QChar(0x039d), 0x4d }, // Ν GREEK CAPITAL LETTER NU
|
||||||
{ 0x039e, 0x4e }, // Ξ GREEK CAPITAL LETTER XI
|
{ QChar(0x039e), 0x4e }, // Ξ GREEK CAPITAL LETTER XI
|
||||||
{ 0x039f, 0x4f }, // Ο GREEK CAPITAL LETTER OMICRON
|
{ QChar(0x039f), 0x4f }, // Ο GREEK CAPITAL LETTER OMICRON
|
||||||
{ 0x03a0, 0x50 }, // Π GREEK CAPITAL LETTER PI
|
{ QChar(0x03a0), 0x50 }, // Π GREEK CAPITAL LETTER PI
|
||||||
{ 0x03a1, 0x51 }, // Ρ GREEK CAPITAL LETTER RHO
|
{ QChar(0x03a1), 0x51 }, // Ρ GREEK CAPITAL LETTER RHO
|
||||||
{ 0x03a3, 0x53 }, // Σ GREEK CAPITAL LETTER SIGMA
|
{ QChar(0x03a3), 0x53 }, // Σ GREEK CAPITAL LETTER SIGMA
|
||||||
{ 0x03a4, 0x54 }, // Τ GREEK CAPITAL LETTER TAU
|
{ QChar(0x03a4), 0x54 }, // Τ GREEK CAPITAL LETTER TAU
|
||||||
{ 0x03a5, 0x55 }, // Υ GREEK CAPITAL LETTER UPSILON
|
{ QChar(0x03a5), 0x55 }, // Υ GREEK CAPITAL LETTER UPSILON
|
||||||
{ 0x03a6, 0x56 }, // Φ GREEK CAPITAL LETTER PHI
|
{ QChar(0x03a6), 0x56 }, // Φ GREEK CAPITAL LETTER PHI
|
||||||
{ 0x03a7, 0x57 }, // Χ GREEK CAPITAL LETTER CHI
|
{ QChar(0x03a7), 0x57 }, // Χ GREEK CAPITAL LETTER CHI
|
||||||
{ 0x03a8, 0x58 }, // Ψ GREEK CAPITAL LETTER PSI
|
{ QChar(0x03a8), 0x58 }, // Ψ GREEK CAPITAL LETTER PSI
|
||||||
{ 0x03a9, 0x59 }, // Ω GREEK CAPITAL LETTER OMEGA
|
{ QChar(0x03a9), 0x59 }, // Ω GREEK CAPITAL LETTER OMEGA
|
||||||
{ 0x03aa, 0x5a }, // Ϊ GREEK CAPITAL LETTER IOTA WITH DIALYTIKA
|
{ QChar(0x03aa), 0x5a }, // Ϊ GREEK CAPITAL LETTER IOTA WITH DIALYTIKA
|
||||||
{ 0x03ab, 0x5b }, // Ϋ GREEK CAPITAL LETTER UPSILON WITH DIALYTIKA
|
{ QChar(0x03ab), 0x5b }, // Ϋ GREEK CAPITAL LETTER UPSILON WITH DIALYTIKA
|
||||||
{ 0x03ac, 0x5c }, // ά GREEK SMALL LETTER ALPHA WITH TONOS
|
{ QChar(0x03ac), 0x5c }, // ά GREEK SMALL LETTER ALPHA WITH TONOS
|
||||||
{ 0x03ad, 0x5d }, // έ GREEK SMALL LETTER EPSILON WITH TONOS
|
{ QChar(0x03ad), 0x5d }, // έ GREEK SMALL LETTER EPSILON WITH TONOS
|
||||||
{ 0x03ae, 0x5e }, // ή GREEK SMALL LETTER ETA WITH TONOS
|
{ QChar(0x03ae), 0x5e }, // ή GREEK SMALL LETTER ETA WITH TONOS
|
||||||
{ 0x03af, 0x5f }, // ί GREEK SMALL LETTER IOTO WITH TONOS
|
{ QChar(0x03af), 0x5f }, // ί GREEK SMALL LETTER IOTO WITH TONOS
|
||||||
{ 0x03b0, 0x60 }, // ΰ GREEK SMALL LETTER UPSILON WITH DIALYTIKA AND TONOS
|
{ QChar(0x03b0), 0x60 }, // ΰ GREEK SMALL LETTER UPSILON WITH DIALYTIKA AND TONOS
|
||||||
{ 0x03b1, 0x61 }, // α GREEK SMALL LETTER ALPHA
|
{ QChar(0x03b1), 0x61 }, // α GREEK SMALL LETTER ALPHA
|
||||||
{ 0x03b2, 0x62 }, // β GREEK SMALL LETTER BETA
|
{ QChar(0x03b2), 0x62 }, // β GREEK SMALL LETTER BETA
|
||||||
{ 0x03b3, 0x63 }, // γ GREEK SMALL LETTER GAMMA
|
{ QChar(0x03b3), 0x63 }, // γ GREEK SMALL LETTER GAMMA
|
||||||
{ 0x03b4, 0x64 }, // δ GREEK SMALL LETTER DELTA
|
{ QChar(0x03b4), 0x64 }, // δ GREEK SMALL LETTER DELTA
|
||||||
{ 0x03b5, 0x65 }, // ε GREEK SMALL LETTER EPSILON
|
{ QChar(0x03b5), 0x65 }, // ε GREEK SMALL LETTER EPSILON
|
||||||
{ 0x03b6, 0x66 }, // ζ GREEK SMALL LETTER ZETA
|
{ QChar(0x03b6), 0x66 }, // ζ GREEK SMALL LETTER ZETA
|
||||||
{ 0x03b7, 0x67 }, // η GREEK SMALL LETTER ETA
|
{ QChar(0x03b7), 0x67 }, // η GREEK SMALL LETTER ETA
|
||||||
{ 0x03b8, 0x68 }, // θ GREEK SMALL LETTER THETA
|
{ QChar(0x03b8), 0x68 }, // θ GREEK SMALL LETTER THETA
|
||||||
{ 0x03b9, 0x69 }, // ι GREEK SMALL LETTER IOTA
|
{ QChar(0x03b9), 0x69 }, // ι GREEK SMALL LETTER IOTA
|
||||||
{ 0x03ba, 0x6a }, // κ GREEK SMALL LETTER KAPPA
|
{ QChar(0x03ba), 0x6a }, // κ GREEK SMALL LETTER KAPPA
|
||||||
{ 0x03bb, 0x6b }, // λ GREEK SMALL LETTER LAMBDA
|
{ QChar(0x03bb), 0x6b }, // λ GREEK SMALL LETTER LAMBDA
|
||||||
{ 0x03bc, 0x6c }, // μ GREEK SMALL LETTER MU
|
{ QChar(0x03bc), 0x6c }, // μ GREEK SMALL LETTER MU
|
||||||
{ 0x03bd, 0x6d }, // ν GREEK SMALL LETTER NU
|
{ QChar(0x03bd), 0x6d }, // ν GREEK SMALL LETTER NU
|
||||||
{ 0x03be, 0x6e }, // ξ GREEK SMALL LETTER XI
|
{ QChar(0x03be), 0x6e }, // ξ GREEK SMALL LETTER XI
|
||||||
{ 0x03bf, 0x6f }, // ο GREEK SMALL LETTER OMICRON
|
{ QChar(0x03bf), 0x6f }, // ο GREEK SMALL LETTER OMICRON
|
||||||
{ 0x03c0, 0x70 }, // π GREEK SMALL LETTER PI
|
{ QChar(0x03c0), 0x70 }, // π GREEK SMALL LETTER PI
|
||||||
{ 0x03c1, 0x71 }, // ρ GREEK SMALL LETTER RHO
|
{ QChar(0x03c1), 0x71 }, // ρ GREEK SMALL LETTER RHO
|
||||||
{ 0x03c2, 0x72 }, // ς GREEK SMALL LETTER FINAL SIGMA
|
{ QChar(0x03c2), 0x72 }, // ς GREEK SMALL LETTER FINAL SIGMA
|
||||||
{ 0x03c3, 0x73 }, // σ GREEK SMALL LETTER SIGMA
|
{ QChar(0x03c3), 0x73 }, // σ GREEK SMALL LETTER SIGMA
|
||||||
{ 0x03c4, 0x74 }, // τ GREEK SMALL LETTER TAU
|
{ QChar(0x03c4), 0x74 }, // τ GREEK SMALL LETTER TAU
|
||||||
{ 0x03c5, 0x75 }, // υ GREEK SMALL LETTER UPSILON
|
{ QChar(0x03c5), 0x75 }, // υ GREEK SMALL LETTER UPSILON
|
||||||
{ 0x03c6, 0x76 }, // φ GREEK SMALL LETTER PHI
|
{ QChar(0x03c6), 0x76 }, // φ GREEK SMALL LETTER PHI
|
||||||
{ 0x03c7, 0x77 }, // χ GREEK SMALL LETTER CHI
|
{ QChar(0x03c7), 0x77 }, // χ GREEK SMALL LETTER CHI
|
||||||
{ 0x03c8, 0x78 }, // ψ GREEK SMALL LETTER PSI
|
{ QChar(0x03c8), 0x78 }, // ψ GREEK SMALL LETTER PSI
|
||||||
{ 0x03c9, 0x79 }, // ω GREEK SMALL LETTER OMEGA
|
{ QChar(0x03c9), 0x79 }, // ω GREEK SMALL LETTER OMEGA
|
||||||
{ 0x03ca, 0x7a }, // ϊ GREEK SMALL LETTER IOTA WITH DIALYTIKA
|
{ QChar(0x03ca), 0x7a }, // ϊ GREEK SMALL LETTER IOTA WITH DIALYTIKA
|
||||||
{ 0x03cb, 0x7b }, // ϋ GREEK SMALL LETTER UPSILON WITH DIALYTIKA
|
{ QChar(0x03cb), 0x7b }, // ϋ GREEK SMALL LETTER UPSILON WITH DIALYTIKA
|
||||||
{ 0x03cc, 0x7c }, // ό GREEK SMALL LETTER OMICRON WITH TONOS
|
{ QChar(0x03cc), 0x7c }, // ό GREEK SMALL LETTER OMICRON WITH TONOS
|
||||||
{ 0x03cd, 0x7d }, // ύ GREEK SMALL LETTER UPSILON WITH TONOS
|
{ QChar(0x03cd), 0x7d }, // ύ GREEK SMALL LETTER UPSILON WITH TONOS
|
||||||
{ 0x03ce, 0x7e } // ώ GREEK SMALL LETTER OMEGA WITH TONOS
|
{ QChar(0x03ce), 0x7e } // ώ GREEK SMALL LETTER OMEGA WITH TONOS
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // TODO 5 Arabic G0
|
{ // TODO 5 Arabic G0
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 6 Hebrew G0
|
{ // 6 Hebrew G0
|
||||||
{ 0x05d0, 0x60 }, // א HEBREW LETTER ALEF
|
{ QChar(0x05d0), 0x60 }, // א HEBREW LETTER ALEF
|
||||||
{ 0x05d1, 0x61 }, // ב HEBREW LETTER BET
|
{ QChar(0x05d1), 0x61 }, // ב HEBREW LETTER BET
|
||||||
{ 0x05d2, 0x62 }, // ג HEBREW LETTER GIMEL
|
{ QChar(0x05d2), 0x62 }, // ג HEBREW LETTER GIMEL
|
||||||
{ 0x05d3, 0x63 }, // ד HEBREW LETTER DALET
|
{ QChar(0x05d3), 0x63 }, // ד HEBREW LETTER DALET
|
||||||
{ 0x05d4, 0x64 }, // ה HEBREW LETTER HE
|
{ QChar(0x05d4), 0x64 }, // ה HEBREW LETTER HE
|
||||||
{ 0x05d5, 0x65 }, // ו HEBREW LETTER VAV
|
{ QChar(0x05d5), 0x65 }, // ו HEBREW LETTER VAV
|
||||||
{ 0x05d6, 0x66 }, // ז HEBREW LETTER ZAYIN
|
{ QChar(0x05d6), 0x66 }, // ז HEBREW LETTER ZAYIN
|
||||||
{ 0x05d7, 0x67 }, // ח HEBREW LETTER HET
|
{ QChar(0x05d7), 0x67 }, // ח HEBREW LETTER HET
|
||||||
{ 0x05d8, 0x68 }, // ט HEBREW LETTER TET
|
{ QChar(0x05d8), 0x68 }, // ט HEBREW LETTER TET
|
||||||
{ 0x05d9, 0x69 }, // י HEBREW LETTER YOD
|
{ QChar(0x05d9), 0x69 }, // י HEBREW LETTER YOD
|
||||||
{ 0x05da, 0x6a }, // ך HEBREW LETTER FINAL KAF
|
{ QChar(0x05da), 0x6a }, // ך HEBREW LETTER FINAL KAF
|
||||||
{ 0x05db, 0x6b }, // כ HEBREW LETTER KAF
|
{ QChar(0x05db), 0x6b }, // כ HEBREW LETTER KAF
|
||||||
{ 0x05dc, 0x6c }, // ל HEBREW LETTER LAMED
|
{ QChar(0x05dc), 0x6c }, // ל HEBREW LETTER LAMED
|
||||||
{ 0x05dd, 0x6d }, // ם HEBREW LETTER FINAL MEM
|
{ QChar(0x05dd), 0x6d }, // ם HEBREW LETTER FINAL MEM
|
||||||
{ 0x05de, 0x6e }, // מ HEBREW LETTER MEM
|
{ QChar(0x05de), 0x6e }, // מ HEBREW LETTER MEM
|
||||||
{ 0x05df, 0x6f }, // ן HEBREW LETTER FINAL NUN
|
{ QChar(0x05df), 0x6f }, // ן HEBREW LETTER FINAL NUN
|
||||||
{ 0x05e0, 0x70 }, // נ HEBREW LETTER NUN
|
{ QChar(0x05e0), 0x70 }, // נ HEBREW LETTER NUN
|
||||||
{ 0x05e1, 0x71 }, // ס HEBREW LETTER SAMEKH
|
{ QChar(0x05e1), 0x71 }, // ס HEBREW LETTER SAMEKH
|
||||||
{ 0x05e2, 0x72 }, // ע HEBREW LETTER AYIN
|
{ QChar(0x05e2), 0x72 }, // ע HEBREW LETTER AYIN
|
||||||
{ 0x05e3, 0x73 }, // ף HEBREW LETTER FINAL PE
|
{ QChar(0x05e3), 0x73 }, // ף HEBREW LETTER FINAL PE
|
||||||
{ 0x05e4, 0x74 }, // פ HEBREW LETTER PE
|
{ QChar(0x05e4), 0x74 }, // פ HEBREW LETTER PE
|
||||||
{ 0x05e5, 0x75 }, // ץ HEBREW LETTER FINAL TSADI
|
{ QChar(0x05e5), 0x75 }, // ץ HEBREW LETTER FINAL TSADI
|
||||||
{ 0x05e6, 0x76 }, // צ HEBREW LETTER TSADI
|
{ QChar(0x05e6), 0x76 }, // צ HEBREW LETTER TSADI
|
||||||
{ 0x05e7, 0x77 }, // ק HEBREW LETTER QOF
|
{ QChar(0x05e7), 0x77 }, // ק HEBREW LETTER QOF
|
||||||
{ 0x05e8, 0x78 }, // ר HEBREW LETTER RESH
|
{ QChar(0x05e8), 0x78 }, // ר HEBREW LETTER RESH
|
||||||
{ 0x05e9, 0x79 }, // ש HEBREW LETTER SHIN
|
{ QChar(0x05e9), 0x79 }, // ש HEBREW LETTER SHIN
|
||||||
{ 0x05ea, 0x7a }, // ת HEBREW LETTER TAV
|
{ QChar(0x05ea), 0x7a }, // ת HEBREW LETTER TAV
|
||||||
{ 0x20aa, 0x7b }, // ₪ NEW SHEQEL SIGN
|
{ QChar(0x20aa), 0x7b }, // ₪ NEW SHEQEL SIGN
|
||||||
{ 0x00be, 0x7d }, // ½ VULGAR FRACTION THREE QUARTERS
|
{ QChar(0x00be), 0x7d }, // ½ VULGAR FRACTION THREE QUARTERS
|
||||||
{ 0x00f7, 0x7e } // ÷ DIVISION SIGN
|
{ QChar(0x00f7), 0x7e } // ÷ DIVISION SIGN
|
||||||
},
|
},
|
||||||
|
|
||||||
// Only used by X/26 enhancements, not directly by keyboard
|
// Only used by X/26 enhancements, not directly by keyboard
|
||||||
@@ -339,204 +339,204 @@ static const QMap<QChar, char> keymapping[24] = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
{ // 11 Czech/Slovak
|
{ // 11 Czech/Slovak
|
||||||
{ 0x016f, 0x24 }, // ů LATIN SMALL LETTER U WITH RING ABOVE
|
{ QChar(0x016f), 0x24 }, // ů LATIN SMALL LETTER U WITH RING ABOVE
|
||||||
{ 0x010d, 0x40 }, // č LATIN SMALL LETTER C WITH CARON
|
{ QChar(0x010d), 0x40 }, // č LATIN SMALL LETTER C WITH CARON
|
||||||
{ 0x0165, 0x5b }, // ť LATIN SMALL LETTER T WITH CARON
|
{ QChar(0x0165), 0x5b }, // ť LATIN SMALL LETTER T WITH CARON
|
||||||
{ 0x017e, 0x5c }, // ž LATIN SMALL LETTER Z WITH CARON
|
{ QChar(0x017e), 0x5c }, // ž LATIN SMALL LETTER Z WITH CARON
|
||||||
{ 0x00fd, 0x5d }, // ý LATIN SMALL LETTER Y WITH ACUTE
|
{ QChar(0x00fd), 0x5d }, // ý LATIN SMALL LETTER Y WITH ACUTE
|
||||||
{ 0x00ed, 0x5e }, // í LATIN SMALL LETTER I WITH ACUTE
|
{ QChar(0x00ed), 0x5e }, // í LATIN SMALL LETTER I WITH ACUTE
|
||||||
{ 0x0159, 0x5f }, // ř LATIN SMALL LETTER R WITH CARON
|
{ QChar(0x0159), 0x5f }, // ř LATIN SMALL LETTER R WITH CARON
|
||||||
{ 0x00e9, 0x60 }, // é LATIN SMALL LETTER E WITH ACUTE
|
{ QChar(0x00e9), 0x60 }, // é LATIN SMALL LETTER E WITH ACUTE
|
||||||
{ 0x00e1, 0x7b }, // á LATIN SMALL LETTER A WITH ACUTE
|
{ QChar(0x00e1), 0x7b }, // á LATIN SMALL LETTER A WITH ACUTE
|
||||||
{ 0x011b, 0x7c }, // ě LATIN SMALL LETTER E WITH CARON
|
{ QChar(0x011b), 0x7c }, // ě LATIN SMALL LETTER E WITH CARON
|
||||||
{ 0x00fa, 0x7d }, // ú LATIN SMALL LETTER U WITH ACUTE
|
{ QChar(0x00fa), 0x7d }, // ú LATIN SMALL LETTER U WITH ACUTE
|
||||||
{ 0x0161, 0x7e } // š LATIN SMALL LETTER S WITH CARON
|
{ QChar(0x0161), 0x7e } // š LATIN SMALL LETTER S WITH CARON
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 12 English
|
{ // 12 English
|
||||||
{ 0x00a3, 0x23 }, // £ POUND SIGN
|
{ QChar(0x00a3), 0x23 }, // £ POUND SIGN
|
||||||
{ 0x0023, 0x5f }, // # NUMBER SIGN
|
{ QChar(0x0023), 0x5f }, // # NUMBER SIGN
|
||||||
{ 0x005f, 0x60 }, // ─ LOW LINE - rendered as U+2500 BOX DRAWINGS LIGHT HORIZONTAL
|
{ QChar(0x005f), 0x60 }, // ─ LOW LINE - rendered as U+2500 BOX DRAWINGS LIGHT HORIZONTAL
|
||||||
|
|
||||||
{ 0x00bc, 0x7b }, // ¼ VULGAR FRACTION ONE QUARTER
|
{ QChar(0x00bc), 0x7b }, // ¼ VULGAR FRACTION ONE QUARTER
|
||||||
{ 0x00bd, 0x5c }, // ½ VULGAR FRACTION ONE HALF
|
{ QChar(0x00bd), 0x5c }, // ½ VULGAR FRACTION ONE HALF
|
||||||
{ 0x00be, 0x7b }, // ¾ VULGAR FRACTION THREE QUARTERS
|
{ QChar(0x00be), 0x7b }, // ¾ VULGAR FRACTION THREE QUARTERS
|
||||||
{ 0x00f7, 0x7e } // ÷ DIVISION SIGN
|
{ QChar(0x00f7), 0x7e } // ÷ DIVISION SIGN
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 13 Estonian
|
{ // 13 Estonian
|
||||||
{ 0x00f5, 0x24 }, // õ LATIN SMALL LETTER O WITH TILDE
|
{ QChar(0x00f5), 0x24 }, // õ LATIN SMALL LETTER O WITH TILDE
|
||||||
{ 0x0160, 0x40 }, // Š LATIN CAPITAL LETTER S WITH CARON
|
{ QChar(0x0160), 0x40 }, // Š LATIN CAPITAL LETTER S WITH CARON
|
||||||
{ 0x00c4, 0x5b }, // Ä LATIN CAPITAL LETTER A WITH DIAERESIS
|
{ QChar(0x00c4), 0x5b }, // Ä LATIN CAPITAL LETTER A WITH DIAERESIS
|
||||||
{ 0x00d6, 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
{ QChar(0x00d6), 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
||||||
{ 0x017d, 0x5d }, // Ž LATIN CAPITAL LETTER Z WITH CARON
|
{ QChar(0x017d), 0x5d }, // Ž LATIN CAPITAL LETTER Z WITH CARON
|
||||||
{ 0x00dc, 0x5e }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
{ QChar(0x00dc), 0x5e }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
||||||
{ 0x00d5, 0x5f }, // Õ LATIN CAPITAL LETTER O WITH TILDE
|
{ QChar(0x00d5), 0x5f }, // Õ LATIN CAPITAL LETTER O WITH TILDE
|
||||||
{ 0x0161, 0x60 }, // š LATIN SMALL LETTER S WITH CARON
|
{ QChar(0x0161), 0x60 }, // š LATIN SMALL LETTER S WITH CARON
|
||||||
{ 0x00e4, 0x7b }, // ä LATIN SMALL LETTER A WITH DIAERESIS
|
{ QChar(0x00e4), 0x7b }, // ä LATIN SMALL LETTER A WITH DIAERESIS
|
||||||
{ 0x00f6, 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
{ QChar(0x00f6), 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
||||||
{ 0x017e, 0x7d }, // ž LATIN SMALL LETTER Z WITH CARON
|
{ QChar(0x017e), 0x7d }, // ž LATIN SMALL LETTER Z WITH CARON
|
||||||
{ 0x00fc, 0x7e } // ü LATIN SMALL LETTER U WITH DIAERESIS
|
{ QChar(0x00fc), 0x7e } // ü LATIN SMALL LETTER U WITH DIAERESIS
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 14 French - aze qsd wxc
|
{ // 14 French - aze qsd wxc
|
||||||
{ 0x00e9, 0x23 }, // é LATIN SMALL LETTER E WITH ACUTE
|
{ QChar(0x00e9), 0x23 }, // é LATIN SMALL LETTER E WITH ACUTE
|
||||||
{ 0x00ef, 0x24 }, // ï LATIN SMALL LETTER I WITH DIAERESIS
|
{ QChar(0x00ef), 0x24 }, // ï LATIN SMALL LETTER I WITH DIAERESIS
|
||||||
{ 0x00e0, 0x40 }, // à LATIN SMALL LETTER A WITH GRAVE
|
{ QChar(0x00e0), 0x40 }, // à LATIN SMALL LETTER A WITH GRAVE
|
||||||
{ 0x00eb, 0x5b }, // ë LATIN SMALL LETTER E WITH DIAERESIS
|
{ QChar(0x00eb), 0x5b }, // ë LATIN SMALL LETTER E WITH DIAERESIS
|
||||||
{ 0x00ea, 0x5c }, // ê LATIN SMALL LETTER E WITH CIRCUMFLEX
|
{ QChar(0x00ea), 0x5c }, // ê LATIN SMALL LETTER E WITH CIRCUMFLEX
|
||||||
{ 0x00f9, 0x5d }, // ù LATIN SMALL LETTER U WITH GRAVE
|
{ QChar(0x00f9), 0x5d }, // ù LATIN SMALL LETTER U WITH GRAVE
|
||||||
{ 0x00ee, 0x5e }, // î LATIN SMALL LETTER I WITH CIRCUMFLEX
|
{ QChar(0x00ee), 0x5e }, // î LATIN SMALL LETTER I WITH CIRCUMFLEX
|
||||||
{ 0x0023, 0x5f }, // # NUMBER SIGN
|
{ QChar(0x0023), 0x5f }, // # NUMBER SIGN
|
||||||
{ 0x00e8, 0x60 }, // è LATIN SMALL LETTER E WITH GRAVE
|
{ QChar(0x00e8), 0x60 }, // è LATIN SMALL LETTER E WITH GRAVE
|
||||||
{ 0x00e2, 0x7b }, // â LATIN SMALL LETTER A WITH CIRCUMFLEX
|
{ QChar(0x00e2), 0x7b }, // â LATIN SMALL LETTER A WITH CIRCUMFLEX
|
||||||
{ 0x00f4, 0x7c }, // ô LATIN SMALL LETTER O WITH CIRCUMFLEX
|
{ QChar(0x00f4), 0x7c }, // ô LATIN SMALL LETTER O WITH CIRCUMFLEX
|
||||||
{ 0x00fb, 0x7d }, // û LATIN SMALL LETTER U WITH CIRCUMFLEX
|
{ QChar(0x00fb), 0x7d }, // û LATIN SMALL LETTER U WITH CIRCUMFLEX
|
||||||
{ 0x00e7, 0x7e } // ç LATIN SMALL LETTER C WITH CEDILLA
|
{ QChar(0x00e7), 0x7e } // ç LATIN SMALL LETTER C WITH CEDILLA
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 15 German
|
{ // 15 German
|
||||||
{ 0x00a7, 0x40 }, // § SECTION SIGN
|
{ QChar(0x00a7), 0x40 }, // § SECTION SIGN
|
||||||
{ 0x00c4, 0x5b }, // Ä LATIN CAPITAL LETTER A WITH DIAERESIS
|
{ QChar(0x00c4), 0x5b }, // Ä LATIN CAPITAL LETTER A WITH DIAERESIS
|
||||||
{ 0x00d6, 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
{ QChar(0x00d6), 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
||||||
{ 0x00dc, 0x5d }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
{ QChar(0x00dc), 0x5d }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
||||||
{ 0x00b0, 0x60 }, // ° DEGREE SIGN
|
{ QChar(0x00b0), 0x60 }, // ° DEGREE SIGN
|
||||||
{ 0x00e4, 0x7b }, // ä LATIN SMALL LETTER A WITH DIAERESIS
|
{ QChar(0x00e4), 0x7b }, // ä LATIN SMALL LETTER A WITH DIAERESIS
|
||||||
{ 0x00f6, 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
{ QChar(0x00f6), 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
||||||
{ 0x00fc, 0x7d }, // ü LATIN SMALL LETTER U WITH DIAERESIS
|
{ QChar(0x00fc), 0x7d }, // ü LATIN SMALL LETTER U WITH DIAERESIS
|
||||||
{ 0x00df, 0x7e } // ß LATIN SMALL LETTER SHARP S
|
{ QChar(0x00df), 0x7e } // ß LATIN SMALL LETTER SHARP S
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 16 Italian
|
{ // 16 Italian
|
||||||
{ 0x00a3, 0x23 }, // £ POUND SIGN
|
{ QChar(0x00a3), 0x23 }, // £ POUND SIGN
|
||||||
{ 0x00e9, 0x40 }, // é LATIN SMALL LETTER E WITH ACUTE
|
{ QChar(0x00e9), 0x40 }, // é LATIN SMALL LETTER E WITH ACUTE
|
||||||
{ 0x00b0, 0x5b }, // ° DEGREE SIGN
|
{ QChar(0x00b0), 0x5b }, // ° DEGREE SIGN
|
||||||
{ 0x00e7, 0x5c }, // ç LATIN SMALL LETTER C WITH CEDILLA
|
{ QChar(0x00e7), 0x5c }, // ç LATIN SMALL LETTER C WITH CEDILLA
|
||||||
{ 0x0023, 0x5f }, // # NUMBER SIGN
|
{ QChar(0x0023), 0x5f }, // # NUMBER SIGN
|
||||||
{ 0x00f9, 0x60 }, // ù LATIN SMALL LETTER U WITH GRAVE
|
{ QChar(0x00f9), 0x60 }, // ù LATIN SMALL LETTER U WITH GRAVE
|
||||||
{ 0x00e0, 0x7b }, // à LATIN SMALL LETTER A WITH GRAVE
|
{ QChar(0x00e0), 0x7b }, // à LATIN SMALL LETTER A WITH GRAVE
|
||||||
{ 0x00f2, 0x7c }, // ò LATIN SMALL LETTER O WITH GRAVE
|
{ QChar(0x00f2), 0x7c }, // ò LATIN SMALL LETTER O WITH GRAVE
|
||||||
{ 0x00e8, 0x7d }, // è LATIN SMALL LETTER E WITH GRAVE
|
{ QChar(0x00e8), 0x7d }, // è LATIN SMALL LETTER E WITH GRAVE
|
||||||
{ 0x00ec, 0x7e } // ì LATIN SMALL LETTER I WITH GRAVE
|
{ QChar(0x00ec), 0x7e } // ì LATIN SMALL LETTER I WITH GRAVE
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 17 Lettish/Lithuanian
|
{ // 17 Lettish/Lithuanian
|
||||||
{ 0x0160, 0x40 }, // Š LATIN CAPITAL LETTER S WITH CARON
|
{ QChar(0x0160), 0x40 }, // Š LATIN CAPITAL LETTER S WITH CARON
|
||||||
{ 0x0117, 0x5b }, // ė LATIN SMALL LETTER E WITH DOT ABOVE
|
{ QChar(0x0117), 0x5b }, // ė LATIN SMALL LETTER E WITH DOT ABOVE
|
||||||
{ 0x0119, 0x5c }, // ę LATIN SMALL LETTER E WITH OGONEK
|
{ QChar(0x0119), 0x5c }, // ę LATIN SMALL LETTER E WITH OGONEK
|
||||||
{ 0x017d, 0x5d }, // Ž LATIN CAPITAL LETTER Z WITH CARON
|
{ QChar(0x017d), 0x5d }, // Ž LATIN CAPITAL LETTER Z WITH CARON
|
||||||
{ 0x010d, 0x5e }, // č LATIN SMALL LETTER C WITH CARON
|
{ QChar(0x010d), 0x5e }, // č LATIN SMALL LETTER C WITH CARON
|
||||||
{ 0x016b, 0x5f }, // ū LATIN SMALL LETTER U WITH MACRON
|
{ QChar(0x016b), 0x5f }, // ū LATIN SMALL LETTER U WITH MACRON
|
||||||
{ 0x0161, 0x60 }, // š LATIN SMALL LETTER S WITH CARON
|
{ QChar(0x0161), 0x60 }, // š LATIN SMALL LETTER S WITH CARON
|
||||||
{ 0x0105, 0x7b }, // ą LATIN SMALL LETTER A WITH OGONEK
|
{ QChar(0x0105), 0x7b }, // ą LATIN SMALL LETTER A WITH OGONEK
|
||||||
{ 0x0173, 0x7c }, // ų LATIN SMALL LETTER U WITH OGONEK
|
{ QChar(0x0173), 0x7c }, // ų LATIN SMALL LETTER U WITH OGONEK
|
||||||
{ 0x017e, 0x7d }, // ž LATIN SMALL LETTER Z WITH CARON
|
{ QChar(0x017e), 0x7d }, // ž LATIN SMALL LETTER Z WITH CARON
|
||||||
{ 0x012f, 0x7e } // į LATIN SMALL LETTER I WITH OGONEK
|
{ QChar(0x012f), 0x7e } // į LATIN SMALL LETTER I WITH OGONEK
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 18 Polish
|
{ // 18 Polish
|
||||||
{ 0x0144, 0x24 }, // ń LATIN SMALL LETTER N WITH ACUTE
|
{ QChar(0x0144), 0x24 }, // ń LATIN SMALL LETTER N WITH ACUTE
|
||||||
{ 0x0105, 0x40 }, // ą LATIN SMALL LETTER A WITH OGONEK
|
{ QChar(0x0105), 0x40 }, // ą LATIN SMALL LETTER A WITH OGONEK
|
||||||
{ 0x017b, 0x5b }, // Ż LATIN CAPITAL LETTER Z WITH DOT ABOVE - rendered as U+01B5 ...WITH STROKE
|
{ QChar(0x017b), 0x5b }, // Ż LATIN CAPITAL LETTER Z WITH DOT ABOVE - rendered as U+01B5 ...WITH STROKE
|
||||||
{ 0x015a, 0x5c }, // Ś LATIN CAPITAL LETTER S WITH ACUTE
|
{ QChar(0x015a), 0x5c }, // Ś LATIN CAPITAL LETTER S WITH ACUTE
|
||||||
{ 0x0141, 0x5d }, // Ł LATIN CAPITAL LETTER L WITH STROKE
|
{ QChar(0x0141), 0x5d }, // Ł LATIN CAPITAL LETTER L WITH STROKE
|
||||||
{ 0x0107, 0x5e }, // ć LATIN SMALL LETTER C WITH ACUTE
|
{ QChar(0x0107), 0x5e }, // ć LATIN SMALL LETTER C WITH ACUTE
|
||||||
{ 0x00f3, 0x5f }, // ó LATIN SMALL LETTER O WITH ACUTE
|
{ QChar(0x00f3), 0x5f }, // ó LATIN SMALL LETTER O WITH ACUTE
|
||||||
{ 0x0119, 0x60 }, // ę LATIN SMALL LETTER E WITH OGONEK
|
{ QChar(0x0119), 0x60 }, // ę LATIN SMALL LETTER E WITH OGONEK
|
||||||
{ 0x017c, 0x7b }, // ż LATIN SMALL LETTER Z WITH DOT ABOVE
|
{ QChar(0x017c), 0x7b }, // ż LATIN SMALL LETTER Z WITH DOT ABOVE
|
||||||
{ 0x015b, 0x7c }, // ś LATIN SMALL LETTER S WITH ACUTE
|
{ QChar(0x015b), 0x7c }, // ś LATIN SMALL LETTER S WITH ACUTE
|
||||||
{ 0x0142, 0x7d }, // ł LATIN SMALL LETTER L WITH STROKE
|
{ QChar(0x0142), 0x7d }, // ł LATIN SMALL LETTER L WITH STROKE
|
||||||
{ 0x017a, 0x7e } // ź LATIN SMALL LETTER Z WITH ACUTE
|
{ QChar(0x017a), 0x7e } // ź LATIN SMALL LETTER Z WITH ACUTE
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 19 Portuguese/Spanish
|
{ // 19 Portuguese/Spanish
|
||||||
{ 0x00e7, 0x23 }, // ç LATIN SMALL LETTER C WITH CEDILLA
|
{ QChar(0x00e7), 0x23 }, // ç LATIN SMALL LETTER C WITH CEDILLA
|
||||||
{ 0x00a1, 0x40 }, // ¡ INVERTED EXCLAMATION MARK
|
{ QChar(0x00a1), 0x40 }, // ¡ INVERTED EXCLAMATION MARK
|
||||||
{ 0x00e1, 0x5b }, // á LATIN SMALL LETTER A WITH ACUTE
|
{ QChar(0x00e1), 0x5b }, // á LATIN SMALL LETTER A WITH ACUTE
|
||||||
{ 0x00e9, 0x5c }, // é LATIN SMALL LETTER E WITH ACUTE
|
{ QChar(0x00e9), 0x5c }, // é LATIN SMALL LETTER E WITH ACUTE
|
||||||
{ 0x00ed, 0x5d }, // í LATIN SMALL LETTER I WITH ACUTE
|
{ QChar(0x00ed), 0x5d }, // í LATIN SMALL LETTER I WITH ACUTE
|
||||||
{ 0x00f3, 0x5e }, // ó LATIN SMALL LETTER O WITH ACUTE
|
{ QChar(0x00f3), 0x5e }, // ó LATIN SMALL LETTER O WITH ACUTE
|
||||||
{ 0x00fa, 0x5f }, // ú LATIN SMALL LETTER U WITH ACUTE
|
{ QChar(0x00fa), 0x5f }, // ú LATIN SMALL LETTER U WITH ACUTE
|
||||||
{ 0x00bf, 0x60 }, // ¿ INVERTED QUESTION MARK
|
{ QChar(0x00bf), 0x60 }, // ¿ INVERTED QUESTION MARK
|
||||||
{ 0x00fc, 0x7b }, // ü LATIN SMALL LETTER U WITH DIAERESIS
|
{ QChar(0x00fc), 0x7b }, // ü LATIN SMALL LETTER U WITH DIAERESIS
|
||||||
{ 0x00f1, 0x7c }, // ñ LATIN SMALL LETTER N WITH TILDE
|
{ QChar(0x00f1), 0x7c }, // ñ LATIN SMALL LETTER N WITH TILDE
|
||||||
{ 0x00e8, 0x7d }, // è LATIN SMALL LETTER E WITH GRAVE
|
{ QChar(0x00e8), 0x7d }, // è LATIN SMALL LETTER E WITH GRAVE
|
||||||
{ 0x00e0, 0x7e } // à LATIN SMALL LETTER A WITH GRAVE
|
{ QChar(0x00e0), 0x7e } // à LATIN SMALL LETTER A WITH GRAVE
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 20 Rumanian
|
{ // 20 Rumanian
|
||||||
{ 0x00a4, 0x24 }, // ¤ CURRENCY SIGN
|
{ QChar(0x00a4), 0x24 }, // ¤ CURRENCY SIGN
|
||||||
{ 0x021a, 0x40 }, // Ț LATIN CAPITAL LETTER T WITH COMMA BELOW
|
{ QChar(0x021a), 0x40 }, // Ț LATIN CAPITAL LETTER T WITH COMMA BELOW
|
||||||
{ 0x00c2, 0x5b }, // Â LATIN CAPITAL LETTER A WITH CIRCUMFLEX
|
{ QChar(0x00c2), 0x5b }, // Â LATIN CAPITAL LETTER A WITH CIRCUMFLEX
|
||||||
{ 0x0218, 0x5c }, // Ș LATIN CAPITAL LETTER S WITH COMMA BELOW
|
{ QChar(0x0218), 0x5c }, // Ș LATIN CAPITAL LETTER S WITH COMMA BELOW
|
||||||
{ 0x0102, 0x5d }, // Ă LATIN CAPITAL LETTER A WITH BREVE
|
{ QChar(0x0102), 0x5d }, // Ă LATIN CAPITAL LETTER A WITH BREVE
|
||||||
{ 0x00c3, 0x5e }, // Î LATIN CAPITAL LETTER I WITH CIRCUMFLEX
|
{ QChar(0x00c3), 0x5e }, // Î LATIN CAPITAL LETTER I WITH CIRCUMFLEX
|
||||||
{ 0x0131, 0x5f }, // ı LATIN SMALL LETTER DOTLESS I
|
{ QChar(0x0131), 0x5f }, // ı LATIN SMALL LETTER DOTLESS I
|
||||||
{ 0x021b, 0x60 }, // ț LATIN SMALL LETTER T WITH COMMA BELOW
|
{ QChar(0x021b), 0x60 }, // ț LATIN SMALL LETTER T WITH COMMA BELOW
|
||||||
{ 0x00e2, 0x7b }, // â LATIN SMALL LETTER A WITH CIRCUMFLEX
|
{ QChar(0x00e2), 0x7b }, // â LATIN SMALL LETTER A WITH CIRCUMFLEX
|
||||||
{ 0x0219, 0x7c }, // ș LATIN SMALL LETTER S WITH COMMA BELOW
|
{ QChar(0x0219), 0x7c }, // ș LATIN SMALL LETTER S WITH COMMA BELOW
|
||||||
{ 0x0103, 0x7d }, // ă LATIN SMALL LETTER A WITH BREVE
|
{ QChar(0x0103), 0x7d }, // ă LATIN SMALL LETTER A WITH BREVE
|
||||||
{ 0x00ee, 0x7e } // î LATIN SMALL LETTER I WITH CIRCUMFLEX
|
{ QChar(0x00ee), 0x7e } // î LATIN SMALL LETTER I WITH CIRCUMFLEX
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 21 Serbian/Croatian/Slovenian
|
{ // 21 Serbian/Croatian/Slovenian
|
||||||
{ 0x00cb, 0x24 }, // Ë LATIN CAPITAL LETTER E WITH DIAERESIS
|
{ QChar(0x00cb), 0x24 }, // Ë LATIN CAPITAL LETTER E WITH DIAERESIS
|
||||||
{ 0x010c, 0x40 }, // Č LATIN CAPITAL LETTER C WITH CARON
|
{ QChar(0x010c), 0x40 }, // Č LATIN CAPITAL LETTER C WITH CARON
|
||||||
{ 0x0106, 0x5b }, // Ć LATIN CAPITAL LETTER C WITH ACUTE
|
{ QChar(0x0106), 0x5b }, // Ć LATIN CAPITAL LETTER C WITH ACUTE
|
||||||
{ 0x017d, 0x5c }, // Ž LATIN CAPITAL LETTER Z WITH CARON
|
{ QChar(0x017d), 0x5c }, // Ž LATIN CAPITAL LETTER Z WITH CARON
|
||||||
{ 0x0110, 0x5d }, // Đ LATIN CAPITAL LETTER D WITH STROKE
|
{ QChar(0x0110), 0x5d }, // Đ LATIN CAPITAL LETTER D WITH STROKE
|
||||||
{ 0x0160, 0x5e }, // Š LATIN CAPITAL LETTER S WITH CARON
|
{ QChar(0x0160), 0x5e }, // Š LATIN CAPITAL LETTER S WITH CARON
|
||||||
{ 0x00eb, 0x5f }, // ë LATIN SMALL LETTER E WITH DIAERESIS
|
{ QChar(0x00eb), 0x5f }, // ë LATIN SMALL LETTER E WITH DIAERESIS
|
||||||
{ 0x010d, 0x60 }, // č LATIN SMALL LETTER C WITH CARON
|
{ QChar(0x010d), 0x60 }, // č LATIN SMALL LETTER C WITH CARON
|
||||||
{ 0x0107, 0x7b }, // ć LATIN SMALL LETTER C WITH ACUTE
|
{ QChar(0x0107), 0x7b }, // ć LATIN SMALL LETTER C WITH ACUTE
|
||||||
{ 0x017e, 0x7c }, // ž LATIN SMALL LETTER Z WITH CARON
|
{ QChar(0x017e), 0x7c }, // ž LATIN SMALL LETTER Z WITH CARON
|
||||||
{ 0x0111, 0x7d }, // đ LATIN SMALL LETTER D WITH STROKE
|
{ QChar(0x0111), 0x7d }, // đ LATIN SMALL LETTER D WITH STROKE
|
||||||
{ 0x0161, 0x7e } // š LATIN SMALL LETTER S WITH CARON
|
{ QChar(0x0161), 0x7e } // š LATIN SMALL LETTER S WITH CARON
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 22 Swedish/Finnish/Hungarian
|
{ // 22 Swedish/Finnish/Hungarian
|
||||||
{ 0x00a4, 0x24 }, // ¤ CURRENCY SIGN
|
{ QChar(0x00a4), 0x24 }, // ¤ CURRENCY SIGN
|
||||||
{ 0x00c9, 0x40 }, // É LATIN CAPITAL LETTER E WITH ACUTE
|
{ QChar(0x00c9), 0x40 }, // É LATIN CAPITAL LETTER E WITH ACUTE
|
||||||
{ 0x00c4, 0x5b }, // Ä LATIN CAPITAL LETTER A WITH DIAERESIS
|
{ QChar(0x00c4), 0x5b }, // Ä LATIN CAPITAL LETTER A WITH DIAERESIS
|
||||||
{ 0x00d6, 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
{ QChar(0x00d6), 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
||||||
{ 0x00c5, 0x5d }, // Å LATIN CAPITAL LETTER A WITH RING ABOVE
|
{ QChar(0x00c5), 0x5d }, // Å LATIN CAPITAL LETTER A WITH RING ABOVE
|
||||||
{ 0x00dc, 0x5e }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
{ QChar(0x00dc), 0x5e }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
||||||
{ 0x00e9, 0x60 }, // é LATIN SMALL LETTER E WITH ACUTE
|
{ QChar(0x00e9), 0x60 }, // é LATIN SMALL LETTER E WITH ACUTE
|
||||||
{ 0x00e4, 0x7b }, // ä LATIN SMALL LETTER A WITH DIAERESIS
|
{ QChar(0x00e4), 0x7b }, // ä LATIN SMALL LETTER A WITH DIAERESIS
|
||||||
{ 0x00f6, 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
{ QChar(0x00f6), 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
||||||
{ 0x00e5, 0x7d }, // å LATIN SMALL LETTER A WITH RING ABOVE
|
{ QChar(0x00e5), 0x7d }, // å LATIN SMALL LETTER A WITH RING ABOVE
|
||||||
{ 0x00fc, 0x7e } // ü LATIN SMALL LETTER U WITH DIAERESIS
|
{ QChar(0x00fc), 0x7e } // ü LATIN SMALL LETTER U WITH DIAERESIS
|
||||||
},
|
},
|
||||||
|
|
||||||
{ // 23 Turkish
|
{ // 23 Turkish
|
||||||
{ 0x20ba, 0x23 }, // ₺ TURKISH LIRA SIGN
|
{ QChar(0x20ba), 0x23 }, // ₺ TURKISH LIRA SIGN
|
||||||
{ 0x011f, 0x24 }, // ğ LATIN SMALL LETTER G WITH BREVE
|
{ QChar(0x011f), 0x24 }, // ğ LATIN SMALL LETTER G WITH BREVE
|
||||||
{ 0x0130, 0x40 }, // İ LATIN CAPITAL LETTER I WITH DOT ABOVE
|
{ QChar(0x0130), 0x40 }, // İ LATIN CAPITAL LETTER I WITH DOT ABOVE
|
||||||
{ 0x015e, 0x5b }, // Ş LATIN CAPITAL LETTER S WITH CEDILLA
|
{ QChar(0x015e), 0x5b }, // Ş LATIN CAPITAL LETTER S WITH CEDILLA
|
||||||
{ 0x00d6, 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
{ QChar(0x00d6), 0x5c }, // Ö LATIN CAPITAL LETTER O WITH DIAERESIS
|
||||||
{ 0x00c7, 0x5d }, // Ç LATIN CAPITAL LETTER C WITH CEDILLA
|
{ QChar(0x00c7), 0x5d }, // Ç LATIN CAPITAL LETTER C WITH CEDILLA
|
||||||
{ 0x00dc, 0x5e }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
{ QChar(0x00dc), 0x5e }, // Ü LATIN CAPITAL LETTER U WITH DIAERESIS
|
||||||
{ 0x011e, 0x5f }, // Ğ LATIN CAPITAL LETTER G WITH BREVE
|
{ QChar(0x011e), 0x5f }, // Ğ LATIN CAPITAL LETTER G WITH BREVE
|
||||||
{ 0x0131, 0x60 }, // ı LATIN SMALL LETTER DOTLESS I
|
{ QChar(0x0131), 0x60 }, // ı LATIN SMALL LETTER DOTLESS I
|
||||||
{ 0x015f, 0x7b }, // ş LATIN SMALL LETTER S WITH CEDILLA
|
{ QChar(0x015f), 0x7b }, // ş LATIN SMALL LETTER S WITH CEDILLA
|
||||||
{ 0x00f6, 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
{ QChar(0x00f6), 0x7c }, // ö LATIN SMALL LETTER O WITH DIAERESIS
|
||||||
{ 0x00e7, 0x7d }, // ç LATIN SMALL LETTER C WITH CEDILLA
|
{ QChar(0x00e7), 0x7d }, // ç LATIN SMALL LETTER C WITH CEDILLA
|
||||||
{ 0x00fc, 0x7e } // ü LATIN SMALL LETTER U WITH DIAERESIS
|
{ QChar(0x00fc), 0x7e } // ü LATIN SMALL LETTER U WITH DIAERESIS
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
// Native scan codes to toggle mosaic bits - different platforms have different scan codes!
|
// Native scan codes to toggle mosaic bits - different platforms have different scan codes!
|
||||||
// Order is top left, top right, middle left, middle right, bottom left, bottom right,
|
// Order is top left, top right, middle left, middle right, bottom left, bottom right,
|
||||||
// invert, set all, clear all
|
// invert, set all, clear all, dither
|
||||||
|
|
||||||
#ifdef Q_OS_UNIX
|
#ifdef Q_OS_UNIX
|
||||||
static constexpr quint32 mosaicNativeScanCodes[9] = {
|
static constexpr quint32 mosaicNativeScanCodes[10] = {
|
||||||
0x18, 0x19, 0x26, 0x27, 0x34, 0x35, 0x1b, 0x29, 0x36
|
0x18, 0x19, 0x26, 0x27, 0x34, 0x35, 0x1b, 0x29, 0x36, 0x28
|
||||||
};
|
};
|
||||||
#elif defined(Q_OS_WIN)
|
#elif defined(Q_OS_WIN)
|
||||||
static constexpr quint32 mosaicNativeScanCodes[9] = {
|
static constexpr quint32 mosaicNativeScanCodes[10] = {
|
||||||
0x10, 0x11, 0x1e, 0x1f, 0x2c, 0x2d, 0x13, 0x21, 0x2e
|
0x10, 0x11, 0x1e, 0x1f, 0x2c, 0x2d, 0x13, 0x21, 0x2e, 0x20
|
||||||
};
|
};
|
||||||
#else
|
#else
|
||||||
#define QTTM_NONATIVESCANCODES
|
#define QTTM_NONATIVESCANCODES
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -17,16 +17,73 @@
|
|||||||
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <QApplication>
|
||||||
|
#include <QByteArray>
|
||||||
|
#include <QByteArrayList>
|
||||||
|
#include <QClipboard>
|
||||||
|
#include <QMimeData>
|
||||||
|
#include <QRegularExpression>
|
||||||
|
|
||||||
#include "levelonecommands.h"
|
#include "levelonecommands.h"
|
||||||
|
|
||||||
#include "document.h"
|
#include "document.h"
|
||||||
|
#include "keymap.h"
|
||||||
|
|
||||||
TypeCharacterCommand::TypeCharacterCommand(TeletextDocument *teletextDocument, unsigned char newCharacter, bool insertMode, QUndoCommand *parent) : QUndoCommand(parent)
|
LevelOneCommand::LevelOneCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : QUndoCommand(parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
m_teletextDocument = teletextDocument;
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
||||||
m_row = teletextDocument->cursorRow();
|
m_row = teletextDocument->cursorRow();
|
||||||
m_columnStart = m_columnEnd = teletextDocument->cursorColumn();
|
m_column = teletextDocument->cursorColumn();
|
||||||
|
m_firstDo = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
StoreOldCharactersCommand::StoreOldCharactersCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void StoreOldCharactersCommand::storeOldCharacters(int topRow, int leftColumn, int bottomRow, int rightColumn)
|
||||||
|
{
|
||||||
|
for (int r=topRow; r<=bottomRow; r++) {
|
||||||
|
QByteArray rowArray;
|
||||||
|
|
||||||
|
for (int c=leftColumn; c<=rightColumn; c++)
|
||||||
|
// Guard against size of pasted block going beyond last line or column
|
||||||
|
if (r < 25 && c < 40)
|
||||||
|
rowArray.append(m_teletextDocument->currentSubPage()->character(r, c));
|
||||||
|
else
|
||||||
|
// Gone beyond last line or column - store a filler character which we won't see
|
||||||
|
// Not sure if this is really necessary as out-of-bounds access might not occur?
|
||||||
|
rowArray.append(0x7f);
|
||||||
|
|
||||||
|
m_oldCharacters.append(rowArray);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void StoreOldCharactersCommand::retrieveOldCharacters(int topRow, int leftColumn, int bottomRow, int rightColumn)
|
||||||
|
{
|
||||||
|
int arrayR = 0;
|
||||||
|
int arrayC;
|
||||||
|
|
||||||
|
for (int r=topRow; r<=bottomRow; r++) {
|
||||||
|
arrayC = 0;
|
||||||
|
for (int c=leftColumn; c<=rightColumn; c++)
|
||||||
|
// Guard against size of pasted block going beyond last line or column
|
||||||
|
if (r < 25 && c < 40) {
|
||||||
|
m_teletextDocument->currentSubPage()->setCharacter(r, c, m_oldCharacters[arrayR].at(arrayC));
|
||||||
|
|
||||||
|
arrayC++;
|
||||||
|
}
|
||||||
|
|
||||||
|
arrayR++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TypeCharacterCommand::TypeCharacterCommand(TeletextDocument *teletextDocument, unsigned char newCharacter, bool insertMode, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
m_columnStart = m_columnEnd = m_column;
|
||||||
m_newCharacter = newCharacter;
|
m_newCharacter = newCharacter;
|
||||||
m_insertMode = insertMode;
|
m_insertMode = insertMode;
|
||||||
|
|
||||||
@@ -37,7 +94,6 @@ TypeCharacterCommand::TypeCharacterCommand(TeletextDocument *teletextDocument, u
|
|||||||
setText(QObject::tr("insert character"));
|
setText(QObject::tr("insert character"));
|
||||||
else
|
else
|
||||||
setText(QObject::tr("overwrite character"));
|
setText(QObject::tr("overwrite character"));
|
||||||
m_firstDo = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCharacterCommand::redo()
|
void TypeCharacterCommand::redo()
|
||||||
@@ -59,7 +115,7 @@ void TypeCharacterCommand::redo()
|
|||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_columnEnd);
|
m_teletextDocument->moveCursor(m_row, m_columnEnd);
|
||||||
m_teletextDocument->cursorRight();
|
m_teletextDocument->cursorRight();
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeCharacterCommand::undo()
|
void TypeCharacterCommand::undo()
|
||||||
@@ -70,7 +126,7 @@ void TypeCharacterCommand::undo()
|
|||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_oldRowContents[c]);
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_oldRowContents[c]);
|
||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_columnStart);
|
m_teletextDocument->moveCursor(m_row, m_columnStart);
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TypeCharacterCommand::mergeWith(const QUndoCommand *command)
|
bool TypeCharacterCommand::mergeWith(const QUndoCommand *command)
|
||||||
@@ -89,15 +145,13 @@ bool TypeCharacterCommand::mergeWith(const QUndoCommand *command)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
ToggleMosaicBitCommand::ToggleMosaicBitCommand(TeletextDocument *teletextDocument, unsigned char bitToToggle, QUndoCommand *parent) : QUndoCommand(parent)
|
ToggleMosaicBitCommand::ToggleMosaicBitCommand(TeletextDocument *teletextDocument, unsigned char bitToToggle, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
|
||||||
m_row = teletextDocument->cursorRow();
|
|
||||||
m_column = teletextDocument->cursorColumn();
|
|
||||||
m_oldCharacter = teletextDocument->currentSubPage()->character(m_row, m_column);
|
m_oldCharacter = teletextDocument->currentSubPage()->character(m_row, m_column);
|
||||||
if (bitToToggle == 0x20 || bitToToggle == 0x7f)
|
if (bitToToggle == 0x20 || bitToToggle == 0x7f)
|
||||||
m_newCharacter = bitToToggle;
|
m_newCharacter = bitToToggle;
|
||||||
|
else if (bitToToggle == 0x66)
|
||||||
|
m_newCharacter = (m_row & 1) ? 0x66 : 0x39;
|
||||||
else
|
else
|
||||||
m_newCharacter = m_oldCharacter ^ bitToToggle;
|
m_newCharacter = m_oldCharacter ^ bitToToggle;
|
||||||
|
|
||||||
@@ -111,7 +165,7 @@ void ToggleMosaicBitCommand::redo()
|
|||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, m_column, m_newCharacter);
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, m_column, m_newCharacter);
|
||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_column);
|
m_teletextDocument->moveCursor(m_row, m_column);
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ToggleMosaicBitCommand::undo()
|
void ToggleMosaicBitCommand::undo()
|
||||||
@@ -120,7 +174,7 @@ void ToggleMosaicBitCommand::undo()
|
|||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, m_column, m_oldCharacter);
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, m_column, m_oldCharacter);
|
||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_column);
|
m_teletextDocument->moveCursor(m_row, m_column);
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ToggleMosaicBitCommand::mergeWith(const QUndoCommand *command)
|
bool ToggleMosaicBitCommand::mergeWith(const QUndoCommand *command)
|
||||||
@@ -134,12 +188,10 @@ bool ToggleMosaicBitCommand::mergeWith(const QUndoCommand *command)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
BackspaceKeyCommand::BackspaceKeyCommand(TeletextDocument *teletextDocument, bool insertMode, QUndoCommand *parent) : QUndoCommand(parent)
|
BackspaceKeyCommand::BackspaceKeyCommand(TeletextDocument *teletextDocument, bool insertMode, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
m_columnStart = m_column - 1;
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
|
||||||
m_row = teletextDocument->cursorRow();
|
|
||||||
m_columnStart = teletextDocument->cursorColumn()-1;
|
|
||||||
if (m_columnStart == -1) {
|
if (m_columnStart == -1) {
|
||||||
m_columnStart = 39;
|
m_columnStart = 39;
|
||||||
if (--m_row == 0)
|
if (--m_row == 0)
|
||||||
@@ -152,7 +204,6 @@ BackspaceKeyCommand::BackspaceKeyCommand(TeletextDocument *teletextDocument, boo
|
|||||||
m_oldRowContents[c] = m_newRowContents[c] = m_teletextDocument->currentSubPage()->character(m_row, c);
|
m_oldRowContents[c] = m_newRowContents[c] = m_teletextDocument->currentSubPage()->character(m_row, c);
|
||||||
|
|
||||||
setText(QObject::tr("backspace"));
|
setText(QObject::tr("backspace"));
|
||||||
m_firstDo = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void BackspaceKeyCommand::redo()
|
void BackspaceKeyCommand::redo()
|
||||||
@@ -175,7 +226,7 @@ void BackspaceKeyCommand::redo()
|
|||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_newRowContents[c]);
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_newRowContents[c]);
|
||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_columnEnd);
|
m_teletextDocument->moveCursor(m_row, m_columnEnd);
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void BackspaceKeyCommand::undo()
|
void BackspaceKeyCommand::undo()
|
||||||
@@ -187,7 +238,7 @@ void BackspaceKeyCommand::undo()
|
|||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_columnStart);
|
m_teletextDocument->moveCursor(m_row, m_columnStart);
|
||||||
m_teletextDocument->cursorRight();
|
m_teletextDocument->cursorRight();
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool BackspaceKeyCommand::mergeWith(const QUndoCommand *command)
|
bool BackspaceKeyCommand::mergeWith(const QUndoCommand *command)
|
||||||
@@ -208,13 +259,8 @@ bool BackspaceKeyCommand::mergeWith(const QUndoCommand *command)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
DeleteKeyCommand::DeleteKeyCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : QUndoCommand(parent)
|
DeleteKeyCommand::DeleteKeyCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
|
||||||
m_row = teletextDocument->cursorRow();
|
|
||||||
m_column = teletextDocument->cursorColumn();
|
|
||||||
|
|
||||||
for (int c=0; c<40; c++)
|
for (int c=0; c<40; c++)
|
||||||
m_oldRowContents[c] = m_newRowContents[c] = m_teletextDocument->currentSubPage()->character(m_row, c);
|
m_oldRowContents[c] = m_newRowContents[c] = m_teletextDocument->currentSubPage()->character(m_row, c);
|
||||||
|
|
||||||
@@ -234,7 +280,7 @@ void DeleteKeyCommand::redo()
|
|||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_newRowContents[c]);
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_newRowContents[c]);
|
||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_column);
|
m_teletextDocument->moveCursor(m_row, m_column);
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeleteKeyCommand::undo()
|
void DeleteKeyCommand::undo()
|
||||||
@@ -245,7 +291,7 @@ void DeleteKeyCommand::undo()
|
|||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_oldRowContents[c]);
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_oldRowContents[c]);
|
||||||
|
|
||||||
m_teletextDocument->moveCursor(m_row, m_column);
|
m_teletextDocument->moveCursor(m_row, m_column);
|
||||||
emit m_teletextDocument->contentsChange(m_row);
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DeleteKeyCommand::mergeWith(const QUndoCommand *command)
|
bool DeleteKeyCommand::mergeWith(const QUndoCommand *command)
|
||||||
@@ -262,11 +308,8 @@ bool DeleteKeyCommand::mergeWith(const QUndoCommand *command)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
InsertRowCommand::InsertRowCommand(TeletextDocument *teletextDocument, bool copyRow, QUndoCommand *parent) : QUndoCommand(parent)
|
InsertRowCommand::InsertRowCommand(TeletextDocument *teletextDocument, bool copyRow, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
|
||||||
m_row = teletextDocument->cursorRow();
|
|
||||||
m_copyRow = copyRow;
|
m_copyRow = copyRow;
|
||||||
|
|
||||||
if (m_copyRow)
|
if (m_copyRow)
|
||||||
@@ -291,7 +334,7 @@ void InsertRowCommand::redo()
|
|||||||
for (int c=0; c<40; c++)
|
for (int c=0; c<40; c++)
|
||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, ' ');
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, ' ');
|
||||||
|
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void InsertRowCommand::undo()
|
void InsertRowCommand::undo()
|
||||||
@@ -306,16 +349,12 @@ void InsertRowCommand::undo()
|
|||||||
for (int c=0; c<40; c++)
|
for (int c=0; c<40; c++)
|
||||||
m_teletextDocument->currentSubPage()->setCharacter(23, c, m_deletedBottomRow[c]);
|
m_teletextDocument->currentSubPage()->setCharacter(23, c, m_deletedBottomRow[c]);
|
||||||
|
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
DeleteRowCommand::DeleteRowCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : QUndoCommand(parent)
|
DeleteRowCommand::DeleteRowCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
|
||||||
m_row = teletextDocument->cursorRow();
|
|
||||||
|
|
||||||
setText(QObject::tr("delete row"));
|
setText(QObject::tr("delete row"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -335,7 +374,7 @@ void DeleteRowCommand::redo()
|
|||||||
for (int c=0; c<40; c++)
|
for (int c=0; c<40; c++)
|
||||||
m_teletextDocument->currentSubPage()->setCharacter(blankingRow, c, ' ');
|
m_teletextDocument->currentSubPage()->setCharacter(blankingRow, c, ' ');
|
||||||
|
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeleteRowCommand::undo()
|
void DeleteRowCommand::undo()
|
||||||
@@ -350,14 +389,308 @@ void DeleteRowCommand::undo()
|
|||||||
for (int c=0; c<40; c++)
|
for (int c=0; c<40; c++)
|
||||||
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_deletedRow[c]);
|
m_teletextDocument->currentSubPage()->setCharacter(m_row, c, m_deletedRow[c]);
|
||||||
|
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
InsertSubPageCommand::InsertSubPageCommand(TeletextDocument *teletextDocument, bool afterCurrentSubPage, bool copySubPage, QUndoCommand *parent) : QUndoCommand(parent)
|
#ifndef QT_NO_CLIPBOARD
|
||||||
|
CutCommand::CutCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : StoreOldCharactersCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
m_selectionTopRow = m_teletextDocument->selectionTopRow();
|
||||||
m_newSubPageIndex = teletextDocument->currentSubPageIndex()+afterCurrentSubPage;
|
m_selectionBottomRow = m_teletextDocument->selectionBottomRow();
|
||||||
|
m_selectionLeftColumn = m_teletextDocument->selectionLeftColumn();
|
||||||
|
m_selectionRightColumn = m_teletextDocument->selectionRightColumn();
|
||||||
|
|
||||||
|
m_selectionCornerRow = m_teletextDocument->selectionCornerRow();
|
||||||
|
m_selectionCornerColumn = m_teletextDocument->selectionCornerColumn();
|
||||||
|
|
||||||
|
storeOldCharacters(m_selectionTopRow, m_selectionLeftColumn, m_selectionBottomRow, m_selectionRightColumn);
|
||||||
|
|
||||||
|
setText(QObject::tr("cut"));
|
||||||
|
}
|
||||||
|
|
||||||
|
void CutCommand::redo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
|
||||||
|
for (int r=m_selectionTopRow; r<=m_selectionBottomRow; r++) {
|
||||||
|
for (int c=m_selectionLeftColumn; c<=m_selectionRightColumn; c++)
|
||||||
|
m_teletextDocument->currentSubPage()->setCharacter(r, c, 0x20);
|
||||||
|
}
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CutCommand::undo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
|
||||||
|
retrieveOldCharacters(m_selectionTopRow, m_selectionLeftColumn, m_selectionBottomRow, m_selectionRightColumn);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
|
||||||
|
m_teletextDocument->setSelectionCorner(m_selectionCornerRow, m_selectionCornerColumn);
|
||||||
|
m_teletextDocument->moveCursor(m_row, m_column, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
PasteCommand::PasteCommand(TeletextDocument *teletextDocument, int pageCharSet, QUndoCommand *parent) : StoreOldCharactersCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
const QClipboard *clipboard = QApplication::clipboard();
|
||||||
|
const QMimeData *mimeData = clipboard->mimeData();
|
||||||
|
QByteArray nativeData;
|
||||||
|
|
||||||
|
m_selectionActive = m_teletextDocument->selectionActive();
|
||||||
|
if (m_selectionActive) {
|
||||||
|
m_selectionCornerRow = m_teletextDocument->selectionCornerRow();
|
||||||
|
m_selectionCornerColumn = m_teletextDocument->selectionCornerColumn();
|
||||||
|
m_pasteTopRow = m_teletextDocument->selectionTopRow();
|
||||||
|
m_pasteBottomRow = m_teletextDocument->selectionBottomRow();
|
||||||
|
m_pasteLeftColumn = m_teletextDocument->selectionLeftColumn();
|
||||||
|
m_pasteRightColumn = m_teletextDocument->selectionRightColumn();
|
||||||
|
} else {
|
||||||
|
m_pasteTopRow = m_row;
|
||||||
|
m_pasteLeftColumn = m_column;
|
||||||
|
// m_pasteBottomRow and m_pasteRightColumn will be filled in later
|
||||||
|
// when the size of the clipboard data is known
|
||||||
|
}
|
||||||
|
|
||||||
|
// Zero size here represents invalid or empty clipboard data
|
||||||
|
m_clipboardDataHeight = m_clipboardDataWidth = 0;
|
||||||
|
|
||||||
|
// Try to get something from the clipboard
|
||||||
|
// FIXME is this a correct "custom" mime type? Or should we use vnd?
|
||||||
|
nativeData = mimeData->data("application/x-teletext");
|
||||||
|
if (nativeData.size() > 2) {
|
||||||
|
// Native clipboard data: we put it there ourselves
|
||||||
|
m_plainText = false;
|
||||||
|
m_clipboardDataHeight = nativeData.at(0);
|
||||||
|
m_clipboardDataWidth = nativeData.at(1);
|
||||||
|
|
||||||
|
// Guard against invalid dimensions or total size not matching stated dimensions
|
||||||
|
if (m_clipboardDataHeight > 0 && m_clipboardDataWidth > 0 && m_clipboardDataHeight <= 25 && m_clipboardDataWidth <= 40 && nativeData.size() == m_clipboardDataHeight * m_clipboardDataWidth + 2)
|
||||||
|
for (int r=0; r<m_clipboardDataHeight; r++)
|
||||||
|
m_pastingCharacters.append(nativeData.mid(2 + r * m_clipboardDataWidth, m_clipboardDataWidth));
|
||||||
|
else {
|
||||||
|
// Invalidate
|
||||||
|
m_clipboardDataHeight = m_clipboardDataWidth = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!m_selectionActive) {
|
||||||
|
m_pasteBottomRow = m_row + m_clipboardDataHeight - 1;
|
||||||
|
m_pasteRightColumn = m_column + m_clipboardDataWidth - 1;
|
||||||
|
}
|
||||||
|
} else if (mimeData->hasText()) {
|
||||||
|
// Plain text
|
||||||
|
m_plainText = true;
|
||||||
|
|
||||||
|
const int rightColumn = m_selectionActive ? m_pasteRightColumn : 39;
|
||||||
|
|
||||||
|
// Parse line-feeds in the clipboard data
|
||||||
|
QStringList plainTextData = mimeData->text().split(QRegularExpression("\n|\r\n|\r"));
|
||||||
|
|
||||||
|
// "if" statement will be false if clipboard data is a single line of text
|
||||||
|
// that will fit from the cursor position
|
||||||
|
if (plainTextData.size() != 1 || m_pasteLeftColumn + plainTextData.at(0).size() - 1 > rightColumn) {
|
||||||
|
bool wrappingNeeded = false;
|
||||||
|
|
||||||
|
if (!m_selectionActive) {
|
||||||
|
// If selection is NOT active, use the full width of the page to paste.
|
||||||
|
// The second and subsequent lines will start at column 1, unless the
|
||||||
|
// cursor is explicitly on column 0.
|
||||||
|
if (m_pasteLeftColumn != 0)
|
||||||
|
m_pasteLeftColumn = 1;
|
||||||
|
|
||||||
|
// Check if first word in the first line will fit from the cursor position
|
||||||
|
bool firstWordFits = true;
|
||||||
|
const int firstSpace = plainTextData.at(0).indexOf(' ');
|
||||||
|
|
||||||
|
if (firstSpace == -1 && m_column + plainTextData.at(0).size() > 40)
|
||||||
|
firstWordFits = false; // Only one word in first line, and it won't fit
|
||||||
|
else if (m_column + firstSpace > 40)
|
||||||
|
firstWordFits = false; // First word in first line won't fit
|
||||||
|
|
||||||
|
// If the first word WILL fit at the cursor position, pad the first line
|
||||||
|
// to match the cursor position using null characters.
|
||||||
|
// In the QString null characters represent character cells in the
|
||||||
|
// pasting rectangle that won't overwrite what's on the page.
|
||||||
|
// If the first word WON'T fit, start pasting at the beginning of the next row.
|
||||||
|
if (firstWordFits)
|
||||||
|
plainTextData[0] = QString(m_column-m_pasteLeftColumn, QChar::Null) + plainTextData.at(0);
|
||||||
|
else if (m_pasteTopRow < 24)
|
||||||
|
m_pasteTopRow++;
|
||||||
|
else
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const int pasteWidth = rightColumn - m_pasteLeftColumn + 1;
|
||||||
|
|
||||||
|
// Find out if we need to word-wrap
|
||||||
|
for (int i=0; i<plainTextData.size(); i++)
|
||||||
|
if (plainTextData.at(i).size() > pasteWidth) {
|
||||||
|
wrappingNeeded = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (wrappingNeeded) {
|
||||||
|
QStringList wrappedText;
|
||||||
|
|
||||||
|
for (int i=0; i<plainTextData.size(); i++) {
|
||||||
|
// Split this line into individual words
|
||||||
|
QStringList lineWords = plainTextData.at(i).split(' ');
|
||||||
|
|
||||||
|
// If there's any words which are too long to fit,
|
||||||
|
// break them across multiple lines
|
||||||
|
for (int j=0; j<lineWords.size(); j++)
|
||||||
|
if (lineWords.at(j).size() > pasteWidth) {
|
||||||
|
lineWords.insert(j+1, lineWords.at(j).mid(pasteWidth));
|
||||||
|
lineWords[j].truncate(pasteWidth);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now reassemble the words into lines that will fit
|
||||||
|
QString currentLine = lineWords.at(0);
|
||||||
|
|
||||||
|
for (int j=1; j<lineWords.size(); j++)
|
||||||
|
if (currentLine.size() + 1 + lineWords.at(j).size() <= pasteWidth)
|
||||||
|
currentLine.append(' ' + lineWords.at(j));
|
||||||
|
else {
|
||||||
|
wrappedText.append(currentLine);
|
||||||
|
currentLine = lineWords.at(j);
|
||||||
|
}
|
||||||
|
|
||||||
|
wrappedText.append(currentLine);
|
||||||
|
}
|
||||||
|
plainTextData.swap(wrappedText);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m_clipboardDataHeight = plainTextData.size();
|
||||||
|
m_clipboardDataWidth = 0;
|
||||||
|
|
||||||
|
// Convert the unicode clipboard text into teletext bytes matching the current Level 1
|
||||||
|
// character set of this page
|
||||||
|
for (int r=0; r<m_clipboardDataHeight; r++) {
|
||||||
|
m_pastingCharacters.append(QByteArray());
|
||||||
|
for (int c=0; c<plainTextData.at(r).size(); c++) {
|
||||||
|
char convertedChar;
|
||||||
|
const QChar charToConvert = plainTextData.at(r).at(c);
|
||||||
|
|
||||||
|
// Map a null character in the QString to 0xff (or -1)
|
||||||
|
// In the QByteArray 0xff bytes represent character cells in the pasting rectangle
|
||||||
|
// that won't overwrite what's on the page
|
||||||
|
if (charToConvert == QChar::Null)
|
||||||
|
convertedChar = -1;
|
||||||
|
else if (charToConvert >= QChar(0x01) && charToConvert <= QChar(0x1f))
|
||||||
|
convertedChar = ' ';
|
||||||
|
else if (keymapping[pageCharSet].contains(charToConvert))
|
||||||
|
// Remapped character or non-Latin character converted successfully
|
||||||
|
convertedChar = keymapping[pageCharSet].value(charToConvert);
|
||||||
|
else {
|
||||||
|
// Either a Latin character or non-Latin character that can't be converted
|
||||||
|
// See if it's a Latin character
|
||||||
|
convertedChar = charToConvert.toLatin1();
|
||||||
|
if (convertedChar <= 0)
|
||||||
|
// Couldn't convert - make it a block character so it doesn't need to be inserted-between later on
|
||||||
|
convertedChar = 0x7f;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_pastingCharacters[r].append(convertedChar);
|
||||||
|
}
|
||||||
|
m_clipboardDataWidth = qMax(m_pastingCharacters.at(r).size(), m_clipboardDataWidth);
|
||||||
|
}
|
||||||
|
// Pad the end of short lines with spaces to make a box
|
||||||
|
for (int r=0; r<m_clipboardDataHeight; r++)
|
||||||
|
m_pastingCharacters[r] = m_pastingCharacters.at(r).leftJustified(m_clipboardDataWidth);
|
||||||
|
|
||||||
|
if (!m_selectionActive) {
|
||||||
|
m_pasteBottomRow = m_pasteTopRow + m_clipboardDataHeight - 1;
|
||||||
|
m_pasteRightColumn = m_pasteLeftColumn + m_clipboardDataWidth - 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_clipboardDataWidth == 0 || m_clipboardDataHeight == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
storeOldCharacters(m_pasteTopRow, m_pasteLeftColumn, m_pasteBottomRow, m_pasteRightColumn);
|
||||||
|
|
||||||
|
setText(QObject::tr("paste"));
|
||||||
|
}
|
||||||
|
|
||||||
|
void PasteCommand::redo()
|
||||||
|
{
|
||||||
|
if (m_clipboardDataWidth == 0 || m_clipboardDataHeight == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
|
||||||
|
int arrayR = 0;
|
||||||
|
int arrayC;
|
||||||
|
|
||||||
|
for (int r=m_pasteTopRow; r<=m_pasteBottomRow; r++) {
|
||||||
|
arrayC = 0;
|
||||||
|
for (int c=m_pasteLeftColumn; c<=m_pasteRightColumn; c++)
|
||||||
|
// Guard against size of pasted block going beyond last line or column
|
||||||
|
if (r < 25 && c < 40) {
|
||||||
|
// Check for 0xff bytes using "-1"
|
||||||
|
// gcc complains about "comparision always true due to limited range"
|
||||||
|
if (m_pastingCharacters.at(arrayR).at(arrayC) != -1)
|
||||||
|
m_teletextDocument->currentSubPage()->setCharacter(r, c, m_pastingCharacters.at(arrayR).at(arrayC));
|
||||||
|
|
||||||
|
arrayC++;
|
||||||
|
|
||||||
|
// If paste area is wider than clipboard data, repeat the pattern
|
||||||
|
// if it wasn't plain text
|
||||||
|
if (arrayC == m_clipboardDataWidth) {
|
||||||
|
if (!m_plainText)
|
||||||
|
arrayC = 0;
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
arrayR++;
|
||||||
|
// If paste area is taller than clipboard data, repeat the pattern
|
||||||
|
// if it wasn't plain text
|
||||||
|
if (arrayR == m_clipboardDataHeight) {
|
||||||
|
if (!m_plainText)
|
||||||
|
arrayR = 0;
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
|
||||||
|
if (m_selectionActive) {
|
||||||
|
m_teletextDocument->setSelectionCorner(m_selectionCornerRow, m_selectionCornerColumn);
|
||||||
|
m_teletextDocument->moveCursor(m_row, m_column, true);
|
||||||
|
} else {
|
||||||
|
m_teletextDocument->moveCursor(m_row, qMin(m_column+m_clipboardDataWidth-1, 39));
|
||||||
|
m_teletextDocument->cursorRight();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void PasteCommand::undo()
|
||||||
|
{
|
||||||
|
if (m_clipboardDataWidth == 0 || m_clipboardDataHeight == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
|
||||||
|
retrieveOldCharacters(m_pasteTopRow, m_pasteLeftColumn, m_pasteBottomRow, m_pasteRightColumn);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
|
||||||
|
if (!m_selectionActive)
|
||||||
|
m_teletextDocument->moveCursor(m_row, m_column);
|
||||||
|
}
|
||||||
|
#endif // !QT_NO_CLIPBOARD
|
||||||
|
|
||||||
|
|
||||||
|
InsertSubPageCommand::InsertSubPageCommand(TeletextDocument *teletextDocument, bool afterCurrentSubPage, bool copySubPage, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
m_newSubPageIndex = m_subPageIndex + afterCurrentSubPage;
|
||||||
m_copySubPage = copySubPage;
|
m_copySubPage = copySubPage;
|
||||||
|
|
||||||
setText(QObject::tr("insert subpage"));
|
setText(QObject::tr("insert subpage"));
|
||||||
@@ -380,30 +713,181 @@ void InsertSubPageCommand::undo()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
DeleteSubPageCommand::DeleteSubPageCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : QUndoCommand(parent)
|
DeleteSubPageCommand::DeleteSubPageCommand(TeletextDocument *teletextDocument, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
|
||||||
m_subPageToDelete = teletextDocument->currentSubPageIndex();
|
|
||||||
setText(QObject::tr("delete subpage"));
|
setText(QObject::tr("delete subpage"));
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeleteSubPageCommand::redo()
|
void DeleteSubPageCommand::redo()
|
||||||
{
|
{
|
||||||
m_teletextDocument->deleteSubPageToRecycle(m_subPageToDelete);
|
m_teletextDocument->deleteSubPageToRecycle(m_subPageIndex);
|
||||||
m_teletextDocument->selectSubPageIndex(qMin(m_subPageToDelete, m_teletextDocument->numberOfSubPages()-1), true);
|
m_teletextDocument->selectSubPageIndex(qMin(m_subPageIndex, m_teletextDocument->numberOfSubPages()-1), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeleteSubPageCommand::undo()
|
void DeleteSubPageCommand::undo()
|
||||||
{
|
{
|
||||||
m_teletextDocument->unDeleteSubPageFromRecycle(m_subPageToDelete);
|
m_teletextDocument->unDeleteSubPageFromRecycle(m_subPageIndex);
|
||||||
m_teletextDocument->selectSubPageIndex(m_subPageToDelete, true);
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
SetColourCommand::SetColourCommand(TeletextDocument *teletextDocument, int colourIndex, int newColour, QUndoCommand *parent) : QUndoCommand(parent)
|
SetFullScreenColourCommand::SetFullScreenColourCommand(TeletextDocument *teletextDocument, int newColour, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
m_oldColour = teletextDocument->currentSubPage()->defaultScreenColour();
|
||||||
|
m_newColour = newColour;
|
||||||
|
|
||||||
|
setText(QObject::tr("full screen colour"));
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetFullScreenColourCommand::redo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setDefaultScreenColour(m_newColour);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetFullScreenColourCommand::undo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setDefaultScreenColour(m_oldColour);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SetFullScreenColourCommand::mergeWith(const QUndoCommand *command)
|
||||||
|
{
|
||||||
|
const SetFullScreenColourCommand *newerCommand = static_cast<const SetFullScreenColourCommand *>(command);
|
||||||
|
|
||||||
|
if (m_subPageIndex != newerCommand->m_subPageIndex)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
m_newColour = newerCommand->m_newColour;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SetFullRowColourCommand::SetFullRowColourCommand(TeletextDocument *teletextDocument, int newColour, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
m_oldColour = teletextDocument->currentSubPage()->defaultRowColour();
|
||||||
|
m_newColour = newColour;
|
||||||
|
|
||||||
|
setText(QObject::tr("full row colour"));
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetFullRowColourCommand::redo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setDefaultRowColour(m_newColour);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetFullRowColourCommand::undo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setDefaultRowColour(m_oldColour);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SetFullRowColourCommand::mergeWith(const QUndoCommand *command)
|
||||||
|
{
|
||||||
|
const SetFullRowColourCommand *newerCommand = static_cast<const SetFullRowColourCommand *>(command);
|
||||||
|
|
||||||
|
if (m_subPageIndex != newerCommand->m_subPageIndex)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
m_newColour = newerCommand->m_newColour;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SetCLUTRemapCommand::SetCLUTRemapCommand(TeletextDocument *teletextDocument, int newMap, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
m_oldMap = teletextDocument->currentSubPage()->colourTableRemap();
|
||||||
|
m_newMap = newMap;
|
||||||
|
|
||||||
|
setText(QObject::tr("CLUT remapping"));
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetCLUTRemapCommand::redo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setColourTableRemap(m_newMap);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetCLUTRemapCommand::undo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setColourTableRemap(m_oldMap);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SetCLUTRemapCommand::mergeWith(const QUndoCommand *command)
|
||||||
|
{
|
||||||
|
const SetCLUTRemapCommand *newerCommand = static_cast<const SetCLUTRemapCommand *>(command);
|
||||||
|
|
||||||
|
if (m_subPageIndex != newerCommand->m_subPageIndex)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
m_newMap = newerCommand->m_newMap;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SetBlackBackgroundSubstCommand::SetBlackBackgroundSubstCommand(TeletextDocument *teletextDocument, bool newSub, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
|
{
|
||||||
|
m_oldSub = teletextDocument->currentSubPage()->blackBackgroundSubst();
|
||||||
|
m_newSub = newSub;
|
||||||
|
|
||||||
|
setText(QObject::tr("black background substitution"));
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetBlackBackgroundSubstCommand::redo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setBlackBackgroundSubst(m_newSub);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetBlackBackgroundSubstCommand::undo()
|
||||||
|
{
|
||||||
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
m_teletextDocument->currentSubPage()->setBlackBackgroundSubst(m_oldSub);
|
||||||
|
|
||||||
|
emit m_teletextDocument->contentsChanged();
|
||||||
|
emit m_teletextDocument->pageOptionsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SetBlackBackgroundSubstCommand::mergeWith(const QUndoCommand *command)
|
||||||
|
{
|
||||||
|
const SetBlackBackgroundSubstCommand *newerCommand = static_cast<const SetBlackBackgroundSubstCommand *>(command);
|
||||||
|
|
||||||
|
if (m_subPageIndex != newerCommand->m_subPageIndex)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
setObsolete(true);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SetColourCommand::SetColourCommand(TeletextDocument *teletextDocument, int colourIndex, int newColour, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
|
||||||
m_colourIndex = colourIndex;
|
m_colourIndex = colourIndex;
|
||||||
m_oldColour = teletextDocument->currentSubPage()->CLUT(colourIndex);
|
m_oldColour = teletextDocument->currentSubPage()->CLUT(colourIndex);
|
||||||
m_newColour = newColour;
|
m_newColour = newColour;
|
||||||
@@ -418,7 +902,7 @@ void SetColourCommand::redo()
|
|||||||
m_teletextDocument->currentSubPage()->setCLUT(m_colourIndex, m_newColour);
|
m_teletextDocument->currentSubPage()->setCLUT(m_colourIndex, m_newColour);
|
||||||
|
|
||||||
emit m_teletextDocument->colourChanged(m_colourIndex);
|
emit m_teletextDocument->colourChanged(m_colourIndex);
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetColourCommand::undo()
|
void SetColourCommand::undo()
|
||||||
@@ -427,14 +911,12 @@ void SetColourCommand::undo()
|
|||||||
m_teletextDocument->currentSubPage()->setCLUT(m_colourIndex, m_oldColour);
|
m_teletextDocument->currentSubPage()->setCLUT(m_colourIndex, m_oldColour);
|
||||||
|
|
||||||
emit m_teletextDocument->colourChanged(m_colourIndex);
|
emit m_teletextDocument->colourChanged(m_colourIndex);
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
ResetCLUTCommand::ResetCLUTCommand(TeletextDocument *teletextDocument, int colourTable, QUndoCommand *parent) : QUndoCommand(parent)
|
ResetCLUTCommand::ResetCLUTCommand(TeletextDocument *teletextDocument, int colourTable, QUndoCommand *parent) : LevelOneCommand(teletextDocument, parent)
|
||||||
{
|
{
|
||||||
m_teletextDocument = teletextDocument;
|
|
||||||
m_subPageIndex = teletextDocument->currentSubPageIndex();
|
|
||||||
m_colourTable = colourTable;
|
m_colourTable = colourTable;
|
||||||
for (int i=m_colourTable*8; i<m_colourTable*8+8; i++)
|
for (int i=m_colourTable*8; i<m_colourTable*8+8; i++)
|
||||||
m_oldColourEntry[i&7] = teletextDocument->currentSubPage()->CLUT(i);
|
m_oldColourEntry[i&7] = teletextDocument->currentSubPage()->CLUT(i);
|
||||||
@@ -450,7 +932,7 @@ void ResetCLUTCommand::redo()
|
|||||||
emit m_teletextDocument->colourChanged(i);
|
emit m_teletextDocument->colourChanged(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ResetCLUTCommand::undo()
|
void ResetCLUTCommand::undo()
|
||||||
@@ -461,5 +943,5 @@ void ResetCLUTCommand::undo()
|
|||||||
emit m_teletextDocument->colourChanged(i);
|
emit m_teletextDocument->colourChanged(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
emit m_teletextDocument->refreshNeeded();
|
emit m_teletextDocument->contentsChanged();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -20,164 +20,266 @@
|
|||||||
#ifndef LEVELONECOMMANDS_H
|
#ifndef LEVELONECOMMANDS_H
|
||||||
#define LEVELONECOMMANDS_H
|
#define LEVELONECOMMANDS_H
|
||||||
|
|
||||||
|
#include <QByteArrayList>
|
||||||
#include <QUndoCommand>
|
#include <QUndoCommand>
|
||||||
|
|
||||||
#include "document.h"
|
#include "document.h"
|
||||||
|
|
||||||
class TypeCharacterCommand : public QUndoCommand
|
class LevelOneCommand : public QUndoCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LevelOneCommand(TeletextDocument *teletextDocument, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
TeletextDocument *m_teletextDocument;
|
||||||
|
int m_subPageIndex, m_row, m_column;
|
||||||
|
bool m_firstDo;
|
||||||
|
};
|
||||||
|
|
||||||
|
class StoreOldCharactersCommand : public LevelOneCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
StoreOldCharactersCommand(TeletextDocument *teletextDocument, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void storeOldCharacters(int topRow, int leftColumn, int bottomRow, int rightColumn);
|
||||||
|
void retrieveOldCharacters(int topRow, int leftColumn, int bottomRow, int rightColumn);
|
||||||
|
|
||||||
|
QByteArrayList m_oldCharacters;
|
||||||
|
};
|
||||||
|
|
||||||
|
class TypeCharacterCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
enum { Id = 101 };
|
enum { Id = 101 };
|
||||||
|
|
||||||
TypeCharacterCommand(TeletextDocument *, unsigned char, bool, QUndoCommand *parent = 0);
|
TypeCharacterCommand(TeletextDocument *teletextDocument, unsigned char newCharacter, bool insertMode, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
bool mergeWith(const QUndoCommand *) override;
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
int id() const override { return Id; }
|
int id() const override { return Id; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
unsigned char m_newCharacter, m_oldRowContents[40], m_newRowContents[40];
|
unsigned char m_newCharacter, m_oldRowContents[40], m_newRowContents[40];
|
||||||
int m_subPageIndex, m_row, m_columnStart, m_columnEnd;
|
int m_columnStart, m_columnEnd;
|
||||||
bool m_firstDo, m_insertMode;
|
bool m_insertMode;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ToggleMosaicBitCommand : public QUndoCommand
|
class ToggleMosaicBitCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
enum { Id = 102 };
|
enum { Id = 102 };
|
||||||
|
|
||||||
ToggleMosaicBitCommand(TeletextDocument *, unsigned char, QUndoCommand *parent = 0);
|
ToggleMosaicBitCommand(TeletextDocument *teletextDocument, unsigned char bitToToggle, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
bool mergeWith(const QUndoCommand *) override;
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
int id() const override { return Id; }
|
int id() const override { return Id; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
unsigned char m_oldCharacter, m_newCharacter;
|
unsigned char m_oldCharacter, m_newCharacter;
|
||||||
int m_subPageIndex, m_row, m_column;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class BackspaceKeyCommand : public QUndoCommand
|
class BackspaceKeyCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
enum { Id = 103 };
|
enum { Id = 103 };
|
||||||
|
|
||||||
BackspaceKeyCommand(TeletextDocument *, bool insertMode, QUndoCommand *parent = 0);
|
BackspaceKeyCommand(TeletextDocument *teletextDocument, bool insertMode, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
bool mergeWith(const QUndoCommand *) override;
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
int id() const override { return Id; }
|
int id() const override { return Id; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
unsigned char m_oldRowContents[40], m_newRowContents[40];
|
unsigned char m_oldRowContents[40], m_newRowContents[40];
|
||||||
int m_subPageIndex, m_row, m_columnStart, m_columnEnd;
|
int m_columnStart, m_columnEnd;
|
||||||
bool m_firstDo, m_insertMode;
|
bool m_insertMode;
|
||||||
};
|
};
|
||||||
|
|
||||||
class DeleteKeyCommand : public QUndoCommand
|
class DeleteKeyCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
enum { Id = 104 };
|
enum { Id = 104 };
|
||||||
|
|
||||||
DeleteKeyCommand(TeletextDocument *, QUndoCommand *parent = 0);
|
DeleteKeyCommand(TeletextDocument *teletextDocument, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
bool mergeWith(const QUndoCommand *) override;
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
int id() const override { return Id; }
|
int id() const override { return Id; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
unsigned char m_oldRowContents[40], m_newRowContents[40];
|
unsigned char m_oldRowContents[40], m_newRowContents[40];
|
||||||
int m_subPageIndex, m_row, m_column;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class InsertSubPageCommand : public QUndoCommand
|
class InsertSubPageCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
InsertSubPageCommand(TeletextDocument *, bool, bool, QUndoCommand *parent = 0);
|
InsertSubPageCommand(TeletextDocument *teletextDocument, bool afterCurrentSubPage, bool copySubPage, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
int m_newSubPageIndex;
|
int m_newSubPageIndex;
|
||||||
bool m_copySubPage;
|
bool m_copySubPage;
|
||||||
};
|
};
|
||||||
|
|
||||||
class DeleteSubPageCommand : public QUndoCommand
|
class DeleteSubPageCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
DeleteSubPageCommand(TeletextDocument *, QUndoCommand *parent = 0);
|
DeleteSubPageCommand(TeletextDocument *teletextDocument, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
|
|
||||||
private:
|
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
int m_subPageToDelete;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class InsertRowCommand : public QUndoCommand
|
class InsertRowCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
InsertRowCommand(TeletextDocument *, bool, QUndoCommand *parent = 0);
|
InsertRowCommand(TeletextDocument *teletextDocument, bool copyRow, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
int m_subPageIndex, m_row;
|
|
||||||
bool m_copyRow;
|
bool m_copyRow;
|
||||||
unsigned char m_deletedBottomRow[40];
|
unsigned char m_deletedBottomRow[40];
|
||||||
};
|
};
|
||||||
|
|
||||||
class DeleteRowCommand : public QUndoCommand
|
class DeleteRowCommand : public LevelOneCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
DeleteRowCommand(TeletextDocument *, QUndoCommand *parent = 0);
|
DeleteRowCommand(TeletextDocument *teletextDocument, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
|
||||||
int m_subPageIndex, m_row;
|
|
||||||
unsigned char m_deletedRow[40];
|
unsigned char m_deletedRow[40];
|
||||||
};
|
};
|
||||||
|
|
||||||
class SetColourCommand : public QUndoCommand
|
#ifndef QT_NO_CLIPBOARD
|
||||||
|
class CutCommand : public StoreOldCharactersCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
SetColourCommand(TeletextDocument *, int, int, QUndoCommand *parent = 0);
|
CutCommand(TeletextDocument *teletextDocument, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
int m_selectionTopRow, m_selectionBottomRow, m_selectionLeftColumn, m_selectionRightColumn;
|
||||||
int m_subPageIndex, m_colourIndex, m_oldColour, m_newColour;
|
int m_selectionCornerRow, m_selectionCornerColumn;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ResetCLUTCommand : public QUndoCommand
|
class PasteCommand : public StoreOldCharactersCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
ResetCLUTCommand(TeletextDocument *, int, QUndoCommand *parent = 0);
|
PasteCommand(TeletextDocument *teletextDocument, int pageCharSet, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument *m_teletextDocument;
|
QByteArrayList m_pastingCharacters;
|
||||||
int m_subPageIndex, m_colourTable;
|
int m_pasteTopRow, m_pasteBottomRow, m_pasteLeftColumn, m_pasteRightColumn;
|
||||||
|
int m_clipboardDataHeight, m_clipboardDataWidth;
|
||||||
|
int m_selectionCornerRow, m_selectionCornerColumn;
|
||||||
|
bool m_selectionActive, m_plainText;
|
||||||
|
};
|
||||||
|
#endif // !QT_NO_CLIPBOARD
|
||||||
|
|
||||||
|
class SetFullScreenColourCommand : public LevelOneCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum { Id = 105 };
|
||||||
|
|
||||||
|
SetFullScreenColourCommand(TeletextDocument *teletextDocument, int newColour, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
void redo() override;
|
||||||
|
void undo() override;
|
||||||
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
|
int id() const override { return Id; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_oldColour, m_newColour;
|
||||||
|
};
|
||||||
|
|
||||||
|
class SetFullRowColourCommand : public LevelOneCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum { Id = 106 };
|
||||||
|
|
||||||
|
SetFullRowColourCommand(TeletextDocument *teletextDocument, int newColour, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
void redo() override;
|
||||||
|
void undo() override;
|
||||||
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
|
int id() const override { return Id; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_oldColour, m_newColour;
|
||||||
|
};
|
||||||
|
|
||||||
|
class SetCLUTRemapCommand : public LevelOneCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum { Id = 107 };
|
||||||
|
|
||||||
|
SetCLUTRemapCommand(TeletextDocument *teletextDocument, int newMap, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
void redo() override;
|
||||||
|
void undo() override;
|
||||||
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
|
int id() const override { return Id; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_oldMap, m_newMap;
|
||||||
|
};
|
||||||
|
|
||||||
|
class SetBlackBackgroundSubstCommand : public LevelOneCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum { Id = 108 };
|
||||||
|
|
||||||
|
SetBlackBackgroundSubstCommand(TeletextDocument *teletextDocument, bool newSub, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
void redo() override;
|
||||||
|
void undo() override;
|
||||||
|
bool mergeWith(const QUndoCommand *command) override;
|
||||||
|
int id() const override { return Id; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_oldSub, m_newSub;
|
||||||
|
};
|
||||||
|
|
||||||
|
class SetColourCommand : public LevelOneCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
SetColourCommand(TeletextDocument *teletextDocument, int colourIndex, int newColour, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
void redo() override;
|
||||||
|
void undo() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_colourIndex, m_oldColour, m_newColour;
|
||||||
|
};
|
||||||
|
|
||||||
|
class ResetCLUTCommand : public LevelOneCommand
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
ResetCLUTCommand(TeletextDocument *teletextDocument, int colourTable, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
|
void redo() override;
|
||||||
|
void undo() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_colourTable;
|
||||||
int m_oldColourEntry[8];
|
int m_oldColourEntry[8];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -25,23 +25,25 @@
|
|||||||
|
|
||||||
#include "levelonepage.h"
|
#include "levelonepage.h"
|
||||||
|
|
||||||
|
#include "x26triplets.h"
|
||||||
|
|
||||||
LevelOnePage::LevelOnePage()
|
LevelOnePage::LevelOnePage()
|
||||||
{
|
{
|
||||||
m_enhancements.reserve(208);
|
m_enhancements.reserve(maxEnhancements());
|
||||||
clearPage();
|
clearPage();
|
||||||
}
|
}
|
||||||
|
|
||||||
LevelOnePage::LevelOnePage(const PageBase &other)
|
LevelOnePage::LevelOnePage(const PageBase &other)
|
||||||
{
|
{
|
||||||
m_enhancements.reserve(208);
|
m_enhancements.reserve(maxEnhancements());
|
||||||
clearPage();
|
clearPage();
|
||||||
|
|
||||||
for (int i=0; i<26; i++)
|
for (int i=0; i<26; i++)
|
||||||
if (other.packetNeeded(i))
|
if (other.packetExists(i))
|
||||||
setPacket(i, other.packet(i));
|
setPacket(i, other.packet(i));
|
||||||
for (int i=26; i<30; i++)
|
for (int i=26; i<30; i++)
|
||||||
for (int j=0; j<16; j++)
|
for (int j=0; j<16; j++)
|
||||||
if (other.packetNeeded(i, j))
|
if (other.packetExists(i, j))
|
||||||
setPacket(i, j, other.packet(i));
|
setPacket(i, j, other.packet(i));
|
||||||
|
|
||||||
for (int i=PageBase::C4ErasePage; i<=PageBase::C14NOS; i++)
|
for (int i=PageBase::C4ErasePage; i<=PageBase::C14NOS; i++)
|
||||||
@@ -62,7 +64,7 @@ void LevelOnePage::clearPage()
|
|||||||
m_fastTextLink[i] = { 0x0ff, 0x3f7f };
|
m_fastTextLink[i] = { 0x0ff, 0x3f7f };
|
||||||
|
|
||||||
/* m_subPageNumber = 0x0000; */
|
/* m_subPageNumber = 0x0000; */
|
||||||
m_cycleValue = 8;
|
m_cycleValue = 20;
|
||||||
m_cycleType = CTseconds;
|
m_cycleType = CTseconds;
|
||||||
m_defaultCharSet = 0;
|
m_defaultCharSet = 0;
|
||||||
m_defaultNOS = 0;
|
m_defaultNOS = 0;
|
||||||
@@ -125,12 +127,12 @@ QByteArray LevelOnePage::packet(int packetNumber, int designationCode) const
|
|||||||
result[i*6+1] = m_fastTextLink[i].pageNumber & 0x00f;
|
result[i*6+1] = m_fastTextLink[i].pageNumber & 0x00f;
|
||||||
result[i*6+2] = (m_fastTextLink[i].pageNumber & 0x0f0) >> 4;
|
result[i*6+2] = (m_fastTextLink[i].pageNumber & 0x0f0) >> 4;
|
||||||
result[i*6+3] = m_fastTextLink[i].subPageNumber & 0x000f;
|
result[i*6+3] = m_fastTextLink[i].subPageNumber & 0x000f;
|
||||||
result[i*6+4] = ((m_fastTextLink[i].subPageNumber & 0x0070) >> 4) | ((m_fastTextLink[i].pageNumber & 0x100) >> 8);
|
result[i*6+4] = ((m_fastTextLink[i].subPageNumber & 0x0070) >> 4) | ((m_fastTextLink[i].pageNumber & 0x100) >> 5);
|
||||||
result[i*6+5] = (m_fastTextLink[i].subPageNumber & 0x0f00) >> 8;
|
result[i*6+5] = (m_fastTextLink[i].subPageNumber & 0x0f00) >> 8;
|
||||||
result[i*6+6] = ((m_fastTextLink[i].subPageNumber & 0x3000) >> 12) | ((m_fastTextLink[i].pageNumber & 0x600) >> 7);
|
result[i*6+6] = ((m_fastTextLink[i].subPageNumber & 0x3000) >> 12) | ((m_fastTextLink[i].pageNumber & 0x600) >> 7);
|
||||||
}
|
}
|
||||||
result[43] = 0xf;
|
result[37] = 0xf;
|
||||||
result[44] = result[45] = 0;
|
result[38] = result[39] = 0;
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -254,7 +256,7 @@ bool LevelOnePage::setPacket(int packetNumber, int designationCode, QByteArray p
|
|||||||
return PageBase::setPacket(packetNumber, designationCode, packetContents);
|
return PageBase::setPacket(packetNumber, designationCode, packetContents);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool LevelOnePage::packetNeeded(int packetNumber) const
|
bool LevelOnePage::packetExists(int packetNumber) const
|
||||||
{
|
{
|
||||||
if (packetNumber <= 24) {
|
if (packetNumber <= 24) {
|
||||||
for (int c=0; c<40; c++)
|
for (int c=0; c<40; c++)
|
||||||
@@ -263,10 +265,10 @@ bool LevelOnePage::packetNeeded(int packetNumber) const
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return PageBase::packetNeeded(packetNumber);
|
return PageBase::packetExists(packetNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool LevelOnePage::packetNeeded(int packetNumber, int designationCode) const
|
bool LevelOnePage::packetExists(int packetNumber, int designationCode) const
|
||||||
{
|
{
|
||||||
if (packetNumber == 26)
|
if (packetNumber == 26)
|
||||||
return packetFromEnhancementListNeeded(designationCode);
|
return packetFromEnhancementListNeeded(designationCode);
|
||||||
@@ -295,10 +297,10 @@ bool LevelOnePage::packetNeeded(int packetNumber, int designationCode) const
|
|||||||
return !isPaletteDefault(16, 31);
|
return !isPaletteDefault(16, 31);
|
||||||
}
|
}
|
||||||
if (designationCode == 4)
|
if (designationCode == 4)
|
||||||
return !isPaletteDefault(0,15);
|
return !isPaletteDefault(0, 15);
|
||||||
}
|
}
|
||||||
|
|
||||||
return PageBase::packetNeeded(packetNumber, designationCode);
|
return PageBase::packetExists(packetNumber, designationCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool LevelOnePage::controlBit(int bitNumber) const
|
bool LevelOnePage::controlBit(int bitNumber) const
|
||||||
@@ -381,6 +383,9 @@ QColor LevelOnePage::CLUTtoQColor(int index, int renderLevel) const
|
|||||||
{
|
{
|
||||||
int colour12Bit = CLUT(index, renderLevel);
|
int colour12Bit = CLUT(index, renderLevel);
|
||||||
|
|
||||||
|
if (index == 8)
|
||||||
|
return QColor(Qt::transparent);
|
||||||
|
|
||||||
return QColor(((colour12Bit & 0xf00) >> 8) * 17, ((colour12Bit & 0x0f0) >> 4) * 17, (colour12Bit & 0x00f) * 17);
|
return QColor(((colour12Bit & 0xf00) >> 8) * 17, ((colour12Bit & 0x0f0) >> 4) * 17, (colour12Bit & 0x00f) * 17);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -391,7 +396,7 @@ bool LevelOnePage::isPaletteDefault(int colour) const
|
|||||||
|
|
||||||
bool LevelOnePage::isPaletteDefault(int fromColour, int toColour) const
|
bool LevelOnePage::isPaletteDefault(int fromColour, int toColour) const
|
||||||
{
|
{
|
||||||
for (int i=fromColour; i<toColour; i++)
|
for (int i=fromColour; i<=toColour; i++)
|
||||||
if (m_CLUT[i] != m_defaultCLUT[i])
|
if (m_CLUT[i] != m_defaultCLUT[i])
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@@ -400,17 +405,21 @@ bool LevelOnePage::isPaletteDefault(int fromColour, int toColour) const
|
|||||||
|
|
||||||
int LevelOnePage::levelRequired() const
|
int LevelOnePage::levelRequired() const
|
||||||
{
|
{
|
||||||
|
// X/28/4 present i.e. CLUTs 0 or 1 redefined - Level 3.5
|
||||||
if (!isPaletteDefault(0, 15))
|
if (!isPaletteDefault(0, 15))
|
||||||
return 3;
|
return 3;
|
||||||
|
|
||||||
// TODO Check for X/28/1 for DCLUT for mode 1-3 DRCS characters - return 3
|
// TODO Check for X/28/1 for DCLUT for mode 1-3 DRCS characters - return 3
|
||||||
|
|
||||||
int levelSeen = (!isPaletteDefault(16,31) || m_leftSidePanelDisplayed || m_rightSidePanelDisplayed || m_defaultScreenColour !=0 || m_defaultRowColour !=0 || m_blackBackgroundSubst || m_colourTableRemap !=0 || m_defaultCharSet != 0 || m_secondCharSet != 0xf) ? 2 : 0;
|
// Assume Level 2.5 if any X/28 page enhancements are present, otherwise assume Level 1
|
||||||
|
int levelSeen = (!isPaletteDefault(16, 31) || m_leftSidePanelDisplayed || m_rightSidePanelDisplayed || m_defaultScreenColour !=0 || m_defaultRowColour !=0 || m_blackBackgroundSubst || m_colourTableRemap !=0 || m_defaultCharSet != 0 || m_secondCharSet != 0xf) ? 2 : 0;
|
||||||
|
|
||||||
|
// If there's no X/26 triplets, exit here as Level 1 or 2.5
|
||||||
if (m_enhancements.isEmpty())
|
if (m_enhancements.isEmpty())
|
||||||
return levelSeen;
|
return levelSeen;
|
||||||
|
|
||||||
for (int i=0; i<m_enhancements.size(); i++) {
|
for (int i=0; i<m_enhancements.size(); i++) {
|
||||||
|
// Font style - Level 3.5 only triplet
|
||||||
if (m_enhancements.at(i).modeExt() == 0x2e) // Font style
|
if (m_enhancements.at(i).modeExt() == 0x2e) // Font style
|
||||||
return 3;
|
return 3;
|
||||||
|
|
||||||
@@ -422,36 +431,52 @@ int LevelOnePage::levelRequired() const
|
|||||||
case 0x1f: // Termination
|
case 0x1f: // Termination
|
||||||
case 0x22: // G3 character @ Level 1.5
|
case 0x22: // G3 character @ Level 1.5
|
||||||
case 0x2f: // G2 character
|
case 0x2f: // G2 character
|
||||||
case 0x30 ... 0x3f: // G0 character with diacritical
|
levelSeen = 1;
|
||||||
levelSeen = qMax(levelSeen, 1);
|
|
||||||
break;
|
break;
|
||||||
|
default:
|
||||||
|
if (m_enhancements.at(i).modeExt() >= 0x30 && m_enhancements.at(i).modeExt() <= 0x3f)
|
||||||
|
// G0 character with diacritical
|
||||||
|
levelSeen = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (levelSeen < 2)
|
if (levelSeen < 2)
|
||||||
switch (m_enhancements.at(i).modeExt()) {
|
switch (m_enhancements.at(i).modeExt()) {
|
||||||
// Check for Level 2.5 triplets
|
// Check for Level 2.5 triplets
|
||||||
case 0x00: // Full screen colour
|
case 0x00: // Full screen colour
|
||||||
case 0x01: // Full row colour
|
case 0x01: // Full row colour
|
||||||
case 0x10 ... 0x13: // Origin Modifer and Object Invocation
|
case 0x10: // Origin Modifier
|
||||||
case 0x15 ... 0x17: // Object Definition
|
case 0x11: // Invoke Active Object
|
||||||
// Check if Object Defition is required only at Level 3.5
|
case 0x12: // Invoke Adaptive Object
|
||||||
if ((m_enhancements.at(i).address() & 0x18) == 0x10)
|
case 0x13: // Invoke Passive Object
|
||||||
return 3;
|
case 0x15: // Define Active Object
|
||||||
else
|
case 0x16: // Define Adaptive Object
|
||||||
levelSeen = qMax(levelSeen, 2);
|
case 0x17: // Define Passive Object
|
||||||
break;
|
|
||||||
case 0x18: // DRCS Mode
|
case 0x18: // DRCS Mode
|
||||||
// Check if DRCS is required only at Level 3.5
|
|
||||||
if ((m_enhancements.at(i).data() & 0x30) == 0x20)
|
|
||||||
return 3;
|
|
||||||
else
|
|
||||||
levelSeen = qMax(levelSeen, 2);
|
|
||||||
break;
|
|
||||||
case 0x20: // Foreground colour
|
case 0x20: // Foreground colour
|
||||||
case 0x21: // G1 character
|
case 0x21: // G1 character
|
||||||
case 0x23: // Background colour
|
case 0x23: // Background colour
|
||||||
case 0x27 ... 0x29: // Flash functions, G0 and G2 charset designation, G0 character @ Level 2.5
|
case 0x27: // Flash functions
|
||||||
case 0x2b ... 0x2d: // G3 character @ Level 2.5, display attributes, DRCS character
|
case 0x28: // G0 and G2 charset designation
|
||||||
levelSeen = qMax(levelSeen, 2);
|
case 0x29: // G0 character @ Level 2.5
|
||||||
|
case 0x2b: // G3 character @ Level 2.5
|
||||||
|
case 0x2c: // Display attributes
|
||||||
|
case 0x2d: // DRCS character
|
||||||
|
levelSeen = 2;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (levelSeen == 2)
|
||||||
|
switch (m_enhancements.at(i).modeExt()) {
|
||||||
|
// Check for triplets with "required at Level 3.5 only" parameters
|
||||||
|
case 0x15: // Define Active Object
|
||||||
|
case 0x16: // Define Adaptive Object
|
||||||
|
case 0x17: // Define Passive Object
|
||||||
|
if ((m_enhancements.at(i).address() & 0x18) == 0x10)
|
||||||
|
return 3;
|
||||||
|
break;
|
||||||
|
case 0x18: // DRCS Mode
|
||||||
|
if ((m_enhancements.at(i).data() & 0x30) == 0x20)
|
||||||
|
return 3;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -36,71 +36,73 @@ public:
|
|||||||
enum CycleTypeEnum { CTcycles, CTseconds };
|
enum CycleTypeEnum { CTcycles, CTseconds };
|
||||||
|
|
||||||
LevelOnePage();
|
LevelOnePage();
|
||||||
LevelOnePage(const PageBase &);
|
LevelOnePage(const PageBase &other);
|
||||||
|
|
||||||
bool isEmpty() const override;
|
bool isEmpty() const override;
|
||||||
|
|
||||||
QByteArray packet(int) const override;
|
QByteArray packet(int packetNumber) const override;
|
||||||
QByteArray packet(int, int) const override;
|
QByteArray packet(int packetNumber, int designationCode) const override;
|
||||||
bool packetNeeded(int) const override;
|
bool packetExists(int packetNumber) const override;
|
||||||
bool packetNeeded(int, int) const override;
|
bool packetExists(int packetNumber, int designationCode) const override;
|
||||||
bool setPacket(int, QByteArray) override;
|
bool setPacket(int packetNumber, QByteArray packetContents) override;
|
||||||
bool setPacket(int, int, QByteArray) override;
|
bool setPacket(int packetNumber, int designationCode, QByteArray packetContents) override;
|
||||||
|
|
||||||
bool controlBit(int bitNumber) const override;
|
bool controlBit(int bitNumber) const override;
|
||||||
bool setControlBit(int, bool) override;
|
bool setControlBit(int bitNumber, bool active) override;
|
||||||
|
|
||||||
void clearPage();
|
void clearPage();
|
||||||
|
|
||||||
|
int maxEnhancements() const { return 208; };
|
||||||
|
|
||||||
/* void setSubPageNumber(int); */
|
/* void setSubPageNumber(int); */
|
||||||
int cycleValue() const { return m_cycleValue; };
|
int cycleValue() const { return m_cycleValue; };
|
||||||
void setCycleValue(int);
|
void setCycleValue(int newValue);
|
||||||
CycleTypeEnum cycleType() const { return m_cycleType; };
|
CycleTypeEnum cycleType() const { return m_cycleType; };
|
||||||
void setCycleType(CycleTypeEnum);
|
void setCycleType(CycleTypeEnum newType);
|
||||||
int defaultCharSet() const { return m_defaultCharSet; }
|
int defaultCharSet() const { return m_defaultCharSet; }
|
||||||
void setDefaultCharSet(int);
|
void setDefaultCharSet(int newDefaultCharSet);
|
||||||
int defaultNOS() const { return m_defaultNOS; }
|
int defaultNOS() const { return m_defaultNOS; }
|
||||||
void setDefaultNOS(int);
|
void setDefaultNOS(int defaultNOS);
|
||||||
int secondCharSet() const { return m_secondCharSet; }
|
int secondCharSet() const { return m_secondCharSet; }
|
||||||
void setSecondCharSet(int);
|
void setSecondCharSet(int newSecondCharSet);
|
||||||
int secondNOS() const { return m_secondNOS; }
|
int secondNOS() const { return m_secondNOS; }
|
||||||
void setSecondNOS(int);
|
void setSecondNOS(int newSecondNOS);
|
||||||
unsigned char character(int row, int column) const { return m_level1Page[row][column]; }
|
unsigned char character(int row, int column) const { return m_level1Page[row][column]; }
|
||||||
void setCharacter(int, int, unsigned char);
|
void setCharacter(int row, int column, unsigned char newCharacter);
|
||||||
int defaultScreenColour() const { return m_defaultScreenColour; }
|
int defaultScreenColour() const { return m_defaultScreenColour; }
|
||||||
void setDefaultScreenColour(int);
|
void setDefaultScreenColour(int newDefaultScreenColour);
|
||||||
int defaultRowColour() const { return m_defaultRowColour; }
|
int defaultRowColour() const { return m_defaultRowColour; }
|
||||||
void setDefaultRowColour(int);
|
void setDefaultRowColour(int newDefaultRowColour);
|
||||||
int colourTableRemap() const { return m_colourTableRemap; }
|
int colourTableRemap() const { return m_colourTableRemap; }
|
||||||
void setColourTableRemap(int);
|
void setColourTableRemap(int newColourTableRemap);
|
||||||
bool blackBackgroundSubst() const { return m_blackBackgroundSubst; }
|
bool blackBackgroundSubst() const { return m_blackBackgroundSubst; }
|
||||||
void setBlackBackgroundSubst(bool);
|
void setBlackBackgroundSubst(bool newBlackBackgroundSubst);
|
||||||
int CLUT(int index, int renderLevel=3) const;
|
int CLUT(int index, int renderLevel=3) const;
|
||||||
void setCLUT(int, int);
|
void setCLUT(int index, int newColour);
|
||||||
QColor CLUTtoQColor(int index, int renderlevel=3) const;
|
QColor CLUTtoQColor(int index, int renderlevel=3) const;
|
||||||
bool isPaletteDefault(int) const;
|
bool isPaletteDefault(int colour) const;
|
||||||
bool isPaletteDefault(int, int) const;
|
bool isPaletteDefault(int fromColour, int toColour) const;
|
||||||
int levelRequired() const;
|
int levelRequired() const;
|
||||||
bool leftSidePanelDisplayed() const { return m_leftSidePanelDisplayed; }
|
bool leftSidePanelDisplayed() const { return m_leftSidePanelDisplayed; }
|
||||||
void setLeftSidePanelDisplayed(bool);
|
void setLeftSidePanelDisplayed(bool newLeftSidePanelDisplayed);
|
||||||
bool rightSidePanelDisplayed() const { return m_rightSidePanelDisplayed; }
|
bool rightSidePanelDisplayed() const { return m_rightSidePanelDisplayed; }
|
||||||
void setRightSidePanelDisplayed(bool);
|
void setRightSidePanelDisplayed(bool newRightSidePanelDisplayed);
|
||||||
int sidePanelColumns() const { return m_sidePanelColumns; }
|
int sidePanelColumns() const { return m_sidePanelColumns; }
|
||||||
void setSidePanelColumns(int);
|
void setSidePanelColumns(int newSidePanelColumns);
|
||||||
bool sidePanelStatusL25() const { return m_sidePanelStatusL25; }
|
bool sidePanelStatusL25() const { return m_sidePanelStatusL25; }
|
||||||
void setSidePanelStatusL25(bool);
|
void setSidePanelStatusL25(bool newSidePanelStatusL25);
|
||||||
int fastTextLinkPageNumber(int linkNumber) const { return m_fastTextLink[linkNumber].pageNumber; }
|
int fastTextLinkPageNumber(int linkNumber) const { return m_fastTextLink[linkNumber].pageNumber; }
|
||||||
void setFastTextLinkPageNumber(int, int);
|
void setFastTextLinkPageNumber(int linkNumber, int pageNumber);
|
||||||
int composeLinkFunction(int linkNumber) const { return m_composeLink[linkNumber].function; }
|
int composeLinkFunction(int linkNumber) const { return m_composeLink[linkNumber].function; }
|
||||||
void setComposeLinkFunction(int, int);
|
void setComposeLinkFunction(int linkNumber, int newFunction);
|
||||||
bool composeLinkLevel2p5(int linkNumber) const { return m_composeLink[linkNumber].level2p5; }
|
bool composeLinkLevel2p5(int linkNumber) const { return m_composeLink[linkNumber].level2p5; }
|
||||||
void setComposeLinkLevel2p5(int, bool);
|
void setComposeLinkLevel2p5(int linkNumber, bool newRequired);
|
||||||
bool composeLinkLevel3p5(int linkNumber) const { return m_composeLink[linkNumber].level3p5; }
|
bool composeLinkLevel3p5(int linkNumber) const { return m_composeLink[linkNumber].level3p5; }
|
||||||
void setComposeLinkLevel3p5(int, bool);
|
void setComposeLinkLevel3p5(int linkNumber, bool newRequired);
|
||||||
int composeLinkPageNumber(int linkNumber) const { return m_composeLink[linkNumber].pageNumber; }
|
int composeLinkPageNumber(int linkNumber) const { return m_composeLink[linkNumber].pageNumber; }
|
||||||
void setComposeLinkPageNumber(int, int);
|
void setComposeLinkPageNumber(int linkNumber, int newPageNumber);
|
||||||
int composeLinkSubPageCodes(int linkNumber) const { return m_composeLink[linkNumber].subPageCodes; }
|
int composeLinkSubPageCodes(int linkNumber) const { return m_composeLink[linkNumber].subPageCodes; }
|
||||||
void setComposeLinkSubPageCodes(int, int);
|
void setComposeLinkSubPageCodes(int linkNumber, int newSubPageCodes);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
unsigned char m_level1Page[25][40];
|
unsigned char m_level1Page[25][40];
|
||||||
|
|||||||
467
loadsave.cpp
467
loadsave.cpp
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -20,12 +20,14 @@
|
|||||||
#include "loadsave.h"
|
#include "loadsave.h"
|
||||||
|
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
|
#include <QDataStream>
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
#include <QSaveFile>
|
#include <QSaveFile>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
|
|
||||||
#include "document.h"
|
#include "document.h"
|
||||||
|
#include "hamming.h"
|
||||||
#include "levelonepage.h"
|
#include "levelonepage.h"
|
||||||
#include "pagebase.h"
|
#include "pagebase.h"
|
||||||
|
|
||||||
@@ -52,7 +54,7 @@ void loadTTI(QFile *inFile, TeletextDocument *document)
|
|||||||
document->insertSubPage(document->numberOfSubPages(), false);
|
document->insertSubPage(document->numberOfSubPages(), false);
|
||||||
loadingPage = document->subPage(document->numberOfSubPages()-1);
|
loadingPage = document->subPage(document->numberOfSubPages()-1);
|
||||||
} else {
|
} else {
|
||||||
document->setPageNumber(inLine.mid(3,3));
|
document->setPageNumberFromString(inLine.mid(3,3));
|
||||||
firstSubPageAlreadyFound = true;
|
firstSubPageAlreadyFound = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -146,6 +148,12 @@ void loadTTI(QFile *inFile, TeletextDocument *document)
|
|||||||
}
|
}
|
||||||
for (int i=1; i<=39; i++)
|
for (int i=1; i<=39; i++)
|
||||||
inLine[i] = inLine.at(i) & 0x3f;
|
inLine[i] = inLine.at(i) & 0x3f;
|
||||||
|
// Import M/29 whole-magazine packets as X/28 per-page packets
|
||||||
|
if (lineNumber == 29) {
|
||||||
|
if ((document->pageNumber() & 0xff) != 0xff)
|
||||||
|
qDebug("M/29/%d packet found, but page number is not xFF!", designationCode);
|
||||||
|
lineNumber = 28;
|
||||||
|
}
|
||||||
loadingPage->setPacket(lineNumber, designationCode, inLine);
|
loadingPage->setPacket(lineNumber, designationCode, inLine);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -160,6 +168,198 @@ void loadTTI(QFile *inFile, TeletextDocument *document)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void importT42(QFile *inFile, TeletextDocument *document)
|
||||||
|
{
|
||||||
|
unsigned char inLine[42];
|
||||||
|
int readMagazineNumber, readPacketNumber;
|
||||||
|
int foundMagazineNumber = -1;
|
||||||
|
int foundPageNumber = -1;
|
||||||
|
bool firstPacket0Found = false;
|
||||||
|
bool pageBodyPacketsFound = false;
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
if (inFile->read((char *)inLine, 42) != 42)
|
||||||
|
// Reached end of .t42 file, or less than 42 bytes left
|
||||||
|
break;
|
||||||
|
|
||||||
|
// Magazine and packet numbers
|
||||||
|
inLine[0] = hamming_8_4_decode[inLine[0]];
|
||||||
|
inLine[1] = hamming_8_4_decode[inLine[1]];
|
||||||
|
if (inLine[0] == 0xff || inLine[1] == 0xff)
|
||||||
|
// Error decoding magazine or packet number
|
||||||
|
continue;
|
||||||
|
readMagazineNumber = inLine[0] & 0x07;
|
||||||
|
readPacketNumber = (inLine[0] >> 3) | (inLine[1] << 1);
|
||||||
|
|
||||||
|
if (readPacketNumber == 0) {
|
||||||
|
// Hamming decode page number, subcodes and control bits
|
||||||
|
for (int i=2; i<10; i++)
|
||||||
|
inLine[i] = hamming_8_4_decode[inLine[i]];
|
||||||
|
// See if the page number is valid
|
||||||
|
if (inLine[2] == 0xff || inLine[3] == 0xff)
|
||||||
|
// Error decoding page number
|
||||||
|
continue;
|
||||||
|
|
||||||
|
const int readPageNumber = (inLine[3] << 4) | inLine[2];
|
||||||
|
|
||||||
|
if (readPageNumber == 0xff)
|
||||||
|
// Time filling header
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// A second or subsequent X/0 has been found
|
||||||
|
if (firstPacket0Found) {
|
||||||
|
if (readMagazineNumber != foundMagazineNumber)
|
||||||
|
// Packet from different magazine broadcast in parallel mode
|
||||||
|
continue;
|
||||||
|
if ((readPageNumber == foundPageNumber) && pageBodyPacketsFound)
|
||||||
|
// X/0 with same page number found after page body packets loaded - assume end of page
|
||||||
|
break;
|
||||||
|
if (readPageNumber != foundPageNumber) {
|
||||||
|
// More than one page in .t42 file - end of current page reached
|
||||||
|
qDebug("More than one page in .t42 file");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// Could get here if X/0 with same page number was found with no body packets inbetween
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
// First X/0 found
|
||||||
|
foundMagazineNumber = readMagazineNumber;
|
||||||
|
foundPageNumber = readPageNumber;
|
||||||
|
firstPacket0Found = true;
|
||||||
|
|
||||||
|
if (foundMagazineNumber == 0)
|
||||||
|
document->setPageNumber(0x800 | foundPageNumber);
|
||||||
|
else
|
||||||
|
document->setPageNumber((foundMagazineNumber << 8) | foundPageNumber);
|
||||||
|
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C4ErasePage, inLine[5] & 0x08);
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C5Newsflash, inLine[7] & 0x04);
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C6Subtitle, inLine[7] & 0x08);
|
||||||
|
for (int i=0; i<4; i++)
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C7SuppressHeader+i, inLine[8] & (1 << i));
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C11SerialMagazine, inLine[9] & 0x01);
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C12NOS, inLine[9] & 0x08);
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C13NOS, inLine[9] & 0x04);
|
||||||
|
document->subPage(0)->setControlBit(PageBase::C14NOS, inLine[9] & 0x02);
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// No X/0 has been found yet, keep looking for one
|
||||||
|
if (!firstPacket0Found)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// Disregard whole-magazine packets
|
||||||
|
if (readPacketNumber > 28)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// We get here when a page-body packet belonging to the found X/0 header was found
|
||||||
|
pageBodyPacketsFound = true;
|
||||||
|
|
||||||
|
// At the moment this only loads a Level One Page properly
|
||||||
|
// because it assumes X/1 to X/25 is odd partity
|
||||||
|
if (readPacketNumber < 25) {
|
||||||
|
for (int i=2; i<42; i++)
|
||||||
|
// TODO - obey odd parity?
|
||||||
|
inLine[i] &= 0x7f;
|
||||||
|
document->subPage(0)->setPacket(readPacketNumber, QByteArray((const char *)&inLine[2], 40));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// X/26, X/27 or X/28
|
||||||
|
int readDesignationCode = hamming_8_4_decode[inLine[2]];
|
||||||
|
|
||||||
|
if (readDesignationCode == 0xff)
|
||||||
|
// Error decoding designation code
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (readPacketNumber == 27 && readDesignationCode < 4) {
|
||||||
|
// X/27/0 to X/27/3 for Editorial Linking
|
||||||
|
// Decode Hamming 8/4 on each of the six links, checking for errors on the way
|
||||||
|
for (int i=0; i<6; i++) {
|
||||||
|
bool decodingError = false;
|
||||||
|
const int b = 3 + i*6; // First byte of this link
|
||||||
|
|
||||||
|
for (int j=0; j<6; j++) {
|
||||||
|
inLine[b+j] = hamming_8_4_decode[inLine[b+j]];
|
||||||
|
if (inLine[b+j] == 0xff) {
|
||||||
|
decodingError = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (decodingError) {
|
||||||
|
// Error found in at least one byte of the link
|
||||||
|
// Neutralise the whole link to same magazine, page FF, subcode 3F7F
|
||||||
|
qDebug("X/27/%d link %d decoding error", readDesignationCode, i);
|
||||||
|
inLine[b] = 0xf;
|
||||||
|
inLine[b+1] = 0xf;
|
||||||
|
inLine[b+2] = 0xf;
|
||||||
|
inLine[b+3] = 0x7;
|
||||||
|
inLine[b+4] = 0xf;
|
||||||
|
inLine[b+5] = 0x3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
document->subPage(0)->setPacket(readPacketNumber, readDesignationCode, QByteArray((const char *)&inLine[2], 40));
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// X/26, or X/27/4 to X/27/15, or X/28
|
||||||
|
// Decode Hamming 24/18
|
||||||
|
for (int i=0; i<13; i++) {
|
||||||
|
const int b = 3 + i*3; // First byte of triplet
|
||||||
|
|
||||||
|
const int p0 = inLine[b];
|
||||||
|
const int p1 = inLine[b+1];
|
||||||
|
const int p2 = inLine[b+2];
|
||||||
|
|
||||||
|
unsigned int D1_D4;
|
||||||
|
unsigned int D5_D11;
|
||||||
|
unsigned int D12_D18;
|
||||||
|
unsigned int ABCDEF;
|
||||||
|
int32_t d;
|
||||||
|
|
||||||
|
D1_D4 = hamming_24_18_decode_d1_d4[p0 >> 2];
|
||||||
|
D5_D11 = p1 & 0x7f;
|
||||||
|
D12_D18 = p2 & 0x7f;
|
||||||
|
|
||||||
|
d = D1_D4 | (D5_D11 << 4) | (D12_D18 << 11);
|
||||||
|
|
||||||
|
ABCDEF = (hamming_24_18_parities[0][p0] ^ hamming_24_18_parities[1][p1] ^ hamming_24_18_parities[2][p2]);
|
||||||
|
|
||||||
|
d ^= (int)hamming_24_18_decode_correct[ABCDEF];
|
||||||
|
|
||||||
|
if ((d & 0x80000000) == 0x80000000) {
|
||||||
|
// Error decoding Hamming 24/18
|
||||||
|
qDebug("X/%d/%d triplet %d decoding error", readPacketNumber, readDesignationCode, i);
|
||||||
|
if (readPacketNumber == 26) {
|
||||||
|
// Enhancements packet, set to "dummy" Address 41, Mode 0x1e, Data 0
|
||||||
|
inLine[b] = 41;
|
||||||
|
inLine[b+1] = 0x1e;
|
||||||
|
inLine[b+2] = 0;
|
||||||
|
} else {
|
||||||
|
// Zero out whole decoded triplet, bound to make things go wrong...
|
||||||
|
inLine[b] = 0x00;
|
||||||
|
inLine[b+1] = 0x00;
|
||||||
|
inLine[b+2] = 0x00;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
inLine[b] = d & 0x0003f;
|
||||||
|
inLine[b+1] = (d & 0x00fc0) >> 6;
|
||||||
|
inLine[b+2] = d >> 12;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
document->subPage(0)->setPacket(readPacketNumber, readDesignationCode, QByteArray((const char *)&inLine[2], 40));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!firstPacket0Found)
|
||||||
|
qDebug("No X/0 found");
|
||||||
|
else if (!pageBodyPacketsFound)
|
||||||
|
qDebug("X/0 found, but no page body packets were found");
|
||||||
|
}
|
||||||
|
|
||||||
// Used by saveTTI and HashString
|
// Used by saveTTI and HashString
|
||||||
int controlBitsToPS(PageBase *subPage)
|
int controlBitsToPS(PageBase *subPage)
|
||||||
{
|
{
|
||||||
@@ -182,7 +382,7 @@ void saveTTI(QSaveFile &file, const TeletextDocument &document)
|
|||||||
|
|
||||||
auto write7bitPacket=[&](int packetNumber)
|
auto write7bitPacket=[&](int packetNumber)
|
||||||
{
|
{
|
||||||
if (document.subPage(p)->packetNeeded(packetNumber)) {
|
if (document.subPage(p)->packetExists(packetNumber)) {
|
||||||
QByteArray outLine = document.subPage(p)->packet(packetNumber);
|
QByteArray outLine = document.subPage(p)->packet(packetNumber);
|
||||||
|
|
||||||
outStream << QString("OL,%1,").arg(packetNumber);
|
outStream << QString("OL,%1,").arg(packetNumber);
|
||||||
@@ -193,13 +393,17 @@ void saveTTI(QSaveFile &file, const TeletextDocument &document)
|
|||||||
outLine.insert(c, 0x1b);
|
outLine.insert(c, 0x1b);
|
||||||
c++;
|
c++;
|
||||||
}
|
}
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
outStream << outLine << Qt::endl;
|
outStream << outLine << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << outLine << endl;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
auto writeHammingPacket=[&](int packetNumber, int designationCode=0)
|
auto writeHammingPacket=[&](int packetNumber, int designationCode=0)
|
||||||
{
|
{
|
||||||
if (document.subPage(p)->packetNeeded(packetNumber, designationCode)) {
|
if (document.subPage(p)->packetExists(packetNumber, designationCode)) {
|
||||||
QByteArray outLine = document.subPage(p)->packet(packetNumber, designationCode);
|
QByteArray outLine = document.subPage(p)->packet(packetNumber, designationCode);
|
||||||
|
|
||||||
outStream << QString("OL,%1,").arg(packetNumber);
|
outStream << QString("OL,%1,").arg(packetNumber);
|
||||||
@@ -207,14 +411,26 @@ void saveTTI(QSaveFile &file, const TeletextDocument &document)
|
|||||||
outLine[0] = designationCode | 0x40;
|
outLine[0] = designationCode | 0x40;
|
||||||
for (int c=1; c<outLine.size(); c++)
|
for (int c=1; c<outLine.size(); c++)
|
||||||
outLine[c] = outLine.at(c) | 0x40;
|
outLine[c] = outLine.at(c) | 0x40;
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
outStream << outLine << Qt::endl;
|
outStream << outLine << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << outLine << endl;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
||||||
outStream.setCodec("ISO-8859-1");
|
outStream.setCodec("ISO-8859-1");
|
||||||
|
#else
|
||||||
|
outStream.setEncoding(QStringConverter::Latin1);
|
||||||
|
#endif
|
||||||
|
|
||||||
if (!document.description().isEmpty())
|
if (!document.description().isEmpty())
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
outStream << "DE," << document.description() << Qt::endl;
|
outStream << "DE," << document.description() << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << "DE," << document.description() << endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
// TODO DS and SP commands
|
// TODO DS and SP commands
|
||||||
|
|
||||||
@@ -224,23 +440,50 @@ void saveTTI(QSaveFile &file, const TeletextDocument &document)
|
|||||||
|
|
||||||
for (p=0; p<document.numberOfSubPages(); p++) {
|
for (p=0; p<document.numberOfSubPages(); p++) {
|
||||||
|
|
||||||
outStream << QString("PN,%1%2").arg(document.pageNumber(), 3, 16, QChar('0')).arg(subPageNumber & 0xff, 2, 16, QChar('0')) << Qt::endl;
|
// Page number
|
||||||
|
outStream << QString("PN,%1%2").arg(document.pageNumber(), 3, 16, QChar('0')).arg(subPageNumber & 0xff, 2, 10, QChar('0'));
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Subpage
|
||||||
// Magazine Organisation Table and Magazine Inventory Page don't have subpages
|
// Magazine Organisation Table and Magazine Inventory Page don't have subpages
|
||||||
if (document.pageFunction() != TeletextDocument::PFMOT && document.pageFunction() != TeletextDocument::PFMIP)
|
if (document.pageFunction() != TeletextDocument::PFMOT && document.pageFunction() != TeletextDocument::PFMIP) {
|
||||||
outStream << QString("SC,%1").arg(subPageNumber, 4, 16, QChar('0')) << Qt::endl;
|
outStream << QString("SC,%1").arg(subPageNumber, 4, 10, QChar('0'));
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << endl;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
outStream << QString("PS,%1").arg(0x8000 | controlBitsToPS(document.subPage(p)), 4, 16, QChar('0')) << Qt::endl;
|
// Status bits
|
||||||
|
outStream << QString("PS,%1").arg(0x8000 | controlBitsToPS(document.subPage(p)), 4, 16, QChar('0'));
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Cycle time
|
||||||
if (document.pageFunction() == TeletextDocument::PFLevelOnePage)
|
if (document.pageFunction() == TeletextDocument::PFLevelOnePage)
|
||||||
// Assume that only Level One Pages have configurable cycle times
|
// Assume that only Level One Pages have configurable cycle times
|
||||||
outStream << QString("CT,%1,%2").arg(document.subPage(p)->cycleValue()).arg(document.subPage(p)->cycleType()==LevelOnePage::CTcycles ? 'C' : 'T') << Qt::endl;
|
outStream << QString("CT,%1,%2").arg(document.subPage(p)->cycleValue()).arg(document.subPage(p)->cycleType()==LevelOnePage::CTcycles ? 'C' : 'T');
|
||||||
else
|
else
|
||||||
// X/28/0 specifies page function and coding but the PF command
|
// X/28/0 specifies page function and coding but the PF command
|
||||||
// should make it obvious to a human that this isn't a Level One Page
|
// should make it obvious to a human that this isn't a Level One Page
|
||||||
outStream << QString("PF,%1,%2").arg(document.pageFunction()).arg(document.packetCoding()) << Qt::endl;
|
outStream << QString("PF,%1,%2").arg(document.pageFunction()).arg(document.packetCoding());
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// FastText links
|
||||||
bool writeFLCommand = false;
|
bool writeFLCommand = false;
|
||||||
if (document.pageFunction() == TeletextDocument::PFLevelOnePage && document.subPage(p)->packetNeeded(27,0)) {
|
if (document.pageFunction() == TeletextDocument::PFLevelOnePage && document.subPage(p)->packetExists(27,0)) {
|
||||||
// Subpage has FastText links - if any link to a specific subpage, we need to write X/27/0 as raw
|
// Subpage has FastText links - if any link to a specific subpage, we need to write X/27/0 as raw
|
||||||
// otherwise we write the links as a human-readable FL command later on
|
// otherwise we write the links as a human-readable FL command later on
|
||||||
writeFLCommand = true;
|
writeFLCommand = true;
|
||||||
@@ -252,7 +495,7 @@ void saveTTI(QSaveFile &file, const TeletextDocument &document)
|
|||||||
}*/
|
}*/
|
||||||
}
|
}
|
||||||
|
|
||||||
// X27 then X28 always come first
|
// X/27 then X/28 always come first
|
||||||
for (int i=(writeFLCommand ? 1 : 0); i<16; i++)
|
for (int i=(writeFLCommand ? 1 : 0); i<16; i++)
|
||||||
writeHammingPacket(27, i);
|
writeHammingPacket(27, i);
|
||||||
for (int i=0; i<16; i++)
|
for (int i=0; i<16; i++)
|
||||||
@@ -285,16 +528,208 @@ void saveTTI(QSaveFile &file, const TeletextDocument &document)
|
|||||||
if (i<5)
|
if (i<5)
|
||||||
outStream << ',';
|
outStream << ',';
|
||||||
}
|
}
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
outStream << Qt::endl;
|
outStream << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << endl;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
subPageNumber++;
|
subPageNumber++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void exportM29File(QSaveFile &file, const TeletextDocument &document)
|
||||||
|
{
|
||||||
|
const PageBase &subPage = *document.currentSubPage();
|
||||||
|
QTextStream outStream(&file);
|
||||||
|
|
||||||
|
auto writeM29Packet=[&](int designationCode)
|
||||||
|
{
|
||||||
|
if (subPage.packetExists(28, designationCode)) {
|
||||||
|
QByteArray outLine = subPage.packet(28, designationCode);
|
||||||
|
|
||||||
|
outStream << QString("OL,29,");
|
||||||
|
// TTI stores raw values with bit 6 set, doesn't do Hamming encoding
|
||||||
|
outLine[0] = designationCode | 0x40;
|
||||||
|
for (int c=1; c<outLine.size(); c++)
|
||||||
|
outLine[c] = outLine.at(c) | 0x40;
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << outLine << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << outLine << endl;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
||||||
|
outStream.setCodec("ISO-8859-1");
|
||||||
|
#else
|
||||||
|
outStream.setEncoding(QStringConverter::Latin1);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (!document.description().isEmpty())
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << "DE," << document.description() << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << "DE," << document.description() << endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Force page number to xFF
|
||||||
|
outStream << QString("PN,%1ff00").arg(document.pageNumber() >> 8, 1, 16);
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
outStream << "PS,8000";
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
outStream << Qt::endl;
|
||||||
|
#else
|
||||||
|
outStream << endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
writeM29Packet(0);
|
||||||
|
writeM29Packet(1);
|
||||||
|
writeM29Packet(4);
|
||||||
|
}
|
||||||
|
|
||||||
|
void exportT42File(QSaveFile &file, const TeletextDocument &document)
|
||||||
|
{
|
||||||
|
const PageBase &subPage = *document.currentSubPage();
|
||||||
|
|
||||||
|
QDataStream outStream(&file);
|
||||||
|
// Displayable row header we export as spaces, hence the (odd parity valid) 0x20 init value
|
||||||
|
QByteArray outLine(42, 0x20);
|
||||||
|
int magazineNumber = (document.pageNumber() & 0xf00) >> 8;
|
||||||
|
|
||||||
|
auto write7bitPacket=[&](int packetNumber)
|
||||||
|
{
|
||||||
|
if (subPage.packetExists(packetNumber)) {
|
||||||
|
outLine[0] = hamming_8_4_encode[magazineNumber | ((packetNumber & 0x01) << 3)];
|
||||||
|
outLine[1] = hamming_8_4_encode[packetNumber >> 1];
|
||||||
|
outLine.replace(2, 40, subPage.packet(packetNumber));
|
||||||
|
|
||||||
|
// Odd parity encoding
|
||||||
|
for (int c=0; c<outLine.size(); c++) {
|
||||||
|
char p = outLine.at(c);
|
||||||
|
|
||||||
|
// Recursively divide integer into two equal halves and take their XOR until only 1 bit is left
|
||||||
|
p ^= p >> 4;
|
||||||
|
p ^= p >> 2;
|
||||||
|
p ^= p >> 1;
|
||||||
|
// If last bit left is 0 then it started with an even number of bits, so do the odd parity
|
||||||
|
if (!(p & 1))
|
||||||
|
outLine[c] = outLine.at(c) | 0x80;
|
||||||
|
}
|
||||||
|
outStream.writeRawData(outLine.constData(), 42);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
auto writeHamming8_4Packet=[&](int packetNumber, int designationCode=0)
|
||||||
|
{
|
||||||
|
if (subPage.packetExists(packetNumber, designationCode)) {
|
||||||
|
outLine[0] = hamming_8_4_encode[magazineNumber | ((packetNumber & 0x01) << 3)];
|
||||||
|
outLine[1] = hamming_8_4_encode[packetNumber >> 1];
|
||||||
|
outLine.replace(2, 40, subPage.packet(packetNumber, designationCode));
|
||||||
|
outLine[2] = hamming_8_4_encode[designationCode];
|
||||||
|
|
||||||
|
for (int c=3; c<outLine.size(); c++)
|
||||||
|
outLine[c] = hamming_8_4_encode[(int)outLine.at(c)];
|
||||||
|
|
||||||
|
outStream.writeRawData(outLine.constData(), 42);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
auto writeHamming24_18Packet=[&](int packetNumber, int designationCode=0)
|
||||||
|
{
|
||||||
|
if (subPage.packetExists(packetNumber, designationCode)) {
|
||||||
|
outLine[0] = hamming_8_4_encode[magazineNumber | ((packetNumber & 0x01) << 3)];
|
||||||
|
outLine[1] = hamming_8_4_encode[packetNumber >> 1];
|
||||||
|
outLine.replace(2, 40, subPage.packet(packetNumber, designationCode));
|
||||||
|
outLine[2] = hamming_8_4_encode[designationCode];
|
||||||
|
|
||||||
|
for (int c=3; c<outLine.size(); c+=3) {
|
||||||
|
unsigned int D5_D11;
|
||||||
|
unsigned int D12_D18;
|
||||||
|
unsigned int P5, P6;
|
||||||
|
unsigned int Byte_0;
|
||||||
|
|
||||||
|
const unsigned int toEncode = outLine[c] | (outLine[c+1] << 6) | (outLine[c+2] << 12);
|
||||||
|
|
||||||
|
Byte_0 = (hamming_24_18_forward[0][(toEncode >> 0) & 0xff] ^ hamming_24_18_forward[1][(toEncode >> 8) & 0xff] ^ hamming_24_18_forward_2[(toEncode >> 16) & 0x03]);
|
||||||
|
outLine[c] = Byte_0;
|
||||||
|
|
||||||
|
D5_D11 = (toEncode >> 4) & 0x7f;
|
||||||
|
D12_D18 = (toEncode >> 11) & 0x7f;
|
||||||
|
|
||||||
|
P5 = 0x80 & ~(hamming_24_18_parities[0][D12_D18] << 2);
|
||||||
|
outLine[c+1] = D5_D11 | P5;
|
||||||
|
|
||||||
|
P6 = 0x80 & ((hamming_24_18_parities[0][Byte_0] ^ hamming_24_18_parities[0][D5_D11]) << 2);
|
||||||
|
outLine[c+2] = D12_D18 | P6;
|
||||||
|
}
|
||||||
|
|
||||||
|
outStream.writeRawData(outLine.constData(), 42);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
if (magazineNumber == 8)
|
||||||
|
magazineNumber = 0;
|
||||||
|
|
||||||
|
// Write X/0 separately as it features both Hamming 8/4 and 7-bit odd parity within
|
||||||
|
outLine[0] = magazineNumber & 0x07;
|
||||||
|
outLine[1] = 0; // Packet number 0
|
||||||
|
outLine[2] = document.pageNumber() & 0x00f;
|
||||||
|
outLine[3] = (document.pageNumber() & 0x0f0) >> 4;
|
||||||
|
outLine[4] = 0; // Subcode S1 - always export as 0
|
||||||
|
outLine[5] = subPage.controlBit(PageBase::C4ErasePage) << 3;
|
||||||
|
outLine[6] = 0; // Subcode S3 - always export as 0
|
||||||
|
outLine[7] = (subPage.controlBit(PageBase::C5Newsflash) << 2) | (subPage.controlBit(PageBase::C6Subtitle) << 3);
|
||||||
|
outLine[8] = subPage.controlBit(PageBase::C7SuppressHeader) | (subPage.controlBit(PageBase::C8Update) << 1) | (subPage.controlBit(PageBase::C9InterruptedSequence) << 2) | (subPage.controlBit(PageBase::C10InhibitDisplay) << 3);
|
||||||
|
outLine[9] = subPage.controlBit(PageBase::C11SerialMagazine) | (subPage.controlBit(PageBase::C14NOS) << 1) | (subPage.controlBit(PageBase::C13NOS) << 2) | (subPage.controlBit(PageBase::C12NOS) << 3);
|
||||||
|
|
||||||
|
for (int i=0; i<10; i++)
|
||||||
|
outLine[i] = hamming_8_4_encode[(int)outLine.at(i)];
|
||||||
|
|
||||||
|
// If we allow text in the row header, we'd odd-parity encode it here
|
||||||
|
|
||||||
|
outStream.writeRawData(outLine.constData(), 42);
|
||||||
|
|
||||||
|
// After X/0, X/27 then X/28 always come next
|
||||||
|
for (int i=0; i<4; i++)
|
||||||
|
writeHamming8_4Packet(27, i);
|
||||||
|
for (int i=4; i<16; i++)
|
||||||
|
writeHamming24_18Packet(27, i);
|
||||||
|
for (int i=0; i<16; i++)
|
||||||
|
writeHamming24_18Packet(28, i);
|
||||||
|
|
||||||
|
if (document.packetCoding() == TeletextDocument::Coding7bit) {
|
||||||
|
// For 7 bit coding i.e. Level One Pages, X/26 are written before X/1 to X/25
|
||||||
|
for (int i=0; i<16; i++)
|
||||||
|
writeHamming24_18Packet(26, i);
|
||||||
|
for (int i=1; i<=24; i++)
|
||||||
|
write7bitPacket(i);
|
||||||
|
} else {
|
||||||
|
// For others (especially (G)POP pages) X/1 to X/25 are written before X/26
|
||||||
|
if (document.packetCoding() == TeletextDocument::Coding18bit)
|
||||||
|
for (int i=1; i<=25; i++)
|
||||||
|
writeHamming24_18Packet(i);
|
||||||
|
else if (document.packetCoding() == TeletextDocument::Coding4bit)
|
||||||
|
for (int i=1; i<=25; i++)
|
||||||
|
writeHamming8_4Packet(i);
|
||||||
|
else
|
||||||
|
qDebug("Exported broken file as page coding is not supported");
|
||||||
|
for (int i=0; i<16; i++)
|
||||||
|
writeHamming24_18Packet(26, i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
QByteArray rowPacketAlways(PageBase *subPage, int packetNumber)
|
QByteArray rowPacketAlways(PageBase *subPage, int packetNumber)
|
||||||
{
|
{
|
||||||
if (subPage->packetNeeded(packetNumber))
|
if (subPage->packetExists(packetNumber))
|
||||||
return subPage->packet(packetNumber);
|
return subPage->packet(packetNumber);
|
||||||
else
|
else
|
||||||
return QByteArray(40, ' ');
|
return QByteArray(40, ' ');
|
||||||
@@ -345,7 +780,7 @@ QString exportHashStringPackets(LevelOnePage *subPage)
|
|||||||
const char base64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
|
const char base64[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
|
||||||
QString result;
|
QString result;
|
||||||
|
|
||||||
if (subPage->packetNeeded(28,0) || subPage->packetNeeded(28,4)) {
|
if (subPage->packetExists(28,0) || subPage->packetExists(28,4)) {
|
||||||
// X/28/0 and X/28/4 are duplicates apart from the CLUT definitions
|
// X/28/0 and X/28/4 are duplicates apart from the CLUT definitions
|
||||||
// Assemble the duplicate beginning and ending of both packets
|
// Assemble the duplicate beginning and ending of both packets
|
||||||
QString x28StringBegin, x28StringEnd;
|
QString x28StringBegin, x28StringEnd;
|
||||||
@@ -356,9 +791,9 @@ QString exportHashStringPackets(LevelOnePage *subPage)
|
|||||||
|
|
||||||
x28StringEnd = QString("%1%2%3%4").arg(subPage->defaultScreenColour(), 2, 16, QChar('0')).arg(subPage->defaultRowColour(), 2, 16, QChar('0')).arg(subPage->blackBackgroundSubst(), 1, 10).arg(subPage->colourTableRemap(), 1, 10);
|
x28StringEnd = QString("%1%2%3%4").arg(subPage->defaultScreenColour(), 2, 16, QChar('0')).arg(subPage->defaultRowColour(), 2, 16, QChar('0')).arg(subPage->blackBackgroundSubst(), 1, 10).arg(subPage->colourTableRemap(), 1, 10);
|
||||||
|
|
||||||
if (subPage->packetNeeded(28,0))
|
if (subPage->packetExists(28,0))
|
||||||
result.append(":X280=" + x28StringBegin + colourToHexString(2) + colourToHexString(3) + x28StringEnd);
|
result.append(":X280=" + x28StringBegin + colourToHexString(2) + colourToHexString(3) + x28StringEnd);
|
||||||
if (subPage->packetNeeded(28,4))
|
if (subPage->packetExists(28,4))
|
||||||
result.append(":X284=" + x28StringBegin + colourToHexString(0) + colourToHexString(1) + x28StringEnd);
|
result.append(":X284=" + x28StringBegin + colourToHexString(0) + colourToHexString(1) + x28StringEnd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
13
loadsave.h
13
loadsave.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -31,14 +31,17 @@
|
|||||||
#include "pagebase.h"
|
#include "pagebase.h"
|
||||||
|
|
||||||
void loadTTI(QFile *inFile, TeletextDocument *document);
|
void loadTTI(QFile *inFile, TeletextDocument *document);
|
||||||
|
void importT42(QFile *inFile, TeletextDocument *document);
|
||||||
|
|
||||||
int controlBitsToPS(PageBase *);
|
int controlBitsToPS(PageBase *subPage);
|
||||||
|
|
||||||
void saveTTI(QSaveFile &, const TeletextDocument &);
|
void saveTTI(QSaveFile &file, const TeletextDocument &document);
|
||||||
|
void exportT42File(QSaveFile &file, const TeletextDocument &document);
|
||||||
|
void exportM29File(QSaveFile &file, const TeletextDocument &document);
|
||||||
|
|
||||||
QByteArray rowPacketAlways(PageBase *, int);
|
QByteArray rowPacketAlways(PageBase *subPage, int packetNumber);
|
||||||
|
|
||||||
QString exportHashStringPage(LevelOnePage *);
|
QString exportHashStringPage(LevelOnePage *subPage);
|
||||||
QString exportHashStringPackets(LevelOnePage *subPage);
|
QString exportHashStringPackets(LevelOnePage *subPage);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
4
main.cpp
4
main.cpp
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -30,7 +30,7 @@ int main(int argc, char *argv[])
|
|||||||
QApplication::setApplicationDisplayName(QApplication::applicationName());
|
QApplication::setApplicationDisplayName(QApplication::applicationName());
|
||||||
QApplication::setOrganizationName("gkmac.co.uk");
|
QApplication::setOrganizationName("gkmac.co.uk");
|
||||||
QApplication::setOrganizationDomain("gkmac.co.uk");
|
QApplication::setOrganizationDomain("gkmac.co.uk");
|
||||||
QApplication::setApplicationVersion("0.1-alpha");
|
QApplication::setApplicationVersion("0.6.3-beta");
|
||||||
QCommandLineParser parser;
|
QCommandLineParser parser;
|
||||||
parser.setApplicationDescription(QApplication::applicationName());
|
parser.setApplicationDescription(QApplication::applicationName());
|
||||||
parser.addHelpOption();
|
parser.addHelpOption();
|
||||||
|
|||||||
425
mainwidget.cpp
425
mainwidget.cpp
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -17,13 +17,18 @@
|
|||||||
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <QApplication>
|
||||||
#include <QBitmap>
|
#include <QBitmap>
|
||||||
|
#include <QClipboard>
|
||||||
#include <QFrame>
|
#include <QFrame>
|
||||||
#include <QGraphicsItem>
|
#include <QGraphicsItem>
|
||||||
|
#include <QGraphicsItemGroup>
|
||||||
#include <QGraphicsProxyWidget>
|
#include <QGraphicsProxyWidget>
|
||||||
#include <QGraphicsScene>
|
#include <QGraphicsScene>
|
||||||
|
#include <QGraphicsSceneEvent>
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
#include <QMenu>
|
#include <QMenu>
|
||||||
|
#include <QMimeData>
|
||||||
#include <QPainter>
|
#include <QPainter>
|
||||||
#include <QPair>
|
#include <QPair>
|
||||||
#include <QUndoCommand>
|
#include <QUndoCommand>
|
||||||
@@ -33,6 +38,7 @@
|
|||||||
|
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
|
|
||||||
|
#include "decode.h"
|
||||||
#include "document.h"
|
#include "document.h"
|
||||||
#include "keymap.h"
|
#include "keymap.h"
|
||||||
#include "levelonecommands.h"
|
#include "levelonecommands.h"
|
||||||
@@ -47,18 +53,17 @@ TeletextWidget::TeletextWidget(QFrame *parent) : QFrame(parent)
|
|||||||
this->setAttribute(Qt::WA_InputMethodEnabled, true);
|
this->setAttribute(Qt::WA_InputMethodEnabled, true);
|
||||||
m_teletextDocument = new TeletextDocument();
|
m_teletextDocument = new TeletextDocument();
|
||||||
m_levelOnePage = m_teletextDocument->currentSubPage();
|
m_levelOnePage = m_teletextDocument->currentSubPage();
|
||||||
m_pageRender.setTeletextPage(m_levelOnePage);
|
m_pageDecode.setTeletextPage(m_levelOnePage);
|
||||||
|
m_pageRender.setDecoder(&m_pageDecode);
|
||||||
m_insertMode = false;
|
m_insertMode = false;
|
||||||
m_selectionInProgress = false;
|
m_selectionInProgress = false;
|
||||||
m_grid = false;
|
|
||||||
setFocusPolicy(Qt::StrongFocus);
|
setFocusPolicy(Qt::StrongFocus);
|
||||||
m_flashTiming = m_flashPhase = 0;
|
m_flashTiming = m_flashPhase = 0;
|
||||||
connect(&m_pageRender, &TeletextPageRender::flashChanged, this, &TeletextWidget::updateFlashTimer);
|
connect(&m_pageRender, &TeletextPageRender::flashChanged, this, &TeletextWidget::updateFlashTimer);
|
||||||
connect(&m_pageRender, &TeletextPageRender::sidePanelsChanged, this, &TeletextWidget::changeSize);
|
connect(&m_pageDecode, &TeletextPageDecode::sidePanelsChanged, this, &TeletextWidget::changeSize);
|
||||||
connect(m_teletextDocument, &TeletextDocument::subPageSelected, this, &TeletextWidget::subPageSelected);
|
connect(m_teletextDocument, &TeletextDocument::subPageSelected, this, &TeletextWidget::subPageSelected);
|
||||||
connect(m_teletextDocument, &TeletextDocument::contentsChange, this, &TeletextWidget::refreshRow);
|
connect(m_teletextDocument, &TeletextDocument::contentsChanged, this, &TeletextWidget::refreshPage);
|
||||||
connect(m_teletextDocument, &TeletextDocument::refreshNeeded, this, &TeletextWidget::refreshPage);
|
connect(m_teletextDocument, &TeletextDocument::colourChanged, &m_pageRender, &TeletextPageRender::colourChanged);
|
||||||
connect(m_teletextDocument, &TeletextDocument::selectionMoved, this, QOverload<>::of(&TeletextWidget::update));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TeletextWidget::~TeletextWidget()
|
TeletextWidget::~TeletextWidget()
|
||||||
@@ -82,20 +87,15 @@ void TeletextWidget::inputMethodEvent(QInputMethodEvent* event)
|
|||||||
void TeletextWidget::subPageSelected()
|
void TeletextWidget::subPageSelected()
|
||||||
{
|
{
|
||||||
m_levelOnePage = m_teletextDocument->currentSubPage();
|
m_levelOnePage = m_teletextDocument->currentSubPage();
|
||||||
m_pageRender.setTeletextPage(m_levelOnePage);
|
m_pageDecode.setTeletextPage(m_levelOnePage);
|
||||||
refreshPage();
|
m_pageDecode.decodePage();
|
||||||
}
|
m_pageRender.renderPage(true);
|
||||||
|
|
||||||
void TeletextWidget::refreshRow(int rowChanged)
|
|
||||||
{
|
|
||||||
m_pageRender.renderPage(rowChanged);
|
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::refreshPage()
|
void TeletextWidget::refreshPage()
|
||||||
{
|
{
|
||||||
m_pageRender.decodePage();
|
m_pageDecode.decodePage();
|
||||||
m_pageRender.renderPage();
|
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -104,18 +104,12 @@ void TeletextWidget::paintEvent(QPaintEvent *event)
|
|||||||
Q_UNUSED(event);
|
Q_UNUSED(event);
|
||||||
QPainter widgetPainter(this);
|
QPainter widgetPainter(this);
|
||||||
|
|
||||||
widgetPainter.drawPixmap(m_pageRender.leftSidePanelColumns()*12, 0, *m_pageRender.pagePixmap(m_flashPhase), 0, 0, 480, 250);
|
m_pageRender.renderPage();
|
||||||
if (m_pageRender.leftSidePanelColumns())
|
widgetPainter.drawImage(m_pageDecode.leftSidePanelColumns()*12, 0, *m_pageRender.image(m_flashPhase), 0, 0, 480, 250);
|
||||||
widgetPainter.drawPixmap(0, 0, *m_pageRender.pagePixmap(m_flashPhase), 864-m_pageRender.leftSidePanelColumns()*12, 0, m_pageRender.leftSidePanelColumns()*12, 250);
|
if (m_pageDecode.leftSidePanelColumns())
|
||||||
if (m_pageRender.rightSidePanelColumns())
|
widgetPainter.drawImage(0, 0, *m_pageRender.image(m_flashPhase), 864-m_pageDecode.leftSidePanelColumns()*12, 0, m_pageDecode.leftSidePanelColumns()*12, 250);
|
||||||
widgetPainter.drawPixmap(480+m_pageRender.leftSidePanelColumns()*12, 0, *m_pageRender.pagePixmap(m_flashPhase), 480, 0, m_pageRender.rightSidePanelColumns()*12, 250);
|
if (m_pageDecode.rightSidePanelColumns())
|
||||||
if (this->hasFocus())
|
widgetPainter.drawImage(480+m_pageDecode.leftSidePanelColumns()*12, 0, *m_pageRender.image(m_flashPhase), 480, 0, m_pageDecode.rightSidePanelColumns()*12, 250);
|
||||||
widgetPainter.fillRect((m_teletextDocument->cursorColumn()+m_pageRender.leftSidePanelColumns())*12, m_teletextDocument->cursorRow()*10, 12, 10, QColor(128, 128, 128, 192));
|
|
||||||
if (m_teletextDocument->selectionActive()) {
|
|
||||||
widgetPainter.setPen(QPen(QColor(192, 192, 192, 224), 1, Qt::DashLine));
|
|
||||||
widgetPainter.setBrush(QBrush(QColor(255, 255, 255, 64)));
|
|
||||||
widgetPainter.drawRect((m_teletextDocument->selectionLeftColumn()+m_pageRender.leftSidePanelColumns())*12, m_teletextDocument->selectionTopRow()*10, m_teletextDocument->selectionWidth()*12-1, m_teletextDocument->selectionHeight()*10-1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::updateFlashTimer(int newFlashTimer)
|
void TeletextWidget::updateFlashTimer(int newFlashTimer)
|
||||||
@@ -144,28 +138,36 @@ void TeletextWidget::timerEvent(QTimerEvent *event)
|
|||||||
QWidget::timerEvent(event);
|
QWidget::timerEvent(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TeletextWidget::pauseFlash(bool pauseNow)
|
||||||
|
{
|
||||||
|
if (pauseNow && m_flashTiming != 0) {
|
||||||
|
m_flashTimer.stop();
|
||||||
|
m_flashPhase = 0;
|
||||||
|
update();
|
||||||
|
} else if (m_flashTiming != 0)
|
||||||
|
m_flashTimer.start((m_flashTiming == 1) ? 500 : 167, this);
|
||||||
|
}
|
||||||
|
|
||||||
void TeletextWidget::setInsertMode(bool insertMode)
|
void TeletextWidget::setInsertMode(bool insertMode)
|
||||||
{
|
{
|
||||||
m_insertMode = insertMode;
|
m_insertMode = insertMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::toggleReveal(bool revealOn)
|
void TeletextWidget::setReveal(bool reveal)
|
||||||
{
|
{
|
||||||
m_pageRender.setReveal(revealOn);
|
m_pageRender.setReveal(reveal);
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::toggleMix(bool mixOn)
|
void TeletextWidget::setMix(bool mix)
|
||||||
{
|
{
|
||||||
m_pageRender.setMix(mixOn);
|
m_pageRender.setMix(mix);
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::toggleGrid(bool gridOn)
|
void TeletextWidget::setShowControlCodes(bool showControlCodes)
|
||||||
{
|
{
|
||||||
m_grid = gridOn;
|
m_pageRender.setShowControlCodes(showControlCodes);
|
||||||
m_pageRender.setGrid(gridOn);
|
|
||||||
m_pageRender.renderPage();
|
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -173,8 +175,9 @@ void TeletextWidget::setControlBit(int bitNumber, bool active)
|
|||||||
{
|
{
|
||||||
m_levelOnePage->setControlBit(bitNumber, active);
|
m_levelOnePage->setControlBit(bitNumber, active);
|
||||||
if (bitNumber == 1 || bitNumber == 2) {
|
if (bitNumber == 1 || bitNumber == 2) {
|
||||||
m_pageRender.decodePage();
|
m_pageDecode.decodePage();
|
||||||
m_pageRender.renderPage();
|
m_pageRender.renderPage(true);
|
||||||
|
update();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -188,37 +191,6 @@ void TeletextWidget::setDefaultNOS(int newDefaultNOS)
|
|||||||
m_levelOnePage->setDefaultNOS(newDefaultNOS);
|
m_levelOnePage->setDefaultNOS(newDefaultNOS);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::setDefaultScreenColour(int newColour)
|
|
||||||
{
|
|
||||||
m_levelOnePage->setDefaultScreenColour(newColour);
|
|
||||||
m_pageRender.decodePage();
|
|
||||||
m_pageRender.renderPage();
|
|
||||||
}
|
|
||||||
|
|
||||||
void TeletextWidget::setDefaultRowColour(int newColour)
|
|
||||||
{
|
|
||||||
m_levelOnePage->setDefaultRowColour(newColour);
|
|
||||||
m_pageRender.decodePage();
|
|
||||||
m_pageRender.renderPage();
|
|
||||||
update();
|
|
||||||
}
|
|
||||||
|
|
||||||
void TeletextWidget::setColourTableRemap(int newMap)
|
|
||||||
{
|
|
||||||
m_levelOnePage->setColourTableRemap(newMap);
|
|
||||||
m_pageRender.decodePage();
|
|
||||||
m_pageRender.renderPage();
|
|
||||||
update();
|
|
||||||
}
|
|
||||||
|
|
||||||
void TeletextWidget::setBlackBackgroundSubst(bool substOn)
|
|
||||||
{
|
|
||||||
m_levelOnePage->setBlackBackgroundSubst(substOn);
|
|
||||||
m_pageRender.decodePage();
|
|
||||||
m_pageRender.renderPage();
|
|
||||||
update();
|
|
||||||
}
|
|
||||||
|
|
||||||
void TeletextWidget::setSidePanelWidths(int newLeftSidePanelColumns, int newRightSidePanelColumns)
|
void TeletextWidget::setSidePanelWidths(int newLeftSidePanelColumns, int newRightSidePanelColumns)
|
||||||
{
|
{
|
||||||
m_levelOnePage->setLeftSidePanelDisplayed(newLeftSidePanelColumns != 0);
|
m_levelOnePage->setLeftSidePanelDisplayed(newLeftSidePanelColumns != 0);
|
||||||
@@ -227,18 +199,18 @@ void TeletextWidget::setSidePanelWidths(int newLeftSidePanelColumns, int newRigh
|
|||||||
m_levelOnePage->setSidePanelColumns((newLeftSidePanelColumns == 16) ? 0 : newLeftSidePanelColumns);
|
m_levelOnePage->setSidePanelColumns((newLeftSidePanelColumns == 16) ? 0 : newLeftSidePanelColumns);
|
||||||
else
|
else
|
||||||
m_levelOnePage->setSidePanelColumns((newRightSidePanelColumns == 0) ? 0 : 16-newRightSidePanelColumns);
|
m_levelOnePage->setSidePanelColumns((newRightSidePanelColumns == 0) ? 0 : 16-newRightSidePanelColumns);
|
||||||
m_pageRender.updateSidePanels();
|
m_pageDecode.updateSidePanels();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::setSidePanelAtL35Only(bool newSidePanelAtL35Only)
|
void TeletextWidget::setSidePanelAtL35Only(bool newSidePanelAtL35Only)
|
||||||
{
|
{
|
||||||
m_levelOnePage->setSidePanelStatusL25(!newSidePanelAtL35Only);
|
m_levelOnePage->setSidePanelStatusL25(!newSidePanelAtL35Only);
|
||||||
m_pageRender.updateSidePanels();
|
m_pageDecode.updateSidePanels();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TeletextWidget::changeSize()
|
void TeletextWidget::changeSize()
|
||||||
{
|
{
|
||||||
setFixedSize(QSize(480+(pageRender()->leftSidePanelColumns()+pageRender()->rightSidePanelColumns())*12, 250));
|
setFixedSize(QSize(480+(pageDecode()->leftSidePanelColumns()+pageDecode()->rightSidePanelColumns())*12, 250));
|
||||||
emit sizeChanged();
|
emit sizeChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -247,16 +219,16 @@ void TeletextWidget::keyPressEvent(QKeyEvent *event)
|
|||||||
if (event->key() < 0x01000000) {
|
if (event->key() < 0x01000000) {
|
||||||
// A character-typing key was pressed
|
// A character-typing key was pressed
|
||||||
// Try to keymap it, if not keymapped then plain ASCII code (may be) returned
|
// Try to keymap it, if not keymapped then plain ASCII code (may be) returned
|
||||||
char mappedKeyPress = keymapping[m_pageRender.level1CharSet(m_teletextDocument->cursorRow(), m_teletextDocument->cursorColumn())].value(event->text().at(0), *qPrintable(event->text()));
|
char mappedKeyPress = keymapping[m_pageDecode.level1CharSet(m_teletextDocument->cursorRow(), m_teletextDocument->cursorColumn())].value(event->text().at(0), *qPrintable(event->text().at(0)));
|
||||||
if (mappedKeyPress < 0x20)
|
if (mappedKeyPress >= 0x00 && mappedKeyPress <= 0x1f)
|
||||||
return;
|
return;
|
||||||
// If outside ASCII map then the character can't be represented by current Level 1 character set
|
// If outside ASCII map then the character can't be represented by current Level 1 character set
|
||||||
// Map it to block character so it doesn't need to be inserted-between later on
|
// Map it to block character so it doesn't need to be inserted-between later on
|
||||||
if (mappedKeyPress & 0x80)
|
if (mappedKeyPress & 0x80)
|
||||||
mappedKeyPress = 0x7f;
|
mappedKeyPress = 0x7f;
|
||||||
if (m_pageRender.level1MosaicAttribute(m_teletextDocument->cursorRow(), m_teletextDocument->cursorColumn()) && (mappedKeyPress < 0x40 || mappedKeyPress > 0x5f)) {
|
if (m_pageDecode.level1MosaicAttribute(m_teletextDocument->cursorRow(), m_teletextDocument->cursorColumn()) && (mappedKeyPress < 0x40 || mappedKeyPress > 0x5f)) {
|
||||||
// We're on a mosaic and a blast-through character was NOT pressed
|
// We're on a mosaic and a blast-through character was NOT pressed
|
||||||
if (event->key() >= Qt::Key_1 && event->key() <= Qt::Key_9 && event->modifiers() & Qt::KeypadModifier) {
|
if (event->key() >= Qt::Key_0 && event->key() <= Qt::Key_9 && event->modifiers() & Qt::KeypadModifier) {
|
||||||
switch (event->key()) {
|
switch (event->key()) {
|
||||||
case Qt::Key_7:
|
case Qt::Key_7:
|
||||||
toggleCharacterBit(0x01); // Top left
|
toggleCharacterBit(0x01); // Top left
|
||||||
@@ -285,6 +257,9 @@ void TeletextWidget::keyPressEvent(QKeyEvent *event)
|
|||||||
case Qt::Key_3:
|
case Qt::Key_3:
|
||||||
toggleCharacterBit(0x20); // Clear all
|
toggleCharacterBit(0x20); // Clear all
|
||||||
break;
|
break;
|
||||||
|
case Qt::Key_0:
|
||||||
|
toggleCharacterBit(0x66); // Dither
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -325,6 +300,9 @@ void TeletextWidget::keyPressEvent(QKeyEvent *event)
|
|||||||
case mosaicNativeScanCodes[8]:
|
case mosaicNativeScanCodes[8]:
|
||||||
toggleCharacterBit(0x20); // Clear all
|
toggleCharacterBit(0x20); // Clear all
|
||||||
break;
|
break;
|
||||||
|
case mosaicNativeScanCodes[9]:
|
||||||
|
toggleCharacterBit(0x66); // Dither
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
} else
|
} else
|
||||||
@@ -354,32 +332,27 @@ void TeletextWidget::keyPressEvent(QKeyEvent *event)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case Qt::Key_Up:
|
case Qt::Key_Up:
|
||||||
m_teletextDocument->cursorUp();
|
m_teletextDocument->cursorUp(event->modifiers() & Qt::ShiftModifier);
|
||||||
update();
|
|
||||||
break;
|
break;
|
||||||
case Qt::Key_Down:
|
case Qt::Key_Down:
|
||||||
m_teletextDocument->cursorDown();
|
m_teletextDocument->cursorDown(event->modifiers() & Qt::ShiftModifier);
|
||||||
update();
|
|
||||||
break;
|
break;
|
||||||
case Qt::Key_Left:
|
case Qt::Key_Left:
|
||||||
m_teletextDocument->cursorLeft();
|
m_teletextDocument->cursorLeft(event->modifiers() & Qt::ShiftModifier);
|
||||||
update();
|
|
||||||
break;
|
break;
|
||||||
case Qt::Key_Right:
|
case Qt::Key_Right:
|
||||||
m_teletextDocument->cursorRight();
|
m_teletextDocument->cursorRight(event->modifiers() & Qt::ShiftModifier);
|
||||||
update();
|
|
||||||
break;
|
break;
|
||||||
case Qt::Key_Return:
|
case Qt::Key_Return:
|
||||||
case Qt::Key_Enter:
|
case Qt::Key_Enter:
|
||||||
m_teletextDocument->cursorDown();
|
m_teletextDocument->cursorDown();
|
||||||
// fall through
|
|
||||||
case Qt::Key_Home:
|
|
||||||
m_teletextDocument->moveCursor(m_teletextDocument->cursorRow(), 0);
|
m_teletextDocument->moveCursor(m_teletextDocument->cursorRow(), 0);
|
||||||
update();
|
break;
|
||||||
|
case Qt::Key_Home:
|
||||||
|
m_teletextDocument->moveCursor(m_teletextDocument->cursorRow(), 0, event->modifiers() & Qt::ShiftModifier);
|
||||||
break;
|
break;
|
||||||
case Qt::Key_End:
|
case Qt::Key_End:
|
||||||
m_teletextDocument->moveCursor(m_teletextDocument->cursorRow(), 39);
|
m_teletextDocument->moveCursor(m_teletextDocument->cursorRow(), 39, event->modifiers() & Qt::ShiftModifier);
|
||||||
update();
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Qt::Key_PageUp:
|
case Qt::Key_PageUp:
|
||||||
@@ -388,9 +361,9 @@ void TeletextWidget::keyPressEvent(QKeyEvent *event)
|
|||||||
case Qt::Key_PageDown:
|
case Qt::Key_PageDown:
|
||||||
m_teletextDocument->selectSubPagePrevious();
|
m_teletextDocument->selectSubPagePrevious();
|
||||||
break;
|
break;
|
||||||
case Qt::Key_F5:
|
case Qt::Key_F6:
|
||||||
m_pageRender.decodePage();
|
m_pageDecode.decodePage();
|
||||||
m_pageRender.renderPage();
|
m_pageRender.renderPage(true);
|
||||||
update();
|
update();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@@ -405,14 +378,68 @@ void TeletextWidget::setCharacter(unsigned char newCharacter)
|
|||||||
|
|
||||||
void TeletextWidget::toggleCharacterBit(unsigned char bitToToggle)
|
void TeletextWidget::toggleCharacterBit(unsigned char bitToToggle)
|
||||||
{
|
{
|
||||||
QUndoCommand *toggleMosaicBitCommand = new ToggleMosaicBitCommand(m_teletextDocument, bitToToggle);
|
m_teletextDocument->undoStack()->push(new ToggleMosaicBitCommand(m_teletextDocument, bitToToggle));
|
||||||
m_teletextDocument->undoStack()->push(toggleMosaicBitCommand);
|
}
|
||||||
|
|
||||||
|
void TeletextWidget::selectionToClipboard()
|
||||||
|
{
|
||||||
|
QByteArray nativeData;
|
||||||
|
QString plainTextData;
|
||||||
|
QClipboard *clipboard = QApplication::clipboard();
|
||||||
|
|
||||||
|
nativeData.resize(2 + m_teletextDocument->selectionWidth() * m_teletextDocument->selectionHeight());
|
||||||
|
nativeData[0] = m_teletextDocument->selectionHeight();
|
||||||
|
nativeData[1] = m_teletextDocument->selectionWidth();
|
||||||
|
|
||||||
|
plainTextData.reserve((m_teletextDocument->selectionWidth()+1) * m_teletextDocument->selectionHeight() - 1);
|
||||||
|
|
||||||
|
int i=2;
|
||||||
|
|
||||||
|
for (int r=m_teletextDocument->selectionTopRow(); r<=m_teletextDocument->selectionBottomRow(); r++) {
|
||||||
|
for (int c=m_teletextDocument->selectionLeftColumn(); c<=m_teletextDocument->selectionRightColumn(); c++) {
|
||||||
|
nativeData[i++] = m_teletextDocument->currentSubPage()->character(r, c);
|
||||||
|
|
||||||
|
if (m_teletextDocument->currentSubPage()->character(r, c) >= 0x20)
|
||||||
|
plainTextData.append(keymapping[m_pageDecode.level1CharSet(r, c)].key(m_teletextDocument->currentSubPage()->character(r, c), QChar(m_teletextDocument->currentSubPage()->character(r, c))));
|
||||||
|
else
|
||||||
|
plainTextData.append(' ');
|
||||||
|
}
|
||||||
|
|
||||||
|
plainTextData.append('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
QMimeData *mimeData = new QMimeData();
|
||||||
|
mimeData->setData("application/x-teletext", nativeData);
|
||||||
|
mimeData->setText(plainTextData);
|
||||||
|
clipboard->setMimeData(mimeData);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TeletextWidget::cut()
|
||||||
|
{
|
||||||
|
if (!m_teletextDocument->selectionActive())
|
||||||
|
return;
|
||||||
|
|
||||||
|
selectionToClipboard();
|
||||||
|
m_teletextDocument->undoStack()->push(new CutCommand(m_teletextDocument));
|
||||||
|
}
|
||||||
|
|
||||||
|
void TeletextWidget::copy()
|
||||||
|
{
|
||||||
|
if (!m_teletextDocument->selectionActive())
|
||||||
|
return;
|
||||||
|
|
||||||
|
selectionToClipboard();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TeletextWidget::paste()
|
||||||
|
{
|
||||||
|
m_teletextDocument->undoStack()->push(new PasteCommand(m_teletextDocument, m_pageDecode.level1CharSet(m_teletextDocument->cursorRow(), m_teletextDocument->cursorColumn())));
|
||||||
}
|
}
|
||||||
|
|
||||||
QPair<int, int> TeletextWidget::mouseToRowAndColumn(const QPoint &mousePosition)
|
QPair<int, int> TeletextWidget::mouseToRowAndColumn(const QPoint &mousePosition)
|
||||||
{
|
{
|
||||||
int row = mousePosition.y() / 10;
|
int row = mousePosition.y() / 10;
|
||||||
int column = mousePosition.x() / 12 - m_pageRender.leftSidePanelColumns();
|
int column = mousePosition.x() / 12 - m_pageDecode.leftSidePanelColumns();
|
||||||
if (row < 1)
|
if (row < 1)
|
||||||
row = 1;
|
row = 1;
|
||||||
if (row > 24)
|
if (row > 24)
|
||||||
@@ -438,25 +465,12 @@ void TeletextWidget::mouseMoveEvent(QMouseEvent *event)
|
|||||||
{
|
{
|
||||||
if (event->buttons() & Qt::LeftButton) {
|
if (event->buttons() & Qt::LeftButton) {
|
||||||
QPair<int, int> position = mouseToRowAndColumn(event->pos());
|
QPair<int, int> position = mouseToRowAndColumn(event->pos());
|
||||||
if (m_selectionInProgress || position.first != m_teletextDocument->cursorRow() || position.second != m_teletextDocument->cursorColumn()) {
|
if (position.first != m_teletextDocument->cursorRow() || position.second != m_teletextDocument->cursorColumn()) {
|
||||||
int topRow, bottomRow, leftColumn, rightColumn;
|
if (!m_selectionInProgress) {
|
||||||
|
|
||||||
m_selectionInProgress = true;
|
m_selectionInProgress = true;
|
||||||
if (m_teletextDocument->cursorRow() < position.first) {
|
m_teletextDocument->setSelectionCorner(m_teletextDocument->cursorRow(), m_teletextDocument->cursorColumn());
|
||||||
topRow = m_teletextDocument->cursorRow();
|
|
||||||
bottomRow = position.first;
|
|
||||||
} else {
|
|
||||||
topRow = position.first;
|
|
||||||
bottomRow = m_teletextDocument->cursorRow();
|
|
||||||
}
|
}
|
||||||
if (m_teletextDocument->cursorColumn() < position.second) {
|
m_teletextDocument->moveCursor(position.first, position.second, true);
|
||||||
leftColumn = m_teletextDocument->cursorColumn();
|
|
||||||
rightColumn = position.second;
|
|
||||||
} else {
|
|
||||||
leftColumn = position.second;
|
|
||||||
rightColumn = m_teletextDocument->cursorColumn();
|
|
||||||
}
|
|
||||||
m_teletextDocument->setSelection(topRow, leftColumn, bottomRow, rightColumn);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -480,7 +494,12 @@ void TeletextWidget::focusOutEvent(QFocusEvent *event)
|
|||||||
|
|
||||||
LevelOneScene::LevelOneScene(QWidget *levelOneWidget, QObject *parent) : QGraphicsScene(parent)
|
LevelOneScene::LevelOneScene(QWidget *levelOneWidget, QObject *parent) : QGraphicsScene(parent)
|
||||||
{
|
{
|
||||||
|
m_grid = false;
|
||||||
|
|
||||||
|
// These dimensions are scratch, setBorderDimensions will get called straight away to adjust them
|
||||||
setSceneRect(0, 0, 600, 288);
|
setSceneRect(0, 0, 600, 288);
|
||||||
|
|
||||||
|
// Full screen colours
|
||||||
m_fullScreenTopRectItem = new QGraphicsRectItem(0, 0, 600, 19);
|
m_fullScreenTopRectItem = new QGraphicsRectItem(0, 0, 600, 19);
|
||||||
m_fullScreenTopRectItem->setPen(Qt::NoPen);
|
m_fullScreenTopRectItem->setPen(Qt::NoPen);
|
||||||
m_fullScreenTopRectItem->setBrush(QBrush(QColor(0, 0, 0)));
|
m_fullScreenTopRectItem->setBrush(QBrush(QColor(0, 0, 0)));
|
||||||
@@ -490,6 +509,7 @@ LevelOneScene::LevelOneScene(QWidget *levelOneWidget, QObject *parent) : QGraphi
|
|||||||
m_fullScreenBottomRectItem->setBrush(QBrush(QColor(0, 0, 0)));
|
m_fullScreenBottomRectItem->setBrush(QBrush(QColor(0, 0, 0)));
|
||||||
addItem(m_fullScreenBottomRectItem);
|
addItem(m_fullScreenBottomRectItem);
|
||||||
|
|
||||||
|
// Full row colours
|
||||||
for (int r=0; r<25; r++) {
|
for (int r=0; r<25; r++) {
|
||||||
m_fullRowLeftRectItem[r] = new QGraphicsRectItem(0, 19+r*10, 60, 10);
|
m_fullRowLeftRectItem[r] = new QGraphicsRectItem(0, 19+r*10, 60, 10);
|
||||||
m_fullRowLeftRectItem[r]->setPen(Qt::NoPen);
|
m_fullRowLeftRectItem[r]->setPen(Qt::NoPen);
|
||||||
@@ -501,16 +521,59 @@ LevelOneScene::LevelOneScene(QWidget *levelOneWidget, QObject *parent) : QGraphi
|
|||||||
addItem(m_fullRowRightRectItem[r]);
|
addItem(m_fullRowRightRectItem[r]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Main text widget
|
||||||
m_levelOneProxyWidget = addWidget(levelOneWidget);
|
m_levelOneProxyWidget = addWidget(levelOneWidget);
|
||||||
m_levelOneProxyWidget->setPos(60, 19);
|
m_levelOneProxyWidget->setPos(60, 19);
|
||||||
m_levelOneProxyWidget->setAutoFillBackground(false);
|
m_levelOneProxyWidget->setAutoFillBackground(false);
|
||||||
|
m_levelOneProxyWidget->setFocus();
|
||||||
|
|
||||||
|
// Selection
|
||||||
|
m_selectionRectItem = new QGraphicsRectItem(0, 0, 12, 10);
|
||||||
|
m_selectionRectItem->setVisible(false);
|
||||||
|
m_selectionRectItem->setPen(QPen(QColor(192, 192, 192), 1, Qt::DashLine));
|
||||||
|
m_selectionRectItem->setBrush(QBrush(QColor(255, 255, 255, 64)));
|
||||||
|
addItem(m_selectionRectItem);
|
||||||
|
|
||||||
|
// Cursor
|
||||||
|
m_cursorRectItem = new QGraphicsRectItem(0, 0, 12, 10);
|
||||||
|
m_cursorRectItem->setPen(Qt::NoPen);
|
||||||
|
m_cursorRectItem->setBrush(QBrush(QColor(128, 128, 128, 192)));
|
||||||
|
addItem(m_cursorRectItem);
|
||||||
|
|
||||||
|
// Optional grid overlay for text widget
|
||||||
|
m_mainGridItemGroup = new QGraphicsItemGroup;
|
||||||
|
m_mainGridItemGroup->setVisible(false);
|
||||||
|
addItem(m_mainGridItemGroup);
|
||||||
|
// Additional vertical pieces of grid for side panels
|
||||||
|
for (int i=0; i<32; i++) {
|
||||||
|
m_sidePanelGridNeeded[i] = false;
|
||||||
|
m_sidePanelGridItemGroup[i] = new QGraphicsItemGroup;
|
||||||
|
m_sidePanelGridItemGroup[i]->setVisible(false);
|
||||||
|
addItem(m_sidePanelGridItemGroup[i]);
|
||||||
|
}
|
||||||
|
for (int r=1; r<25; r++) {
|
||||||
|
for (int c=0; c<40; c++) {
|
||||||
|
QGraphicsRectItem *gridPiece = new QGraphicsRectItem(c*12, r*10, 12, 10);
|
||||||
|
gridPiece->setPen(QPen(QBrush(QColor(128, 128, 128, r<24 ? 192 : 128)), 0));
|
||||||
|
m_mainGridItemGroup->addToGroup(gridPiece);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (r < 24)
|
||||||
|
for (int c=0; c<32; c++) {
|
||||||
|
QGraphicsRectItem *gridPiece = new QGraphicsRectItem(0, r*10, 12, 10);
|
||||||
|
gridPiece->setPen(QPen(QBrush(QColor(128, 128, 128, 64)), 0));
|
||||||
|
m_sidePanelGridItemGroup[c]->addToGroup(gridPiece);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
installEventFilter(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void LevelOneScene::setDimensions(int sceneWidth, int sceneHeight, int widgetWidth)
|
void LevelOneScene::setBorderDimensions(int sceneWidth, int sceneHeight, int widgetWidth, int leftSidePanelColumns, int rightSidePanelColumns)
|
||||||
{
|
{
|
||||||
setSceneRect(0, 0, sceneWidth, sceneHeight);
|
setSceneRect(0, 0, sceneWidth, sceneHeight);
|
||||||
|
|
||||||
// Assume widget height is always 250
|
// Assume text widget height is always 250
|
||||||
int topBottomBorders = (sceneHeight-250) / 2;
|
int topBottomBorders = (sceneHeight-250) / 2;
|
||||||
// Ideally we'd use m_levelOneProxyWidget->size() to discover the widget width ourselves
|
// Ideally we'd use m_levelOneProxyWidget->size() to discover the widget width ourselves
|
||||||
// but this causes a stubborn segfault, so we have to receive the widgetWidth as a parameter
|
// but this causes a stubborn segfault, so we have to receive the widgetWidth as a parameter
|
||||||
@@ -518,23 +581,147 @@ void LevelOneScene::setDimensions(int sceneWidth, int sceneHeight, int widgetWid
|
|||||||
|
|
||||||
m_levelOneProxyWidget->setPos(leftRightBorders, topBottomBorders);
|
m_levelOneProxyWidget->setPos(leftRightBorders, topBottomBorders);
|
||||||
|
|
||||||
|
// Position grid to cover central 40 columns
|
||||||
|
m_mainGridItemGroup->setPos(leftRightBorders + leftSidePanelColumns*12, topBottomBorders);
|
||||||
|
|
||||||
|
updateCursor();
|
||||||
|
updateSelection();
|
||||||
|
|
||||||
|
// Grid for right side panel
|
||||||
|
for (int c=0; c<16; c++)
|
||||||
|
if (rightSidePanelColumns > c) {
|
||||||
|
m_sidePanelGridItemGroup[c]->setPos(leftRightBorders + leftSidePanelColumns*12 + 480 + c*12, topBottomBorders);
|
||||||
|
m_sidePanelGridItemGroup[c]->setVisible(m_grid);
|
||||||
|
m_sidePanelGridNeeded[c] = true;
|
||||||
|
} else {
|
||||||
|
m_sidePanelGridItemGroup[c]->setVisible(false);
|
||||||
|
m_sidePanelGridNeeded[c] = false;
|
||||||
|
}
|
||||||
|
// Grid for left side panel
|
||||||
|
for (int c=0; c<16; c++)
|
||||||
|
if (c < leftSidePanelColumns) {
|
||||||
|
m_sidePanelGridItemGroup[31-c]->setPos(leftRightBorders + (leftSidePanelColumns-c-1)*12, topBottomBorders);
|
||||||
|
m_sidePanelGridItemGroup[31-c]->setVisible(m_grid);
|
||||||
|
m_sidePanelGridNeeded[31-c] = true;
|
||||||
|
} else {
|
||||||
|
m_sidePanelGridItemGroup[31-c]->setVisible(false);
|
||||||
|
m_sidePanelGridNeeded[31-c] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Full screen colours
|
||||||
m_fullScreenTopRectItem->setRect(0, 0, sceneWidth, topBottomBorders);
|
m_fullScreenTopRectItem->setRect(0, 0, sceneWidth, topBottomBorders);
|
||||||
m_fullScreenBottomRectItem->setRect(0, 250+topBottomBorders, sceneWidth, topBottomBorders);
|
m_fullScreenBottomRectItem->setRect(0, 250+topBottomBorders, sceneWidth, topBottomBorders);
|
||||||
|
// Full row colours
|
||||||
for (int r=0; r<25; r++) {
|
for (int r=0; r<25; r++) {
|
||||||
m_fullRowLeftRectItem[r]->setRect(0, topBottomBorders+r*10, leftRightBorders+1, 10);
|
m_fullRowLeftRectItem[r]->setRect(0, topBottomBorders+r*10, leftRightBorders+1, 10);
|
||||||
m_fullRowRightRectItem[r]->setRect(sceneWidth-leftRightBorders-1, topBottomBorders+r*10, leftRightBorders+1, 10);
|
m_fullRowRightRectItem[r]->setRect(sceneWidth-leftRightBorders-1, topBottomBorders+r*10, leftRightBorders+1, 10);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void LevelOneScene::updateCursor()
|
||||||
|
{
|
||||||
|
m_cursorRectItem->setPos(m_mainGridItemGroup->pos().x() + static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->cursorColumn()*12, m_mainGridItemGroup->pos().y() + static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->cursorRow()*10);
|
||||||
|
}
|
||||||
|
|
||||||
|
void LevelOneScene::updateSelection()
|
||||||
|
{
|
||||||
|
if (!static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->selectionActive()) {
|
||||||
|
m_selectionRectItem->setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_selectionRectItem->setRect(m_mainGridItemGroup->pos().x() + static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->selectionLeftColumn()*12, m_mainGridItemGroup->pos().y() + static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->selectionTopRow()*10, static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->selectionWidth()*12-1, static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->selectionHeight()*10-1);
|
||||||
|
|
||||||
|
m_selectionRectItem->setVisible(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void LevelOneScene::setMix(bool mix)
|
||||||
|
{
|
||||||
|
if (mix) {
|
||||||
|
m_fullScreenTopRectItem->setBrush(Qt::transparent);
|
||||||
|
m_fullScreenBottomRectItem->setBrush(Qt::transparent);
|
||||||
|
for (int r=0; r<25; r++) {
|
||||||
|
m_fullRowLeftRectItem[r]->setBrush(Qt::transparent);
|
||||||
|
m_fullRowRightRectItem[r]->setBrush(Qt::transparent);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
setFullScreenColour(static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->pageDecode()->fullScreenQColor());
|
||||||
|
for (int r=0; r<25; r++)
|
||||||
|
setFullRowColour(r, static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->pageDecode()->fullRowQColor(r));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void LevelOneScene::toggleGrid(bool gridOn)
|
||||||
|
{
|
||||||
|
m_grid = gridOn;
|
||||||
|
m_mainGridItemGroup->setVisible(gridOn);
|
||||||
|
for (int i=0; i<32; i++)
|
||||||
|
if (m_sidePanelGridNeeded[i])
|
||||||
|
m_sidePanelGridItemGroup[i]->setVisible(gridOn);
|
||||||
|
}
|
||||||
|
|
||||||
|
void LevelOneScene::hideGUIElements(bool hidden)
|
||||||
|
{
|
||||||
|
if (hidden) {
|
||||||
|
m_cursorRectItem->setVisible(false);
|
||||||
|
m_selectionRectItem->setVisible(false);
|
||||||
|
} else {
|
||||||
|
if (static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->document()->selectionActive())
|
||||||
|
m_selectionRectItem->setVisible(true);
|
||||||
|
|
||||||
|
m_cursorRectItem->setVisible(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Implements Ctrl+mousewheel zoom
|
||||||
|
bool LevelOneScene::eventFilter(QObject *object, QEvent *event)
|
||||||
|
{
|
||||||
|
Q_UNUSED(object);
|
||||||
|
|
||||||
|
if (event->type() == QEvent::GraphicsSceneWheel && static_cast<QGraphicsSceneWheelEvent *>(event)->modifiers() == Qt::ControlModifier) {
|
||||||
|
if (static_cast<QGraphicsSceneWheelEvent *>(event)->delta() > 0)
|
||||||
|
emit mouseZoomIn();
|
||||||
|
else if (static_cast<QGraphicsSceneWheelEvent *>(event)->delta() < 0)
|
||||||
|
emit mouseZoomOut();
|
||||||
|
|
||||||
|
event->accept();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clicking outside the main text widget but still within the scene would
|
||||||
|
// cause keyboard focus loss.
|
||||||
|
// So on every keypress within the scene, wrench the focus back to the widget
|
||||||
|
// if necessary.
|
||||||
|
void LevelOneScene::keyPressEvent(QKeyEvent *keyEvent)
|
||||||
|
{
|
||||||
|
if (focusItem() != m_levelOneProxyWidget)
|
||||||
|
setFocusItem(m_levelOneProxyWidget);
|
||||||
|
|
||||||
|
QGraphicsScene::keyPressEvent(keyEvent);
|
||||||
|
}
|
||||||
|
|
||||||
|
void LevelOneScene::keyReleaseEvent(QKeyEvent *keyEvent)
|
||||||
|
{
|
||||||
|
if (focusItem() != m_levelOneProxyWidget)
|
||||||
|
setFocusItem(m_levelOneProxyWidget);
|
||||||
|
|
||||||
|
QGraphicsScene::keyReleaseEvent(keyEvent);
|
||||||
|
}
|
||||||
|
|
||||||
void LevelOneScene::setFullScreenColour(const QColor &newColor)
|
void LevelOneScene::setFullScreenColour(const QColor &newColor)
|
||||||
{
|
{
|
||||||
m_fullScreenTopRectItem->setBrush(QBrush(newColor));
|
if (!static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->pageRender()->mix()) {
|
||||||
m_fullScreenBottomRectItem->setBrush(QBrush(newColor));
|
m_fullScreenTopRectItem->setBrush(newColor);
|
||||||
|
m_fullScreenBottomRectItem->setBrush(newColor);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void LevelOneScene::setFullRowColour(int row, const QColor &newColor)
|
void LevelOneScene::setFullRowColour(int row, const QColor &newColor)
|
||||||
{
|
{
|
||||||
m_fullRowLeftRectItem[row]->setBrush(QBrush(newColor));
|
if (!static_cast<TeletextWidget *>(m_levelOneProxyWidget->widget())->pageRender()->mix()) {
|
||||||
m_fullRowRightRectItem[row]->setBrush(QBrush(newColor));
|
m_fullRowLeftRectItem[row]->setBrush(newColor);
|
||||||
|
m_fullRowRightRectItem[row]->setBrush(newColor);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
79
mainwidget.h
79
mainwidget.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -22,12 +22,14 @@
|
|||||||
|
|
||||||
#include <QBasicTimer>
|
#include <QBasicTimer>
|
||||||
#include <QFrame>
|
#include <QFrame>
|
||||||
|
#include <QGraphicsItemGroup>
|
||||||
#include <QGraphicsProxyWidget>
|
#include <QGraphicsProxyWidget>
|
||||||
#include <QGraphicsScene>
|
#include <QGraphicsScene>
|
||||||
#include <QPair>
|
#include <QPair>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
#include "decode.h"
|
||||||
#include "document.h"
|
#include "document.h"
|
||||||
#include "levelonepage.h"
|
#include "levelonepage.h"
|
||||||
#include "render.h"
|
#include "render.h"
|
||||||
@@ -41,16 +43,18 @@ class TeletextWidget : public QFrame
|
|||||||
public:
|
public:
|
||||||
TeletextWidget(QFrame *parent = 0);
|
TeletextWidget(QFrame *parent = 0);
|
||||||
~TeletextWidget();
|
~TeletextWidget();
|
||||||
void setCharacter(unsigned char);
|
void setCharacter(unsigned char newCharacter);
|
||||||
void toggleCharacterBit(unsigned char);
|
void toggleCharacterBit(unsigned char bitToToggle);
|
||||||
bool insertMode() const { return m_insertMode; };
|
bool insertMode() const { return m_insertMode; };
|
||||||
void setInsertMode(bool);
|
void setInsertMode(bool insertMode);
|
||||||
|
bool showControlCodes() const { return m_pageRender.showControlCodes(); };
|
||||||
|
|
||||||
QSize sizeHint() { return QSize(480+(pageRender()->leftSidePanelColumns()+pageRender()->rightSidePanelColumns())*12, 250); }
|
QSize sizeHint() { return QSize(480+(pageDecode()->leftSidePanelColumns()+pageDecode()->rightSidePanelColumns())*12, 250); }
|
||||||
|
|
||||||
void inputMethodEvent(QInputMethodEvent *);
|
void inputMethodEvent(QInputMethodEvent *event);
|
||||||
|
|
||||||
TeletextDocument* document() const { return m_teletextDocument; }
|
TeletextDocument* document() const { return m_teletextDocument; }
|
||||||
|
TeletextPageDecode *pageDecode() { return &m_pageDecode; }
|
||||||
TeletextPageRender *pageRender() { return &m_pageRender; }
|
TeletextPageRender *pageRender() { return &m_pageRender; }
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
@@ -60,21 +64,22 @@ signals:
|
|||||||
public slots:
|
public slots:
|
||||||
void subPageSelected();
|
void subPageSelected();
|
||||||
void refreshPage();
|
void refreshPage();
|
||||||
void toggleReveal(bool);
|
void setReveal(bool reveal);
|
||||||
void toggleMix(bool);
|
void setMix(bool mix);
|
||||||
void toggleGrid(bool);
|
void setShowControlCodes(bool showControlCodes);
|
||||||
void updateFlashTimer(int);
|
void updateFlashTimer(int newFlashTimer);
|
||||||
void refreshRow(int);
|
void pauseFlash(bool pauseNow);
|
||||||
|
|
||||||
|
void setControlBit(int bitNumber, bool active);
|
||||||
|
void setDefaultCharSet(int newDefaultCharSet);
|
||||||
|
void setDefaultNOS(int newDefaultNOS);
|
||||||
|
void setSidePanelWidths(int newLeftSidePanelColumns, int newRightSidePanelColumns);
|
||||||
|
void setSidePanelAtL35Only(bool newSidePanelAtL35Only);
|
||||||
|
|
||||||
|
void cut();
|
||||||
|
void copy();
|
||||||
|
void paste();
|
||||||
|
|
||||||
void setControlBit(int, bool);
|
|
||||||
void setDefaultCharSet(int);
|
|
||||||
void setDefaultNOS(int);
|
|
||||||
void setDefaultScreenColour(int);
|
|
||||||
void setDefaultRowColour(int);
|
|
||||||
void setColourTableRemap(int);
|
|
||||||
void setBlackBackgroundSubst(bool);
|
|
||||||
void setSidePanelWidths(int, int);
|
|
||||||
void setSidePanelAtL35Only(bool);
|
|
||||||
void changeSize();
|
void changeSize();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -86,20 +91,20 @@ protected:
|
|||||||
void focusInEvent(QFocusEvent *event) override;
|
void focusInEvent(QFocusEvent *event) override;
|
||||||
void focusOutEvent(QFocusEvent *event) override;
|
void focusOutEvent(QFocusEvent *event) override;
|
||||||
|
|
||||||
|
TeletextPageDecode m_pageDecode;
|
||||||
TeletextPageRender m_pageRender;
|
TeletextPageRender m_pageRender;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TeletextDocument* m_teletextDocument;
|
TeletextDocument* m_teletextDocument;
|
||||||
LevelOnePage* m_levelOnePage;
|
LevelOnePage* m_levelOnePage;
|
||||||
bool m_insertMode, m_grid, m_selectionInProgress;
|
bool m_insertMode, m_selectionInProgress;
|
||||||
QBasicTimer m_flashTimer;
|
QBasicTimer m_flashTimer;
|
||||||
int m_flashTiming, m_flashPhase;
|
int m_flashTiming, m_flashPhase;
|
||||||
|
|
||||||
void timerEvent(QTimerEvent *event) override;
|
void timerEvent(QTimerEvent *event) override;
|
||||||
|
void selectionToClipboard();
|
||||||
|
|
||||||
void calculateDimensions();
|
QPair<int, int> mouseToRowAndColumn(const QPoint &mousePosition);
|
||||||
|
|
||||||
QPair<int, int> mouseToRowAndColumn(const QPoint &);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class LevelOneScene : public QGraphicsScene
|
class LevelOneScene : public QGraphicsScene
|
||||||
@@ -107,17 +112,35 @@ class LevelOneScene : public QGraphicsScene
|
|||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
LevelOneScene(QWidget *, QObject *parent = nullptr);
|
LevelOneScene(QWidget *levelOneWidget, QObject *parent = nullptr);
|
||||||
void setDimensions(int, int, int);
|
void setBorderDimensions(int sceneWidth, int sceneHeight, int widgetWidth, int leftSidePanelColumns, int rightSidePanelColumns);
|
||||||
|
QGraphicsRectItem *cursorRectItem() const { return m_cursorRectItem; }
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void setFullScreenColour(const QColor &);
|
void updateCursor();
|
||||||
void setFullRowColour(int, const QColor &);
|
void updateSelection();
|
||||||
|
void setMix(bool mix);
|
||||||
|
void toggleGrid(bool gridOn);
|
||||||
|
void hideGUIElements(bool hidden);
|
||||||
|
void setFullScreenColour(const QColor &newColor);
|
||||||
|
void setFullRowColour(int row, const QColor &newColor);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void mouseZoomIn();
|
||||||
|
void mouseZoomOut();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
bool eventFilter(QObject *object, QEvent *event);
|
||||||
|
void keyPressEvent(QKeyEvent *event);
|
||||||
|
void keyReleaseEvent(QKeyEvent *keyEvent);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QGraphicsRectItem *m_fullScreenTopRectItem, *m_fullScreenBottomRectItem;
|
QGraphicsRectItem *m_fullScreenTopRectItem, *m_fullScreenBottomRectItem;
|
||||||
QGraphicsRectItem *m_fullRowLeftRectItem[25], *m_fullRowRightRectItem[25];
|
QGraphicsRectItem *m_fullRowLeftRectItem[25], *m_fullRowRightRectItem[25];
|
||||||
QGraphicsProxyWidget *m_levelOneProxyWidget;
|
QGraphicsProxyWidget *m_levelOneProxyWidget;
|
||||||
|
QGraphicsRectItem *m_cursorRectItem, *m_selectionRectItem;
|
||||||
|
QGraphicsItemGroup *m_mainGridItemGroup, *m_sidePanelGridItemGroup[32];
|
||||||
|
bool m_grid, m_sidePanelGridNeeded[32];
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
475
mainwindow.cpp
475
mainwindow.cpp
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -17,14 +17,18 @@
|
|||||||
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <QActionGroup>
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QDesktopServices>
|
#include <QDesktopServices>
|
||||||
#include <QFileDialog>
|
#include <QFileDialog>
|
||||||
|
#include <QImage>
|
||||||
#include <QList>
|
#include <QList>
|
||||||
#include <QMenuBar>
|
#include <QMenuBar>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
|
#include <QPainter>
|
||||||
#include <QPushButton>
|
#include <QPushButton>
|
||||||
#include <QRadioButton>
|
#include <QRadioButton>
|
||||||
|
#include <QRegularExpression>
|
||||||
#include <QSaveFile>
|
#include <QSaveFile>
|
||||||
#include <QScreen>
|
#include <QScreen>
|
||||||
#include <QSettings>
|
#include <QSettings>
|
||||||
@@ -39,6 +43,7 @@
|
|||||||
#include "levelonecommands.h"
|
#include "levelonecommands.h"
|
||||||
#include "loadsave.h"
|
#include "loadsave.h"
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
|
#include "pagecomposelinksdockwidget.h"
|
||||||
#include "pageenhancementsdockwidget.h"
|
#include "pageenhancementsdockwidget.h"
|
||||||
#include "pageoptionsdockwidget.h"
|
#include "pageoptionsdockwidget.h"
|
||||||
#include "palettedockwidget.h"
|
#include "palettedockwidget.h"
|
||||||
@@ -106,20 +111,122 @@ void MainWindow::openFile(const QString &fileName)
|
|||||||
other->show();
|
other->show();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline bool hasTTISuffix(const QString &filename)
|
||||||
|
{
|
||||||
|
return filename.endsWith(".tti", Qt::CaseInsensitive) || filename.endsWith(".ttix", Qt::CaseInsensitive);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void changeSuffixFromTTI(QString &filename, const QString &newSuffix)
|
||||||
|
{
|
||||||
|
if (filename.endsWith(".tti", Qt::CaseInsensitive)) {
|
||||||
|
filename.chop(4);
|
||||||
|
filename.append("." + newSuffix);
|
||||||
|
} else if (filename.endsWith(".ttix", Qt::CaseInsensitive)) {
|
||||||
|
filename.chop(5);
|
||||||
|
filename.append("." + newSuffix);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool MainWindow::save()
|
bool MainWindow::save()
|
||||||
{
|
{
|
||||||
return m_isUntitled ? saveAs() : saveFile(m_curFile);
|
// If imported from non-.tti, force "Save As" so we don't clobber the original imported file
|
||||||
|
return m_isUntitled || !hasTTISuffix(m_curFile) ? saveAs() : saveFile(m_curFile);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MainWindow::saveAs()
|
bool MainWindow::saveAs()
|
||||||
{
|
{
|
||||||
QString fileName = QFileDialog::getSaveFileName(this, tr("Save As"), m_curFile);
|
QString suggestedName = m_curFile;
|
||||||
|
|
||||||
|
// If imported from non-.tti, change extension so we don't clobber the original imported file
|
||||||
|
if (suggestedName.endsWith(".t42", Qt::CaseInsensitive)) {
|
||||||
|
suggestedName.chop(4);
|
||||||
|
suggestedName.append(".tti");
|
||||||
|
}
|
||||||
|
|
||||||
|
QString fileName = QFileDialog::getSaveFileName(this, tr("Save As"), suggestedName, "TTI teletext page (*.tti *.ttix)");
|
||||||
if (fileName.isEmpty())
|
if (fileName.isEmpty())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
return saveFile(fileName);
|
return saveFile(fileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainWindow::reload()
|
||||||
|
{
|
||||||
|
if (m_isUntitled)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!m_textWidget->document()->undoStack()->isClean()) {
|
||||||
|
const QMessageBox::StandardButton ret = QMessageBox::warning(this, QApplication::applicationDisplayName(), tr("The document \"%1\" has been modified.\nDo you want to discard your changes?").arg(QFileInfo(m_curFile).fileName()), QMessageBox::Discard | QMessageBox::Cancel);
|
||||||
|
|
||||||
|
if (ret != QMessageBox::Discard)
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
const QMessageBox::StandardButton ret = QMessageBox::warning(this, QApplication::applicationDisplayName(), tr("Do you want to reload the document \"%1\" from disk?").arg(QFileInfo(m_curFile).fileName()), QMessageBox::Yes | QMessageBox::No);
|
||||||
|
|
||||||
|
if (ret != QMessageBox::Yes)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int subPageIndex = m_textWidget->document()->currentSubPageIndex();
|
||||||
|
|
||||||
|
m_textWidget->document()->clear();
|
||||||
|
loadFile(m_curFile);
|
||||||
|
|
||||||
|
if (subPageIndex >= m_textWidget->document()->numberOfSubPages())
|
||||||
|
subPageIndex = m_textWidget->document()->numberOfSubPages()-1;
|
||||||
|
|
||||||
|
m_textWidget->document()->selectSubPageIndex(subPageIndex, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MainWindow::exportPNG()
|
||||||
|
{
|
||||||
|
QString exportFileName = QFileDialog::getSaveFileName(this, tr("Export PNG"), QString(), "PNG image (*.png)");
|
||||||
|
if (exportFileName.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Prepare widget image for extraction
|
||||||
|
m_textWidget->pauseFlash(true);
|
||||||
|
m_textScene->hideGUIElements(true);
|
||||||
|
// Disable exporting in Mix mode as it corrupts the background
|
||||||
|
bool reMix = m_textWidget->pageRender()->mix();
|
||||||
|
if (reMix) {
|
||||||
|
m_textWidget->setMix(false);
|
||||||
|
m_textScene->setMix(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extract the image from the scene
|
||||||
|
QImage interImage = QImage(m_textScene->sceneRect().size().toSize(), QImage::Format_RGB32);
|
||||||
|
// This ought to make the background transparent in Mix mode, but it doesn't
|
||||||
|
// if (m_textWidget->pageDecode()->mix())
|
||||||
|
// interImage.fill(QColor(0, 0, 0, 0));
|
||||||
|
QPainter interPainter(&interImage);
|
||||||
|
m_textScene->render(&interPainter);
|
||||||
|
|
||||||
|
// Now we've extracted the image we can put the GUI things back
|
||||||
|
m_textScene->hideGUIElements(false);
|
||||||
|
if (reMix) {
|
||||||
|
m_textWidget->setMix(true);
|
||||||
|
m_textScene->setMix(true);
|
||||||
|
}
|
||||||
|
m_textWidget->pauseFlash(false);
|
||||||
|
|
||||||
|
// Now scale the extracted image to the selected aspect ratio
|
||||||
|
// We do this in two steps so that anti-aliasing only occurs on vertical lines
|
||||||
|
|
||||||
|
// Double the vertical height first
|
||||||
|
const QImage doubleHeightImage = interImage.scaled(interImage.width(), interImage.height()*2, Qt::IgnoreAspectRatio, Qt::FastTransformation);
|
||||||
|
|
||||||
|
// If aspect ratio is Pixel 1:2 we're already at the correct scale
|
||||||
|
if (m_viewAspectRatio != 3) {
|
||||||
|
// Scale it horizontally to the selected aspect ratio
|
||||||
|
const QImage scaledImage = doubleHeightImage.scaled((int)((float)doubleHeightImage.width() * aspectRatioHorizontalScaling[m_viewAspectRatio] * 2), doubleHeightImage.height(), Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
||||||
|
|
||||||
|
if (!scaledImage.save(exportFileName, "PNG"))
|
||||||
|
QMessageBox::warning(this, QApplication::applicationDisplayName(), tr("Cannot export file %1.").arg(QDir::toNativeSeparators(exportFileName)));
|
||||||
|
} else if (!doubleHeightImage.save(exportFileName, "PNG"))
|
||||||
|
QMessageBox::warning(this, QApplication::applicationDisplayName(), tr("Cannot export file %1.").arg(QDir::toNativeSeparators(exportFileName)));
|
||||||
|
}
|
||||||
|
|
||||||
void MainWindow::exportZXNet()
|
void MainWindow::exportZXNet()
|
||||||
{
|
{
|
||||||
QDesktopServices::openUrl(QUrl("http://zxnet.co.uk/teletext/editor/" + exportHashStringPage(m_textWidget->document()->currentSubPage()) + exportHashStringPackets(m_textWidget->document()->currentSubPage())));
|
QDesktopServices::openUrl(QUrl("http://zxnet.co.uk/teletext/editor/" + exportHashStringPage(m_textWidget->document()->currentSubPage()) + exportHashStringPackets(m_textWidget->document()->currentSubPage())));
|
||||||
@@ -132,12 +239,12 @@ void MainWindow::exportEditTF()
|
|||||||
|
|
||||||
void MainWindow::about()
|
void MainWindow::about()
|
||||||
{
|
{
|
||||||
QMessageBox::about(this, tr("About QTeletextMaker"), tr("<b>QTeletextMaker</b><br>"
|
QMessageBox::about(this, tr("About"), QString("<b>%1</b><br>"
|
||||||
"An open source Level 2.5 teletext page editor.<br>"
|
"An open source Level 2.5 teletext page editor.<br>"
|
||||||
"<i>Version 0.1-alpha</i><br><br>"
|
"<i>Version %2</i><br><br>"
|
||||||
"Copyright (C) 2020, 2021 Gavin MacGregor<br><br>"
|
"Copyright (C) 2020-2024 Gavin MacGregor<br><br>"
|
||||||
"Released under the GNU General Public License version 3<br>"
|
"Released under the GNU General Public License version 3<br>"
|
||||||
"<a href=\"https://github.com/gkthemac/qteletextmaker\">https://github.com/gkthemac/qteletextmaker</a>"));
|
"<a href=\"https://github.com/gkthemac/qteletextmaker\">https://github.com/gkthemac/qteletextmaker</a>").arg(QApplication::applicationDisplayName()).arg(QApplication::applicationVersion()));
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::init()
|
void MainWindow::init()
|
||||||
@@ -156,36 +263,46 @@ void MainWindow::init()
|
|||||||
addDockWidget(Qt::RightDockWidgetArea, m_x26DockWidget);
|
addDockWidget(Qt::RightDockWidgetArea, m_x26DockWidget);
|
||||||
m_paletteDockWidget = new PaletteDockWidget(m_textWidget);
|
m_paletteDockWidget = new PaletteDockWidget(m_textWidget);
|
||||||
addDockWidget(Qt::RightDockWidgetArea, m_paletteDockWidget);
|
addDockWidget(Qt::RightDockWidgetArea, m_paletteDockWidget);
|
||||||
|
m_pageComposeLinksDockWidget = new PageComposeLinksDockWidget(m_textWidget);
|
||||||
|
addDockWidget(Qt::RightDockWidgetArea, m_pageComposeLinksDockWidget);
|
||||||
|
|
||||||
|
m_textScene = new LevelOneScene(m_textWidget, this);
|
||||||
|
|
||||||
createActions();
|
createActions();
|
||||||
createStatusBar();
|
createStatusBar();
|
||||||
|
|
||||||
readSettings();
|
readSettings();
|
||||||
|
|
||||||
m_textScene = new LevelOneScene(m_textWidget, this);
|
|
||||||
|
|
||||||
m_textView = new QGraphicsView(this);
|
m_textView = new QGraphicsView(this);
|
||||||
m_textView->setScene(m_textScene);
|
m_textView->setScene(m_textScene);
|
||||||
|
if (m_viewSmoothTransform)
|
||||||
m_textView->setRenderHints(QPainter::SmoothPixmapTransform);
|
m_textView->setRenderHints(QPainter::SmoothPixmapTransform);
|
||||||
m_textView->setBackgroundBrush(QBrush(QColor(32, 48, 96)));
|
m_textView->setBackgroundBrush(QBrush(QColor(32, 48, 96)));
|
||||||
setSceneDimensions();
|
setSceneDimensions();
|
||||||
setCentralWidget(m_textView);
|
setCentralWidget(m_textView);
|
||||||
|
|
||||||
connect(m_textWidget->document(), &TeletextDocument::cursorMoved, this, &MainWindow::updateCursorPosition);
|
connect(m_textWidget->document(), &TeletextDocument::cursorMoved, this, &MainWindow::updateCursorPosition);
|
||||||
|
connect(m_textWidget->document(), &TeletextDocument::selectionMoved, m_textScene, &LevelOneScene::updateSelection);
|
||||||
connect(m_textWidget->document()->undoStack(), &QUndoStack::cleanChanged, this, [=]() { setWindowModified(!m_textWidget->document()->undoStack()->isClean()); } );
|
connect(m_textWidget->document()->undoStack(), &QUndoStack::cleanChanged, this, [=]() { setWindowModified(!m_textWidget->document()->undoStack()->isClean()); } );
|
||||||
connect(m_textWidget->document(), &TeletextDocument::aboutToChangeSubPage, m_x26DockWidget, &X26DockWidget::unloadX26List);
|
connect(m_textWidget->document(), &TeletextDocument::aboutToChangeSubPage, m_x26DockWidget, &X26DockWidget::unloadX26List);
|
||||||
connect(m_textWidget->document(), &TeletextDocument::subPageSelected, this, &MainWindow::updatePageWidgets);
|
connect(m_textWidget->document(), &TeletextDocument::subPageSelected, this, &MainWindow::updatePageWidgets);
|
||||||
|
connect(m_textWidget->document(), &TeletextDocument::pageOptionsChanged, this, &MainWindow::updatePageWidgets);
|
||||||
connect(m_textWidget, &TeletextWidget::sizeChanged, this, &MainWindow::setSceneDimensions);
|
connect(m_textWidget, &TeletextWidget::sizeChanged, this, &MainWindow::setSceneDimensions);
|
||||||
connect(m_textWidget->pageRender(), &TeletextPageRender::fullScreenColourChanged, m_textScene, &LevelOneScene::setFullScreenColour);
|
connect(m_textWidget->pageDecode(), &TeletextPageDecode::fullScreenColourChanged, m_textScene, &LevelOneScene::setFullScreenColour);
|
||||||
connect(m_textWidget->pageRender(), &TeletextPageRender::fullRowColourChanged, m_textScene, &LevelOneScene::setFullRowColour);
|
connect(m_textWidget->pageDecode(), &TeletextPageDecode::fullRowColourChanged, m_textScene, &LevelOneScene::setFullRowColour);
|
||||||
connect(m_textWidget, &TeletextWidget::insertKeyPressed, this, &MainWindow::toggleInsertMode);
|
connect(m_textWidget, &TeletextWidget::insertKeyPressed, this, &MainWindow::toggleInsertMode);
|
||||||
|
|
||||||
|
connect(m_textScene, &LevelOneScene::mouseZoomIn, this, &MainWindow::zoomIn);
|
||||||
|
connect(m_textScene, &LevelOneScene::mouseZoomOut, this, &MainWindow::zoomOut);
|
||||||
|
|
||||||
QShortcut *blockShortCut = new QShortcut(QKeySequence(Qt::Key_Escape, Qt::Key_J), m_textView);
|
QShortcut *blockShortCut = new QShortcut(QKeySequence(Qt::Key_Escape, Qt::Key_J), m_textView);
|
||||||
connect(blockShortCut, &QShortcut::activated, [=]() { m_textWidget->setCharacter(0x7f); });
|
connect(blockShortCut, &QShortcut::activated, [=]() { m_textWidget->setCharacter(0x7f); });
|
||||||
|
|
||||||
setUnifiedTitleAndToolBarOnMac(true);
|
setUnifiedTitleAndToolBarOnMac(true);
|
||||||
|
|
||||||
updatePageWidgets();
|
updatePageWidgets();
|
||||||
|
|
||||||
|
m_textView->setFocus();
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::tile(const QMainWindow *previous)
|
void MainWindow::tile(const QMainWindow *previous)
|
||||||
@@ -236,6 +353,11 @@ void MainWindow::createActions()
|
|||||||
saveAsAct->setShortcuts(QKeySequence::SaveAs);
|
saveAsAct->setShortcuts(QKeySequence::SaveAs);
|
||||||
saveAsAct->setStatusTip(tr("Save the document under a new name"));
|
saveAsAct->setStatusTip(tr("Save the document under a new name"));
|
||||||
|
|
||||||
|
const QIcon reloadIcon = QIcon::fromTheme("document-revert");
|
||||||
|
QAction *reloadAct = fileMenu->addAction(reloadIcon, tr("Reload"), this, &MainWindow::reload);
|
||||||
|
reloadAct->setShortcut(QKeySequence(Qt::Key_F5));
|
||||||
|
reloadAct->setStatusTip(tr("Reload the document from disk"));
|
||||||
|
|
||||||
fileMenu->addSeparator();
|
fileMenu->addSeparator();
|
||||||
|
|
||||||
QMenu *recentMenu = fileMenu->addMenu(tr("Recent"));
|
QMenu *recentMenu = fileMenu->addMenu(tr("Recent"));
|
||||||
@@ -251,16 +373,35 @@ void MainWindow::createActions()
|
|||||||
|
|
||||||
setRecentFilesVisible(MainWindow::hasRecentFiles());
|
setRecentFilesVisible(MainWindow::hasRecentFiles());
|
||||||
|
|
||||||
QMenu *exportHashStringSubMenu = fileMenu->addMenu(tr("Export to online editor"));
|
m_exportAutoAct = fileMenu->addAction(tr("Export subpage..."));
|
||||||
|
m_exportAutoAct->setEnabled(false);
|
||||||
|
m_exportAutoAct->setShortcut(tr("Ctrl+E"));
|
||||||
|
m_exportAutoAct->setStatusTip("Export this subpage back to the imported file");
|
||||||
|
connect(fileMenu, &QMenu::aboutToShow, this, &MainWindow::updateExportAutoAction);
|
||||||
|
connect(m_exportAutoAct, &QAction::triggered, this, &MainWindow::exportAuto);
|
||||||
|
|
||||||
|
QAction *exportT42Act = fileMenu->addAction(tr("Export subpage as t42..."));
|
||||||
|
exportT42Act->setStatusTip("Export this subpage as a t42 file");
|
||||||
|
connect(exportT42Act, &QAction::triggered, this, [=]() { exportT42(false); });
|
||||||
|
|
||||||
|
QMenu *exportHashStringSubMenu = fileMenu->addMenu(tr("Export subpage to online editor"));
|
||||||
|
|
||||||
QAction *exportZXNetAct = exportHashStringSubMenu->addAction(tr("Open in zxnet.co.uk"));
|
QAction *exportZXNetAct = exportHashStringSubMenu->addAction(tr("Open in zxnet.co.uk"));
|
||||||
exportZXNetAct->setStatusTip("Export and open page in zxnet.co.uk online editor");
|
exportZXNetAct->setStatusTip("Export and open this subpage in the zxnet.co.uk online editor");
|
||||||
connect(exportZXNetAct, &QAction::triggered, this, &MainWindow::exportZXNet);
|
connect(exportZXNetAct, &QAction::triggered, this, &MainWindow::exportZXNet);
|
||||||
|
|
||||||
QAction *exportEditTFAct = exportHashStringSubMenu->addAction(tr("Open in edit.tf"));
|
QAction *exportEditTFAct = exportHashStringSubMenu->addAction(tr("Open in edit.tf"));
|
||||||
exportEditTFAct->setStatusTip("Export and open page in edit.tf online editor");
|
exportEditTFAct->setStatusTip("Export and open this subpage in the edit.tf online editor");
|
||||||
connect(exportEditTFAct, &QAction::triggered, this, &MainWindow::exportEditTF);
|
connect(exportEditTFAct, &QAction::triggered, this, &MainWindow::exportEditTF);
|
||||||
|
|
||||||
|
QAction *exportPNGAct = fileMenu->addAction(tr("Export subpage as PNG..."));
|
||||||
|
exportPNGAct->setStatusTip("Export a PNG image of this subpage");
|
||||||
|
connect(exportPNGAct, &QAction::triggered, this, &MainWindow::exportPNG);
|
||||||
|
|
||||||
|
QAction *exportM29Act = fileMenu->addAction(tr("Export subpage X/28 as M/29..."));
|
||||||
|
exportM29Act->setStatusTip("Export this subpage's X/28 packets as a tti file with M/29 packets");
|
||||||
|
connect(exportM29Act, &QAction::triggered, this, &MainWindow::exportM29);
|
||||||
|
|
||||||
fileMenu->addSeparator();
|
fileMenu->addSeparator();
|
||||||
|
|
||||||
QAction *closeAct = fileMenu->addAction(tr("&Close"), this, &QWidget::close);
|
QAction *closeAct = fileMenu->addAction(tr("&Close"), this, &QWidget::close);
|
||||||
@@ -291,50 +432,47 @@ void MainWindow::createActions()
|
|||||||
redoAction->setShortcuts(QKeySequence::Redo);
|
redoAction->setShortcuts(QKeySequence::Redo);
|
||||||
|
|
||||||
editMenu->addSeparator();
|
editMenu->addSeparator();
|
||||||
|
|
||||||
#ifndef QT_NO_CLIPBOARD
|
#ifndef QT_NO_CLIPBOARD
|
||||||
/* const QIcon cutIcon = QIcon::fromTheme("edit-cut", QIcon(":/images/cut.png"));
|
const QIcon cutIcon = QIcon::fromTheme("edit-cut", QIcon(":/images/cut.png"));
|
||||||
QAction *cutAct = new QAction(cutIcon, tr("Cu&t"), this);
|
QAction *cutAct = new QAction(cutIcon, tr("Cu&t"), this);
|
||||||
cutAct->setShortcuts(QKeySequence::Cut);
|
cutAct->setShortcuts(QKeySequence::Cut);
|
||||||
cutAct->setStatusTip(tr("Cut the current selection's contents to the "
|
cutAct->setStatusTip(tr("Cut the current selection's contents to the clipboard"));
|
||||||
"clipboard"));
|
connect(cutAct, &QAction::triggered, m_textWidget, &TeletextWidget::cut);
|
||||||
connect(cutAct, &QAction::triggered, textWidget, &QTextEdit::cut);
|
|
||||||
editMenu->addAction(cutAct);
|
editMenu->addAction(cutAct);
|
||||||
editToolBar->addAction(cutAct);
|
editToolBar->addAction(cutAct);
|
||||||
|
|
||||||
const QIcon copyIcon = QIcon::fromTheme("edit-copy", QIcon(":/images/copy.png"));
|
const QIcon copyIcon = QIcon::fromTheme("edit-copy", QIcon(":/images/copy.png"));
|
||||||
QAction *copyAct = new QAction(copyIcon, tr("&Copy"), this);
|
QAction *copyAct = new QAction(copyIcon, tr("&Copy"), this);
|
||||||
copyAct->setShortcuts(QKeySequence::Copy);
|
copyAct->setShortcuts(QKeySequence::Copy);
|
||||||
copyAct->setStatusTip(tr("Copy the current selection's contents to the "
|
copyAct->setStatusTip(tr("Copy the current selection's contents to the clipboard"));
|
||||||
"clipboard"));
|
connect(copyAct, &QAction::triggered, m_textWidget, &TeletextWidget::copy);
|
||||||
connect(copyAct, &QAction::triggered, textWidget, &QTextEdit::copy);
|
|
||||||
editMenu->addAction(copyAct);
|
editMenu->addAction(copyAct);
|
||||||
editToolBar->addAction(copyAct);
|
editToolBar->addAction(copyAct);
|
||||||
|
|
||||||
const QIcon pasteIcon = QIcon::fromTheme("edit-paste", QIcon(":/images/paste.png"));
|
const QIcon pasteIcon = QIcon::fromTheme("edit-paste", QIcon(":/images/paste.png"));
|
||||||
QAction *pasteAct = new QAction(pasteIcon, tr("&Paste"), this);
|
QAction *pasteAct = new QAction(pasteIcon, tr("&Paste"), this);
|
||||||
pasteAct->setShortcuts(QKeySequence::Paste);
|
pasteAct->setShortcuts(QKeySequence::Paste);
|
||||||
pasteAct->setStatusTip(tr("Paste the clipboard's contents into the current "
|
pasteAct->setStatusTip(tr("Paste the clipboard's contents into the current selection"));
|
||||||
"selection"));
|
connect(pasteAct, &QAction::triggered, m_textWidget, &TeletextWidget::paste);
|
||||||
connect(pasteAct, &QAction::triggered, textWidget, &QTextEdit::paste);
|
|
||||||
editMenu->addAction(pasteAct);
|
editMenu->addAction(pasteAct);
|
||||||
editToolBar->addAction(pasteAct);
|
editToolBar->addAction(pasteAct);
|
||||||
|
|
||||||
editMenu->addSeparator();
|
editMenu->addSeparator();
|
||||||
*/
|
|
||||||
#endif // !QT_NO_CLIPBOARD
|
#endif // !QT_NO_CLIPBOARD
|
||||||
|
|
||||||
QAction *insertBlankRowAct = editMenu->addAction(tr("Insert blank row"));
|
QAction *insertBlankRowAct = editMenu->addAction(tr("Insert blank row"));
|
||||||
insertBlankRowAct->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_I));
|
insertBlankRowAct->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_I));
|
||||||
insertBlankRowAct->setStatusTip(tr("Insert a blank row at the cursor position"));
|
insertBlankRowAct->setStatusTip(tr("Insert a blank row at the cursor position"));
|
||||||
connect(insertBlankRowAct, &QAction::triggered, [=]() { insertRow(false); } );
|
connect(insertBlankRowAct, &QAction::triggered, [=]() { insertRow(false); } );
|
||||||
|
|
||||||
QAction *insertCopyRowAct = editMenu->addAction(tr("Insert copy row"));
|
QAction *insertCopyRowAct = editMenu->addAction(tr("Insert copy row"));
|
||||||
insertCopyRowAct->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_I));
|
insertCopyRowAct->setShortcut(QKeySequence(Qt::CTRL | Qt::SHIFT | Qt::Key_I));
|
||||||
insertCopyRowAct->setStatusTip(tr("Insert a row that's a copy of the row at the cursor position"));
|
insertCopyRowAct->setStatusTip(tr("Insert a row that's a copy of the row at the cursor position"));
|
||||||
connect(insertCopyRowAct, &QAction::triggered, [=]() { insertRow(true); } );
|
connect(insertCopyRowAct, &QAction::triggered, [=]() { insertRow(true); } );
|
||||||
|
|
||||||
QAction *deleteRowAct = editMenu->addAction(tr("Delete row"));
|
QAction *deleteRowAct = editMenu->addAction(tr("Delete row"));
|
||||||
deleteRowAct->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_D));
|
deleteRowAct->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_D));
|
||||||
deleteRowAct->setStatusTip(tr("Delete the row at the cursor position"));
|
deleteRowAct->setStatusTip(tr("Delete the row at the cursor position"));
|
||||||
connect(deleteRowAct, &QAction::triggered, this, &MainWindow::deleteRow);
|
connect(deleteRowAct, &QAction::triggered, this, &MainWindow::deleteRow);
|
||||||
|
|
||||||
@@ -360,27 +498,28 @@ void MainWindow::createActions()
|
|||||||
|
|
||||||
QAction *revealAct = viewMenu->addAction(tr("&Reveal"));
|
QAction *revealAct = viewMenu->addAction(tr("&Reveal"));
|
||||||
revealAct->setCheckable(true);
|
revealAct->setCheckable(true);
|
||||||
revealAct->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_R));
|
revealAct->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_R));
|
||||||
revealAct->setStatusTip(tr("Toggle reveal"));
|
revealAct->setStatusTip(tr("Toggle reveal"));
|
||||||
connect(revealAct, &QAction::toggled, m_textWidget, &TeletextWidget::toggleReveal);
|
connect(revealAct, &QAction::toggled, m_textWidget, &TeletextWidget::setReveal);
|
||||||
|
|
||||||
QAction *mixAct = viewMenu->addAction(tr("&Mix"));
|
QAction *mixAct = viewMenu->addAction(tr("&Mix"));
|
||||||
mixAct->setCheckable(true);
|
mixAct->setCheckable(true);
|
||||||
mixAct->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_M));
|
mixAct->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_M));
|
||||||
mixAct->setStatusTip(tr("Toggle mix"));
|
mixAct->setStatusTip(tr("Toggle mix"));
|
||||||
connect(mixAct, &QAction::toggled, m_textWidget, &TeletextWidget::toggleMix);
|
connect(mixAct, &QAction::toggled, m_textWidget, &TeletextWidget::setMix);
|
||||||
|
connect(mixAct, &QAction::toggled, m_textScene, &LevelOneScene::setMix);
|
||||||
|
|
||||||
QAction *gridAct = viewMenu->addAction(tr("&Grid"));
|
QAction *gridAct = viewMenu->addAction(tr("&Grid"));
|
||||||
gridAct->setCheckable(true);
|
gridAct->setCheckable(true);
|
||||||
gridAct->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_G));
|
gridAct->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_G));
|
||||||
gridAct->setStatusTip(tr("Toggle the text grid"));
|
gridAct->setStatusTip(tr("Toggle the text grid"));
|
||||||
connect(gridAct, &QAction::toggled, m_textWidget, &TeletextWidget::toggleGrid);
|
connect(gridAct, &QAction::toggled, m_textScene, &LevelOneScene::toggleGrid);
|
||||||
|
|
||||||
QAction *showCodesAct = viewMenu->addAction(tr("Show codes"));
|
QAction *showControlCodesAct = viewMenu->addAction(tr("Show control codes"));
|
||||||
showCodesAct->setCheckable(true);
|
showControlCodesAct->setCheckable(true);
|
||||||
showCodesAct->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_T));
|
showControlCodesAct->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_T));
|
||||||
showCodesAct->setStatusTip(tr("Toggle showing of control codes"));
|
showControlCodesAct->setStatusTip(tr("Toggle showing of control codes"));
|
||||||
connect(showCodesAct, &QAction::toggled, m_textWidget->pageRender(), &TeletextPageRender::setShowCodes);
|
connect(showControlCodesAct, &QAction::toggled, m_textWidget, &TeletextWidget::setShowControlCodes);
|
||||||
|
|
||||||
viewMenu->addSeparator();
|
viewMenu->addSeparator();
|
||||||
|
|
||||||
@@ -415,6 +554,13 @@ void MainWindow::createActions()
|
|||||||
borderGroup->addAction(m_borderActs[i]);
|
borderGroup->addAction(m_borderActs[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
viewMenu->addSeparator();
|
||||||
|
|
||||||
|
m_smoothTransformAction = viewMenu->addAction(tr("Smooth font scaling"));
|
||||||
|
m_smoothTransformAction->setCheckable(true);
|
||||||
|
m_smoothTransformAction->setStatusTip(tr("Toggle smooth font scaling"));
|
||||||
|
connect(m_smoothTransformAction, &QAction::toggled, this, &MainWindow::setSmoothTransform);
|
||||||
|
|
||||||
QAction *zoomInAct = viewMenu->addAction(tr("Zoom In"));
|
QAction *zoomInAct = viewMenu->addAction(tr("Zoom In"));
|
||||||
zoomInAct->setShortcuts(QKeySequence::ZoomIn);
|
zoomInAct->setShortcuts(QKeySequence::ZoomIn);
|
||||||
zoomInAct->setStatusTip(tr("Zoom in"));
|
zoomInAct->setStatusTip(tr("Zoom in"));
|
||||||
@@ -426,7 +572,7 @@ void MainWindow::createActions()
|
|||||||
connect(zoomOutAct, &QAction::triggered, this, &MainWindow::zoomOut);
|
connect(zoomOutAct, &QAction::triggered, this, &MainWindow::zoomOut);
|
||||||
|
|
||||||
QAction *zoomResetAct = viewMenu->addAction(tr("Reset zoom"));
|
QAction *zoomResetAct = viewMenu->addAction(tr("Reset zoom"));
|
||||||
zoomResetAct->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_0));
|
zoomResetAct->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_0));
|
||||||
zoomResetAct->setStatusTip(tr("Reset zoom level"));
|
zoomResetAct->setStatusTip(tr("Reset zoom level"));
|
||||||
connect(zoomResetAct, &QAction::triggered, this, &MainWindow::zoomReset);
|
connect(zoomResetAct, &QAction::triggered, this, &MainWindow::zoomReset);
|
||||||
|
|
||||||
@@ -438,19 +584,19 @@ void MainWindow::createActions()
|
|||||||
const char *colours[] = { "Black", "Red", "Green", "Yellow", "Blue", "Magenta", "Cyan", "White" };
|
const char *colours[] = { "Black", "Red", "Green", "Yellow", "Blue", "Magenta", "Cyan", "White" };
|
||||||
|
|
||||||
QAction *alphaColour = alphaColourSubMenu->addAction(tr(colours[i]));
|
QAction *alphaColour = alphaColourSubMenu->addAction(tr(colours[i]));
|
||||||
alphaColour->setShortcut(QKeySequence(Qt::Key_Escape, Qt::Key_0 + i));
|
alphaColour->setShortcut(QKeySequence(QString("Esc, %1").arg(i)));
|
||||||
alphaColour->setStatusTip(QString("Insert alphanumeric %1 attribute").arg(QString(colours[i]).toLower()));
|
alphaColour->setStatusTip(QString("Insert alphanumeric %1 attribute").arg(QString(colours[i]).toLower()));
|
||||||
connect(alphaColour, &QAction::triggered, [=]() { m_textWidget->setCharacter(i); });
|
connect(alphaColour, &QAction::triggered, [=]() { m_textWidget->setCharacter(i); });
|
||||||
|
|
||||||
QAction *mosaicColour = mosaicColourSubMenu->addAction(tr(colours[i]));
|
QAction *mosaicColour = mosaicColourSubMenu->addAction(tr(colours[i]));
|
||||||
mosaicColour->setShortcut(QKeySequence(Qt::Key_Escape, Qt::SHIFT + Qt::Key_0 + i));
|
mosaicColour->setShortcut(QKeySequence(QString("Esc, Shift+%1").arg(i)));
|
||||||
mosaicColour->setStatusTip(QString("Insert mosaic %1 attribute").arg(QString(colours[i]).toLower()));
|
mosaicColour->setStatusTip(QString("Insert mosaic %1 attribute").arg(QString(colours[i]).toLower()));
|
||||||
connect(mosaicColour, &QAction::triggered, [=]() { m_textWidget->setCharacter(i+0x10); });
|
connect(mosaicColour, &QAction::triggered, [=]() { m_textWidget->setCharacter(i+0x10); });
|
||||||
}
|
}
|
||||||
|
|
||||||
QMenu *mosaicsStyleSubMenu = insertMenu->addMenu(tr("Mosaics style"));
|
QMenu *mosaicsStyleSubMenu = insertMenu->addMenu(tr("Mosaics style"));
|
||||||
QAction *mosaicsSeparatedAct = mosaicsStyleSubMenu->addAction(tr("Separated mosaics"));
|
QAction *mosaicsSeparatedAct = mosaicsStyleSubMenu->addAction(tr("Separated mosaics"));
|
||||||
mosaicsSeparatedAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::SHIFT + Qt::Key_S));
|
mosaicsSeparatedAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::SHIFT | Qt::Key_S));
|
||||||
mosaicsSeparatedAct->setStatusTip(tr("Insert separated mosaics attribute"));
|
mosaicsSeparatedAct->setStatusTip(tr("Insert separated mosaics attribute"));
|
||||||
connect(mosaicsSeparatedAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1a); });
|
connect(mosaicsSeparatedAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1a); });
|
||||||
QAction *mosaicsContiguousAct = mosaicsStyleSubMenu->addAction(tr("Contiguous mosaics"));
|
QAction *mosaicsContiguousAct = mosaicsStyleSubMenu->addAction(tr("Contiguous mosaics"));
|
||||||
@@ -460,7 +606,7 @@ void MainWindow::createActions()
|
|||||||
|
|
||||||
QMenu *mosaicsHoldSubMenu = insertMenu->addMenu(tr("Mosaics hold"));
|
QMenu *mosaicsHoldSubMenu = insertMenu->addMenu(tr("Mosaics hold"));
|
||||||
QAction *mosaicsHoldAct = mosaicsHoldSubMenu->addAction(tr("Hold mosaics"));
|
QAction *mosaicsHoldAct = mosaicsHoldSubMenu->addAction(tr("Hold mosaics"));
|
||||||
mosaicsHoldAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::SHIFT + Qt::Key_H));
|
mosaicsHoldAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::SHIFT | Qt::Key_H));
|
||||||
mosaicsHoldAct->setStatusTip(tr("Insert hold mosaics attribute"));
|
mosaicsHoldAct->setStatusTip(tr("Insert hold mosaics attribute"));
|
||||||
connect(mosaicsHoldAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1e); });
|
connect(mosaicsHoldAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1e); });
|
||||||
QAction *mosaicsReleaseAct = mosaicsHoldSubMenu->addAction(tr("Release mosaics"));
|
QAction *mosaicsReleaseAct = mosaicsHoldSubMenu->addAction(tr("Release mosaics"));
|
||||||
@@ -470,7 +616,7 @@ void MainWindow::createActions()
|
|||||||
|
|
||||||
QMenu *backgroundColourSubMenu = insertMenu->addMenu(tr("Background colour"));
|
QMenu *backgroundColourSubMenu = insertMenu->addMenu(tr("Background colour"));
|
||||||
QAction *backgroundNewAct = backgroundColourSubMenu->addAction(tr("New background"));
|
QAction *backgroundNewAct = backgroundColourSubMenu->addAction(tr("New background"));
|
||||||
backgroundNewAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::SHIFT + Qt::Key_N));
|
backgroundNewAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::SHIFT | Qt::Key_N));
|
||||||
backgroundNewAct->setStatusTip(tr("Insert new background attribute"));
|
backgroundNewAct->setStatusTip(tr("Insert new background attribute"));
|
||||||
connect(backgroundNewAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1d); });
|
connect(backgroundNewAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1d); });
|
||||||
QAction *backgroundBlackAct = backgroundColourSubMenu->addAction(tr("Black background"));
|
QAction *backgroundBlackAct = backgroundColourSubMenu->addAction(tr("Black background"));
|
||||||
@@ -484,15 +630,15 @@ void MainWindow::createActions()
|
|||||||
textSizeNormalAct->setStatusTip(tr("Insert normal size attribute"));
|
textSizeNormalAct->setStatusTip(tr("Insert normal size attribute"));
|
||||||
connect(textSizeNormalAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0c); });
|
connect(textSizeNormalAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0c); });
|
||||||
QAction *textSizeDoubleHeightAct = textSizeSubMenu->addAction(tr("Double height"));
|
QAction *textSizeDoubleHeightAct = textSizeSubMenu->addAction(tr("Double height"));
|
||||||
textSizeDoubleHeightAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::SHIFT + Qt::Key_D));
|
textSizeDoubleHeightAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::SHIFT | Qt::Key_D));
|
||||||
textSizeDoubleHeightAct->setStatusTip(tr("Insert double height attribute"));
|
textSizeDoubleHeightAct->setStatusTip(tr("Insert double height attribute"));
|
||||||
connect(textSizeDoubleHeightAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0d); });
|
connect(textSizeDoubleHeightAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0d); });
|
||||||
QAction *textSizeDoubleWidthAct = textSizeSubMenu->addAction(tr("Double width"));
|
QAction *textSizeDoubleWidthAct = textSizeSubMenu->addAction(tr("Double width"));
|
||||||
textSizeDoubleWidthAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::CTRL + Qt::Key_D));
|
textSizeDoubleWidthAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::CTRL | Qt::Key_D));
|
||||||
textSizeDoubleWidthAct->setStatusTip(tr("Insert double width attribute"));
|
textSizeDoubleWidthAct->setStatusTip(tr("Insert double width attribute"));
|
||||||
connect(textSizeDoubleWidthAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0e); });
|
connect(textSizeDoubleWidthAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0e); });
|
||||||
QAction *textSizeDoubleSizeAct = textSizeSubMenu->addAction(tr("Double size"));
|
QAction *textSizeDoubleSizeAct = textSizeSubMenu->addAction(tr("Double size"));
|
||||||
textSizeDoubleSizeAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::CTRL + Qt::SHIFT + Qt::Key_D));
|
textSizeDoubleSizeAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::CTRL | Qt::SHIFT | Qt::Key_D));
|
||||||
textSizeDoubleSizeAct->setStatusTip(tr("Insert double size attribute"));
|
textSizeDoubleSizeAct->setStatusTip(tr("Insert double size attribute"));
|
||||||
connect(textSizeDoubleSizeAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0f); });
|
connect(textSizeDoubleSizeAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0f); });
|
||||||
|
|
||||||
@@ -503,7 +649,7 @@ void MainWindow::createActions()
|
|||||||
|
|
||||||
QMenu *flashSubMenu = insertMenu->addMenu(tr("Flash"));
|
QMenu *flashSubMenu = insertMenu->addMenu(tr("Flash"));
|
||||||
QAction *flashFlashingAct = flashSubMenu->addAction(tr("Flashing"));
|
QAction *flashFlashingAct = flashSubMenu->addAction(tr("Flashing"));
|
||||||
flashFlashingAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::SHIFT + Qt::Key_F));
|
flashFlashingAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::SHIFT | Qt::Key_F));
|
||||||
flashFlashingAct->setStatusTip(tr("Insert flashing attribute"));
|
flashFlashingAct->setStatusTip(tr("Insert flashing attribute"));
|
||||||
connect(flashFlashingAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x08); });
|
connect(flashFlashingAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x08); });
|
||||||
QAction *flashSteadyAct = flashSubMenu->addAction(tr("Steady"));
|
QAction *flashSteadyAct = flashSubMenu->addAction(tr("Steady"));
|
||||||
@@ -513,7 +659,7 @@ void MainWindow::createActions()
|
|||||||
|
|
||||||
QMenu *boxingSubMenu = insertMenu->addMenu(tr("Box"));
|
QMenu *boxingSubMenu = insertMenu->addMenu(tr("Box"));
|
||||||
QAction *boxingStartAct = boxingSubMenu->addAction(tr("Start box"));
|
QAction *boxingStartAct = boxingSubMenu->addAction(tr("Start box"));
|
||||||
boxingStartAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::SHIFT + Qt::Key_X));
|
boxingStartAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::SHIFT | Qt::Key_X));
|
||||||
boxingStartAct->setStatusTip(tr("Insert start box attribute"));
|
boxingStartAct->setStatusTip(tr("Insert start box attribute"));
|
||||||
connect(boxingStartAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0b); });
|
connect(boxingStartAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0b); });
|
||||||
QAction *boxingEndAct = boxingSubMenu->addAction(tr("End box"));
|
QAction *boxingEndAct = boxingSubMenu->addAction(tr("End box"));
|
||||||
@@ -522,7 +668,7 @@ void MainWindow::createActions()
|
|||||||
connect(boxingEndAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0a); });
|
connect(boxingEndAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x0a); });
|
||||||
|
|
||||||
QAction *escSwitchAct = insertMenu->addAction(tr("ESC/switch"));
|
QAction *escSwitchAct = insertMenu->addAction(tr("ESC/switch"));
|
||||||
escSwitchAct->setShortcut(QKeySequence(Qt::Key_Escape, Qt::CTRL + Qt::Key_S));
|
escSwitchAct->setShortcut(QKeySequence(Qt::NoModifier | Qt::Key_Escape, Qt::CTRL | Qt::Key_S));
|
||||||
escSwitchAct->setStatusTip(tr("Insert ESC/switch character set attribute"));
|
escSwitchAct->setStatusTip(tr("Insert ESC/switch character set attribute"));
|
||||||
connect(escSwitchAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1b); });
|
connect(escSwitchAct, &QAction::triggered, [=]() { m_textWidget->setCharacter(0x1b); });
|
||||||
|
|
||||||
@@ -532,6 +678,7 @@ void MainWindow::createActions()
|
|||||||
toolsMenu->addAction(m_x26DockWidget->toggleViewAction());
|
toolsMenu->addAction(m_x26DockWidget->toggleViewAction());
|
||||||
toolsMenu->addAction(m_pageEnhancementsDockWidget->toggleViewAction());
|
toolsMenu->addAction(m_pageEnhancementsDockWidget->toggleViewAction());
|
||||||
toolsMenu->addAction(m_paletteDockWidget->toggleViewAction());
|
toolsMenu->addAction(m_paletteDockWidget->toggleViewAction());
|
||||||
|
toolsMenu->addAction(m_pageComposeLinksDockWidget->toggleViewAction());
|
||||||
|
|
||||||
//FIXME is this main menubar separator to put help menu towards the right?
|
//FIXME is this main menubar separator to put help menu towards the right?
|
||||||
menuBar()->addSeparator();
|
menuBar()->addSeparator();
|
||||||
@@ -552,7 +699,6 @@ void MainWindow::createActions()
|
|||||||
|
|
||||||
void MainWindow::setSceneDimensions()
|
void MainWindow::setSceneDimensions()
|
||||||
{
|
{
|
||||||
const float aspectRatioHorizontalScaling[4] = { 0.6, 0.6, 0.8, 0.5 };
|
|
||||||
const int topBottomBorders[3] = { 0, 10, 19 };
|
const int topBottomBorders[3] = { 0, 10, 19 };
|
||||||
const int pillarBoxSizes[3] = { 672, 720, 854 };
|
const int pillarBoxSizes[3] = { 672, 720, 854 };
|
||||||
const int leftRightBorders[3] = { 0, 24, 77 };
|
const int leftRightBorders[3] = { 0, 24, 77 };
|
||||||
@@ -568,7 +714,7 @@ void MainWindow::setSceneDimensions()
|
|||||||
else
|
else
|
||||||
newSceneWidth = m_textWidget->width() + leftRightBorders[m_viewBorder]*2;
|
newSceneWidth = m_textWidget->width() + leftRightBorders[m_viewBorder]*2;
|
||||||
|
|
||||||
m_textScene->setDimensions(newSceneWidth, 250+topBottomBorders[m_viewBorder]*2, m_textWidget->width());
|
m_textScene->setBorderDimensions(newSceneWidth, 250+topBottomBorders[m_viewBorder]*2, m_textWidget->width(), m_textWidget->pageDecode()->leftSidePanelColumns(), m_textWidget->pageDecode()->rightSidePanelColumns());
|
||||||
m_textView->setTransform(QTransform((1+(float)m_viewZoom/2)*aspectRatioHorizontalScaling[m_viewAspectRatio], 0, 0, 1+(float)m_viewZoom/2, 0, 0));
|
m_textView->setTransform(QTransform((1+(float)m_viewZoom/2)*aspectRatioHorizontalScaling[m_viewAspectRatio], 0, 0, 1+(float)m_viewZoom/2, 0, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -576,20 +722,17 @@ void MainWindow::insertRow(bool copyRow)
|
|||||||
{
|
{
|
||||||
if (m_textWidget->document()->cursorRow() == 24)
|
if (m_textWidget->document()->cursorRow() == 24)
|
||||||
return;
|
return;
|
||||||
QUndoCommand *insertRowCommand = new InsertRowCommand(m_textWidget->document(), copyRow);
|
m_textWidget->document()->undoStack()->push(new InsertRowCommand(m_textWidget->document(), copyRow));
|
||||||
m_textWidget->document()->undoStack()->push(insertRowCommand);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::deleteRow()
|
void MainWindow::deleteRow()
|
||||||
{
|
{
|
||||||
QUndoCommand *deleteRowCommand = new DeleteRowCommand(m_textWidget->document());
|
m_textWidget->document()->undoStack()->push(new DeleteRowCommand(m_textWidget->document()));
|
||||||
m_textWidget->document()->undoStack()->push(deleteRowCommand);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::insertSubPage(bool afterCurrentSubPage, bool copyCurrentSubPage)
|
void MainWindow::insertSubPage(bool afterCurrentSubPage, bool copyCurrentSubPage)
|
||||||
{
|
{
|
||||||
QUndoCommand *insertSubPageCommand = new InsertSubPageCommand(m_textWidget->document(), afterCurrentSubPage, copyCurrentSubPage);
|
m_textWidget->document()->undoStack()->push(new InsertSubPageCommand(m_textWidget->document(), afterCurrentSubPage, copyCurrentSubPage));
|
||||||
m_textWidget->document()->undoStack()->push(insertSubPageCommand);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::deleteSubPage()
|
void MainWindow::deleteSubPage()
|
||||||
@@ -612,16 +755,29 @@ void MainWindow::setAspectRatio(int newViewAspectRatio)
|
|||||||
setSceneDimensions();
|
setSceneDimensions();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainWindow::setSmoothTransform(bool smoothTransform)
|
||||||
|
{
|
||||||
|
m_viewSmoothTransform = smoothTransform;
|
||||||
|
if (smoothTransform)
|
||||||
|
m_textView->setRenderHints(QPainter::SmoothPixmapTransform);
|
||||||
|
else
|
||||||
|
m_textView->setRenderHints({ });
|
||||||
|
}
|
||||||
|
|
||||||
void MainWindow::zoomIn()
|
void MainWindow::zoomIn()
|
||||||
{
|
{
|
||||||
if (m_viewZoom < 4)
|
if (m_viewZoom < 4)
|
||||||
m_viewZoom++;
|
m_viewZoom++;
|
||||||
|
else if (m_viewZoom < 12)
|
||||||
|
m_viewZoom += 2;
|
||||||
setSceneDimensions();
|
setSceneDimensions();
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::zoomOut()
|
void MainWindow::zoomOut()
|
||||||
{
|
{
|
||||||
if (m_viewZoom > 0)
|
if (m_viewZoom > 4)
|
||||||
|
m_viewZoom -= 2;
|
||||||
|
else if (m_viewZoom > 0)
|
||||||
m_viewZoom--;
|
m_viewZoom--;
|
||||||
setSceneDimensions();
|
setSceneDimensions();
|
||||||
}
|
}
|
||||||
@@ -643,33 +799,32 @@ void MainWindow::toggleInsertMode()
|
|||||||
|
|
||||||
void MainWindow::createStatusBar()
|
void MainWindow::createStatusBar()
|
||||||
{
|
{
|
||||||
QLabel *subPageLabel = new QLabel("Subpage");
|
|
||||||
statusBar()->insertWidget(0, subPageLabel);
|
|
||||||
|
|
||||||
m_previousSubPageButton = new QToolButton;
|
m_previousSubPageButton = new QToolButton;
|
||||||
m_previousSubPageButton->setMinimumSize(subPageLabel->height(), subPageLabel->height());
|
|
||||||
m_previousSubPageButton->setMaximumSize(subPageLabel->height(), subPageLabel->height());
|
|
||||||
m_previousSubPageButton->setAutoRaise(true);
|
m_previousSubPageButton->setAutoRaise(true);
|
||||||
m_previousSubPageButton->setArrowType(Qt::LeftArrow);
|
m_previousSubPageButton->setArrowType(Qt::LeftArrow);
|
||||||
statusBar()->insertWidget(1, m_previousSubPageButton);
|
statusBar()->insertWidget(0, m_previousSubPageButton);
|
||||||
connect(m_previousSubPageButton, &QToolButton::clicked, m_textWidget->document(), &TeletextDocument::selectSubPagePrevious);
|
connect(m_previousSubPageButton, &QToolButton::clicked, m_textWidget->document(), &TeletextDocument::selectSubPagePrevious);
|
||||||
|
|
||||||
m_subPageLabel = new QLabel("1/1");
|
m_subPageLabel = new QLabel("1/1");
|
||||||
statusBar()->insertWidget(2, m_subPageLabel);
|
statusBar()->insertWidget(1, m_subPageLabel);
|
||||||
|
|
||||||
m_nextSubPageButton = new QToolButton;
|
m_nextSubPageButton = new QToolButton;
|
||||||
m_nextSubPageButton->setMinimumSize(subPageLabel->height(), subPageLabel->height());
|
m_previousSubPageButton->setMinimumSize(m_subPageLabel->height(), m_subPageLabel->height());
|
||||||
m_nextSubPageButton->setMaximumSize(subPageLabel->height(), subPageLabel->height());
|
m_previousSubPageButton->setMaximumSize(m_subPageLabel->height(), m_subPageLabel->height());
|
||||||
|
m_nextSubPageButton->setMinimumSize(m_subPageLabel->height(), m_subPageLabel->height());
|
||||||
|
m_nextSubPageButton->setMaximumSize(m_subPageLabel->height(), m_subPageLabel->height());
|
||||||
m_nextSubPageButton->setAutoRaise(true);
|
m_nextSubPageButton->setAutoRaise(true);
|
||||||
m_nextSubPageButton->setArrowType(Qt::RightArrow);
|
m_nextSubPageButton->setArrowType(Qt::RightArrow);
|
||||||
statusBar()->insertWidget(3, m_nextSubPageButton);
|
statusBar()->insertWidget(2, m_nextSubPageButton);
|
||||||
connect(m_nextSubPageButton, &QToolButton::clicked, m_textWidget->document(), &TeletextDocument::selectSubPageNext);
|
connect(m_nextSubPageButton, &QToolButton::clicked, m_textWidget->document(), &TeletextDocument::selectSubPageNext);
|
||||||
|
|
||||||
m_cursorPositionLabel = new QLabel("Row 1 Column 1");
|
m_cursorPositionLabel = new QLabel("1, 1");
|
||||||
statusBar()->insertWidget(4, m_cursorPositionLabel);
|
statusBar()->insertWidget(3, m_cursorPositionLabel);
|
||||||
|
|
||||||
m_insertModePushButton = new QPushButton("OVERWRITE");
|
m_insertModePushButton = new QPushButton("OVERWRITE");
|
||||||
m_insertModePushButton->setFlat(true);
|
m_insertModePushButton->setFlat(true);
|
||||||
|
m_insertModePushButton->setMinimumHeight(m_subPageLabel->height());
|
||||||
|
m_insertModePushButton->setMaximumHeight(m_subPageLabel->height());
|
||||||
m_insertModePushButton->setFocusProxy(m_textWidget);
|
m_insertModePushButton->setFocusProxy(m_textWidget);
|
||||||
statusBar()->addPermanentWidget(m_insertModePushButton);
|
statusBar()->addPermanentWidget(m_insertModePushButton);
|
||||||
connect(m_insertModePushButton, &QPushButton::clicked, this, &MainWindow::toggleInsertMode);
|
connect(m_insertModePushButton, &QPushButton::clicked, this, &MainWindow::toggleInsertMode);
|
||||||
@@ -685,10 +840,11 @@ void MainWindow::createStatusBar()
|
|||||||
statusBar()->addPermanentWidget(m_levelRadioButton[i]);
|
statusBar()->addPermanentWidget(m_levelRadioButton[i]);
|
||||||
}
|
}
|
||||||
m_levelRadioButton[0]->toggle();
|
m_levelRadioButton[0]->toggle();
|
||||||
connect(m_levelRadioButton[0], &QAbstractButton::clicked, [=]() { m_textWidget->pageRender()->setRenderLevel(0); m_textWidget->update(); });
|
connect(m_levelRadioButton[0], &QAbstractButton::clicked, [=]() { m_textWidget->pageDecode()->setLevel(0); m_textWidget->update(); m_paletteDockWidget->setLevel3p5Accepted(false); });
|
||||||
connect(m_levelRadioButton[1], &QAbstractButton::clicked, [=]() { m_textWidget->pageRender()->setRenderLevel(1); m_textWidget->update(); });
|
connect(m_levelRadioButton[1], &QAbstractButton::clicked, [=]() { m_textWidget->pageDecode()->setLevel(1); m_textWidget->update(); m_paletteDockWidget->setLevel3p5Accepted(false);});
|
||||||
connect(m_levelRadioButton[2], &QAbstractButton::clicked, [=]() { m_textWidget->pageRender()->setRenderLevel(2); m_textWidget->update(); });
|
connect(m_levelRadioButton[2], &QAbstractButton::clicked, [=]() { m_textWidget->pageDecode()->setLevel(2); m_textWidget->update(); m_paletteDockWidget->setLevel3p5Accepted(false);});
|
||||||
connect(m_levelRadioButton[3], &QAbstractButton::clicked, [=]() { m_textWidget->pageRender()->setRenderLevel(3); m_textWidget->update(); });
|
connect(m_levelRadioButton[3], &QAbstractButton::clicked, [=]() { m_textWidget->pageDecode()->setLevel(3); m_textWidget->update(); m_paletteDockWidget->setLevel3p5Accepted(true);});
|
||||||
|
|
||||||
statusBar()->showMessage(tr("Ready"));
|
statusBar()->showMessage(tr("Ready"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -705,8 +861,12 @@ void MainWindow::readSettings()
|
|||||||
m_viewAspectRatio = settings.value("aspectratio", 0).toInt();
|
m_viewAspectRatio = settings.value("aspectratio", 0).toInt();
|
||||||
m_viewAspectRatio = (m_viewAspectRatio < 0 || m_viewAspectRatio > 2) ? 0 : m_viewAspectRatio;
|
m_viewAspectRatio = (m_viewAspectRatio < 0 || m_viewAspectRatio > 2) ? 0 : m_viewAspectRatio;
|
||||||
m_aspectRatioActs[m_viewAspectRatio]->setChecked(true);
|
m_aspectRatioActs[m_viewAspectRatio]->setChecked(true);
|
||||||
|
m_viewSmoothTransform = settings.value("smoothTransform", 0).toBool();
|
||||||
|
m_smoothTransformAction->blockSignals(true);
|
||||||
|
m_smoothTransformAction->setChecked(m_viewSmoothTransform);
|
||||||
|
m_smoothTransformAction->blockSignals(false);
|
||||||
m_viewZoom = settings.value("zoom", 2).toInt();
|
m_viewZoom = settings.value("zoom", 2).toInt();
|
||||||
m_viewZoom = (m_viewZoom < 0 || m_viewZoom > 4) ? 2 : m_viewZoom;
|
m_viewZoom = (m_viewZoom < 0 || m_viewZoom > 12) ? 2 : m_viewZoom;
|
||||||
|
|
||||||
// zoom 0 = 420,426px, 1 = 620,570px, 2 = 780,720px
|
// zoom 0 = 420,426px, 1 = 620,570px, 2 = 780,720px
|
||||||
if (geometry.isEmpty()) {
|
if (geometry.isEmpty()) {
|
||||||
@@ -732,6 +892,8 @@ void MainWindow::readSettings()
|
|||||||
m_x26DockWidget->setFloating(true);
|
m_x26DockWidget->setFloating(true);
|
||||||
m_paletteDockWidget->hide();
|
m_paletteDockWidget->hide();
|
||||||
m_paletteDockWidget->setFloating(true);
|
m_paletteDockWidget->setFloating(true);
|
||||||
|
m_pageComposeLinksDockWidget->hide();
|
||||||
|
m_pageComposeLinksDockWidget->setFloating(true);
|
||||||
} else
|
} else
|
||||||
restoreState(windowState);
|
restoreState(windowState);
|
||||||
}
|
}
|
||||||
@@ -743,6 +905,7 @@ void MainWindow::writeSettings()
|
|||||||
settings.setValue("windowState", saveState());
|
settings.setValue("windowState", saveState());
|
||||||
settings.setValue("border", m_viewBorder);
|
settings.setValue("border", m_viewBorder);
|
||||||
settings.setValue("aspectratio", m_viewAspectRatio);
|
settings.setValue("aspectratio", m_viewAspectRatio);
|
||||||
|
settings.setValue("smoothTransform", m_viewSmoothTransform);
|
||||||
settings.setValue("zoom", m_viewZoom);
|
settings.setValue("zoom", m_viewZoom);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -750,7 +913,7 @@ bool MainWindow::maybeSave()
|
|||||||
{
|
{
|
||||||
if (m_textWidget->document()->undoStack()->isClean())
|
if (m_textWidget->document()->undoStack()->isClean())
|
||||||
return true;
|
return true;
|
||||||
const QMessageBox::StandardButton ret = QMessageBox::warning(this, tr("QTeletextMaker"), tr("The document has been modified.\nDo you want to save your changes?"), QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel);
|
const QMessageBox::StandardButton ret = QMessageBox::warning(this, QApplication::applicationDisplayName(), tr("The document \"%1\" has been modified.\nDo you want to save your changes or discard them?").arg(QFileInfo(m_curFile).fileName()), QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel);
|
||||||
switch (ret) {
|
switch (ret) {
|
||||||
case QMessageBox::Save:
|
case QMessageBox::Save:
|
||||||
return save();
|
return save();
|
||||||
@@ -767,17 +930,35 @@ void MainWindow::loadFile(const QString &fileName)
|
|||||||
int levelSeen;
|
int levelSeen;
|
||||||
|
|
||||||
QFile file(fileName);
|
QFile file(fileName);
|
||||||
if (!file.open(QFile::ReadOnly | QFile::Text)) {
|
const QFileInfo fileInfo(file);
|
||||||
QMessageBox::warning(this, tr("QTeletextMaker"), tr("Cannot read file %1:\n%2.").arg(QDir::toNativeSeparators(fileName), file.errorString()));
|
QIODevice::OpenMode fileOpenMode;
|
||||||
|
|
||||||
|
if (fileInfo.suffix() == "t42")
|
||||||
|
fileOpenMode = QFile::ReadOnly;
|
||||||
|
else
|
||||||
|
fileOpenMode = QFile::ReadOnly | QFile::Text;
|
||||||
|
|
||||||
|
if (!file.open(fileOpenMode)) {
|
||||||
|
QMessageBox::warning(this, QApplication::applicationDisplayName(), tr("Cannot read file %1:\n%2.").arg(QDir::toNativeSeparators(fileName), file.errorString()));
|
||||||
setCurrentFile(QString());
|
setCurrentFile(QString());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QApplication::setOverrideCursor(Qt::WaitCursor);
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
||||||
|
|
||||||
|
if (fileInfo.suffix() == "t42") {
|
||||||
|
importT42(&file, m_textWidget->document());
|
||||||
|
m_exportAutoFileName = fileName;
|
||||||
|
} else {
|
||||||
loadTTI(&file, m_textWidget->document());
|
loadTTI(&file, m_textWidget->document());
|
||||||
|
m_exportAutoFileName.clear();
|
||||||
|
}
|
||||||
|
|
||||||
levelSeen = m_textWidget->document()->levelRequired();
|
levelSeen = m_textWidget->document()->levelRequired();
|
||||||
m_levelRadioButton[levelSeen]->toggle();
|
m_levelRadioButton[levelSeen]->toggle();
|
||||||
m_textWidget->pageRender()->setRenderLevel(levelSeen);
|
m_textWidget->pageDecode()->setLevel(levelSeen);
|
||||||
|
if (levelSeen == 3)
|
||||||
|
m_paletteDockWidget->setLevel3p5Accepted(true);
|
||||||
updatePageWidgets();
|
updatePageWidgets();
|
||||||
|
|
||||||
QApplication::restoreOverrideCursor();
|
QApplication::restoreOverrideCursor();
|
||||||
@@ -857,6 +1038,18 @@ void MainWindow::updateRecentFileActions()
|
|||||||
m_recentFileActs[i]->setVisible(false);
|
m_recentFileActs[i]->setVisible(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainWindow::updateExportAutoAction()
|
||||||
|
{
|
||||||
|
if (m_exportAutoFileName.isEmpty()) {
|
||||||
|
m_exportAutoAct->setText(tr("Export subpage..."));
|
||||||
|
m_exportAutoAct->setEnabled(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_exportAutoAct->setText(tr("Overwrite &%1").arg(MainWindow::strippedName(m_exportAutoFileName)));
|
||||||
|
m_exportAutoAct->setEnabled(true);
|
||||||
|
}
|
||||||
|
|
||||||
void MainWindow::openRecentFile()
|
void MainWindow::openRecentFile()
|
||||||
{
|
{
|
||||||
if (const QAction *action = qobject_cast<const QAction *>(sender()))
|
if (const QAction *action = qobject_cast<const QAction *>(sender()))
|
||||||
@@ -872,13 +1065,13 @@ bool MainWindow::saveFile(const QString &fileName)
|
|||||||
if (file.open(QFile::WriteOnly | QFile::Text)) {
|
if (file.open(QFile::WriteOnly | QFile::Text)) {
|
||||||
saveTTI(file, *m_textWidget->document());
|
saveTTI(file, *m_textWidget->document());
|
||||||
if (!file.commit())
|
if (!file.commit())
|
||||||
errorMessage = tr("Cannot write file %1:\n%2.") .arg(QDir::toNativeSeparators(fileName), file.errorString());
|
errorMessage = tr("Cannot write file %1:\n%2.").arg(QDir::toNativeSeparators(fileName), file.errorString());
|
||||||
} else
|
} else
|
||||||
errorMessage = tr("Cannot open file %1 for writing:\n%2.").arg(QDir::toNativeSeparators(fileName), file.errorString());
|
errorMessage = tr("Cannot open file %1 for writing:\n%2.").arg(QDir::toNativeSeparators(fileName), file.errorString());
|
||||||
QApplication::restoreOverrideCursor();
|
QApplication::restoreOverrideCursor();
|
||||||
|
|
||||||
if (!errorMessage.isEmpty()) {
|
if (!errorMessage.isEmpty()) {
|
||||||
QMessageBox::warning(this, tr("QTeletextMaker"), errorMessage);
|
QMessageBox::warning(this, QApplication::applicationDisplayName(), errorMessage);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -887,6 +1080,103 @@ bool MainWindow::saveFile(const QString &fileName)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MainWindow::exportAuto()
|
||||||
|
{
|
||||||
|
// Menu should be disabled if m_exportAutoFileName is empty, but just in case...
|
||||||
|
if (m_exportAutoFileName.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
exportT42(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MainWindow::exportT42(bool fromAuto)
|
||||||
|
{
|
||||||
|
QString errorMessage;
|
||||||
|
QString exportFileName;
|
||||||
|
|
||||||
|
if (fromAuto)
|
||||||
|
exportFileName = m_exportAutoFileName;
|
||||||
|
else {
|
||||||
|
exportFileName = m_curFile;
|
||||||
|
changeSuffixFromTTI(exportFileName, "t42");
|
||||||
|
|
||||||
|
exportFileName = QFileDialog::getSaveFileName(this, tr("Export t42"), exportFileName, "t42 stream (*.t42)");
|
||||||
|
if (exportFileName.isEmpty())
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
||||||
|
QSaveFile file(exportFileName);
|
||||||
|
if (file.open(QFile::WriteOnly)) {
|
||||||
|
exportT42File(file, *m_textWidget->document());
|
||||||
|
if (!file.commit())
|
||||||
|
errorMessage = tr("Cannot write file %1:\n%2.").arg(QDir::toNativeSeparators(exportFileName), file.errorString());
|
||||||
|
} else
|
||||||
|
errorMessage = tr("Cannot open file %1 for writing:\n%2.").arg(QDir::toNativeSeparators(exportFileName), file.errorString());
|
||||||
|
QApplication::restoreOverrideCursor();
|
||||||
|
|
||||||
|
if (!errorMessage.isEmpty()) {
|
||||||
|
QMessageBox::warning(this, QApplication::applicationDisplayName(), errorMessage);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
MainWindow::prependToRecentFiles(exportFileName);
|
||||||
|
|
||||||
|
m_exportAutoFileName = exportFileName;
|
||||||
|
statusBar()->showMessage(tr("File exported"), 2000);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MainWindow::exportM29()
|
||||||
|
{
|
||||||
|
QString errorMessage;
|
||||||
|
QString exportFileName = m_curFile;
|
||||||
|
|
||||||
|
if (m_isUntitled || !QFileInfo(m_curFile).exists())
|
||||||
|
exportFileName = QString("P%1FF.tti").arg(m_textWidget->document()->pageNumber() >> 8, 1, 16);
|
||||||
|
else {
|
||||||
|
exportFileName = QFileInfo(m_curFile).fileName();
|
||||||
|
// Suggest a new filename to avoid clobbering the original file
|
||||||
|
if (QRegularExpression(("^[Pp]?[1-8][0-9A-Fa-f][0-9A-Fa-f]")).match(exportFileName).hasMatch()) {
|
||||||
|
// Page number forms start of file name, change it to xFF
|
||||||
|
if (exportFileName.at(0) == 'P' || exportFileName.at(0) == 'p') {
|
||||||
|
exportFileName[2] = 'F';
|
||||||
|
exportFileName[3] = 'F';
|
||||||
|
} else {
|
||||||
|
exportFileName[1] = 'F';
|
||||||
|
exportFileName[2] = 'F';
|
||||||
|
}
|
||||||
|
// No page number at start of file name. Try to insert "-m29" while preserving .tti(x) suffix
|
||||||
|
} else if (exportFileName.endsWith(".tti", Qt::CaseInsensitive)) {
|
||||||
|
exportFileName.chop(4);
|
||||||
|
exportFileName.append("-m29.tti");
|
||||||
|
} else if (exportFileName.endsWith(".ttix", Qt::CaseInsensitive)) {
|
||||||
|
exportFileName.chop(5);
|
||||||
|
exportFileName.append("-m29.ttix");
|
||||||
|
} else
|
||||||
|
// Shouldn't get here, bit of a messy escape but still better than clobbering the original file
|
||||||
|
exportFileName.append("-m29.tti");
|
||||||
|
|
||||||
|
exportFileName = QDir(QFileInfo(m_curFile).absoluteDir()).filePath(exportFileName);
|
||||||
|
}
|
||||||
|
|
||||||
|
exportFileName = QFileDialog::getSaveFileName(this, tr("Export M/29 tti"), exportFileName, "TTI teletext page (*.tti *.ttix)");
|
||||||
|
if (exportFileName.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
||||||
|
QSaveFile file(exportFileName);
|
||||||
|
if (file.open(QFile::WriteOnly | QFile::Text)) {
|
||||||
|
exportM29File(file, *m_textWidget->document());
|
||||||
|
if (!file.commit())
|
||||||
|
errorMessage = tr("Cannot write file %1:\n%2.").arg(QDir::toNativeSeparators(exportFileName), file.errorString());
|
||||||
|
} else
|
||||||
|
errorMessage = tr("Cannot open file %1 for writing:\n%2.").arg(QDir::toNativeSeparators(exportFileName), file.errorString());
|
||||||
|
QApplication::restoreOverrideCursor();
|
||||||
|
|
||||||
|
if (!errorMessage.isEmpty())
|
||||||
|
QMessageBox::warning(this, QApplication::applicationDisplayName(), errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
void MainWindow::setCurrentFile(const QString &fileName)
|
void MainWindow::setCurrentFile(const QString &fileName)
|
||||||
{
|
{
|
||||||
static int sequenceNumber = 1;
|
static int sequenceNumber = 1;
|
||||||
@@ -925,7 +1215,22 @@ MainWindow *MainWindow::findMainWindow(const QString &fileName) const
|
|||||||
|
|
||||||
void MainWindow::updateCursorPosition()
|
void MainWindow::updateCursorPosition()
|
||||||
{
|
{
|
||||||
m_cursorPositionLabel->setText(QString("Row %1 Column %2").arg(m_textWidget->document()->cursorRow()).arg(m_textWidget->document()->cursorColumn()));
|
QString result;
|
||||||
|
result.reserve(19);
|
||||||
|
|
||||||
|
if (m_textWidget->document()->cursorRow() == 0)
|
||||||
|
result = QString("Header, %1").arg(m_textWidget->document()->cursorColumn());
|
||||||
|
else if (m_textWidget->document()->cursorRow() == 24)
|
||||||
|
result = QString("FLOF, %1").arg(m_textWidget->document()->cursorColumn());
|
||||||
|
else
|
||||||
|
result = QString("%1, %2").arg(m_textWidget->document()->cursorRow()).arg(m_textWidget->document()->cursorColumn());
|
||||||
|
|
||||||
|
if (m_textWidget->document()->selectionActive())
|
||||||
|
result.append(QString(" (%1, %2)").arg(m_textWidget->document()->selectionHeight()).arg(m_textWidget->document()->selectionWidth()));
|
||||||
|
|
||||||
|
m_cursorPositionLabel->setText(result);
|
||||||
|
m_textScene->updateCursor();
|
||||||
|
m_textView->ensureVisible(m_textScene->cursorRectItem(), 16, 24);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::updatePageWidgets()
|
void MainWindow::updatePageWidgets()
|
||||||
@@ -933,9 +1238,11 @@ void MainWindow::updatePageWidgets()
|
|||||||
m_subPageLabel->setText(QString("%1/%2").arg(m_textWidget->document()->currentSubPageIndex()+1).arg(m_textWidget->document()->numberOfSubPages()));
|
m_subPageLabel->setText(QString("%1/%2").arg(m_textWidget->document()->currentSubPageIndex()+1).arg(m_textWidget->document()->numberOfSubPages()));
|
||||||
m_previousSubPageButton->setEnabled(!(m_textWidget->document()->currentSubPageIndex() == 0));
|
m_previousSubPageButton->setEnabled(!(m_textWidget->document()->currentSubPageIndex() == 0));
|
||||||
m_nextSubPageButton->setEnabled(!(m_textWidget->document()->currentSubPageIndex() == (m_textWidget->document()->numberOfSubPages()) - 1));
|
m_nextSubPageButton->setEnabled(!(m_textWidget->document()->currentSubPageIndex() == (m_textWidget->document()->numberOfSubPages()) - 1));
|
||||||
|
updateCursorPosition();
|
||||||
m_deleteSubPageAction->setEnabled(m_textWidget->document()->numberOfSubPages() > 1);
|
m_deleteSubPageAction->setEnabled(m_textWidget->document()->numberOfSubPages() > 1);
|
||||||
m_pageOptionsDockWidget->updateWidgets();
|
m_pageOptionsDockWidget->updateWidgets();
|
||||||
m_pageEnhancementsDockWidget->updateWidgets();
|
m_pageEnhancementsDockWidget->updateWidgets();
|
||||||
m_x26DockWidget->loadX26List();
|
m_x26DockWidget->loadX26List();
|
||||||
m_paletteDockWidget->updateAllColourButtons();
|
m_paletteDockWidget->updateAllColourButtons();
|
||||||
|
m_pageComposeLinksDockWidget->updateWidgets();
|
||||||
}
|
}
|
||||||
|
|||||||
25
mainwindow.h
25
mainwindow.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -31,6 +31,7 @@
|
|||||||
#include <QToolButton>
|
#include <QToolButton>
|
||||||
|
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
|
#include "pagecomposelinksdockwidget.h"
|
||||||
#include "pageenhancementsdockwidget.h"
|
#include "pageenhancementsdockwidget.h"
|
||||||
#include "pageoptionsdockwidget.h"
|
#include "pageoptionsdockwidget.h"
|
||||||
#include "palettedockwidget.h"
|
#include "palettedockwidget.h"
|
||||||
@@ -58,22 +59,29 @@ private slots:
|
|||||||
void open();
|
void open();
|
||||||
bool save();
|
bool save();
|
||||||
bool saveAs();
|
bool saveAs();
|
||||||
|
void reload();
|
||||||
|
void exportAuto();
|
||||||
|
void exportT42(bool fromAuto);
|
||||||
void exportZXNet();
|
void exportZXNet();
|
||||||
void exportEditTF();
|
void exportEditTF();
|
||||||
|
void exportPNG();
|
||||||
|
void exportM29();
|
||||||
void updateRecentFileActions();
|
void updateRecentFileActions();
|
||||||
|
void updateExportAutoAction();
|
||||||
void openRecentFile();
|
void openRecentFile();
|
||||||
void about();
|
void about();
|
||||||
void updatePageWidgets();
|
void updatePageWidgets();
|
||||||
void updateCursorPosition();
|
void updateCursorPosition();
|
||||||
|
|
||||||
void insertRow(bool);
|
void insertRow(bool copyRow);
|
||||||
void deleteRow();
|
void deleteRow();
|
||||||
void insertSubPage(bool, bool);
|
void insertSubPage(bool afterCurrentSubPage, bool copyCurrentSubPage);
|
||||||
void deleteSubPage();
|
void deleteSubPage();
|
||||||
|
|
||||||
void setSceneDimensions();
|
void setSceneDimensions();
|
||||||
void setBorder(int);
|
void setBorder(int newViewBorder);
|
||||||
void setAspectRatio(int);
|
void setAspectRatio(int newViewAspectRatio);
|
||||||
|
void setSmoothTransform(bool smoothTransform);
|
||||||
void zoomIn();
|
void zoomIn();
|
||||||
void zoomOut();
|
void zoomOut();
|
||||||
void zoomReset();
|
void zoomReset();
|
||||||
@@ -82,6 +90,7 @@ private slots:
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
enum { m_MaxRecentFiles = 10 };
|
enum { m_MaxRecentFiles = 10 };
|
||||||
|
const float aspectRatioHorizontalScaling[4] = { 0.6, 0.6, 0.8, 0.5 };
|
||||||
|
|
||||||
void init();
|
void init();
|
||||||
void createActions();
|
void createActions();
|
||||||
@@ -104,24 +113,28 @@ private:
|
|||||||
QGraphicsView *m_textView;
|
QGraphicsView *m_textView;
|
||||||
|
|
||||||
int m_viewBorder, m_viewAspectRatio, m_viewZoom;
|
int m_viewBorder, m_viewAspectRatio, m_viewZoom;
|
||||||
|
bool m_viewSmoothTransform;
|
||||||
PageOptionsDockWidget *m_pageOptionsDockWidget;
|
PageOptionsDockWidget *m_pageOptionsDockWidget;
|
||||||
PageEnhancementsDockWidget *m_pageEnhancementsDockWidget;
|
PageEnhancementsDockWidget *m_pageEnhancementsDockWidget;
|
||||||
X26DockWidget *m_x26DockWidget;
|
X26DockWidget *m_x26DockWidget;
|
||||||
PaletteDockWidget *m_paletteDockWidget;
|
PaletteDockWidget *m_paletteDockWidget;
|
||||||
|
PageComposeLinksDockWidget *m_pageComposeLinksDockWidget;
|
||||||
|
|
||||||
QAction *m_recentFileActs[m_MaxRecentFiles];
|
QAction *m_recentFileActs[m_MaxRecentFiles];
|
||||||
QAction *m_recentFileSeparator;
|
QAction *m_recentFileSeparator;
|
||||||
QAction *m_recentFileSubMenuAct;
|
QAction *m_recentFileSubMenuAct;
|
||||||
|
QAction *m_exportAutoAct;
|
||||||
QAction *m_deleteSubPageAction;
|
QAction *m_deleteSubPageAction;
|
||||||
QAction *m_borderActs[3];
|
QAction *m_borderActs[3];
|
||||||
QAction *m_aspectRatioActs[4];
|
QAction *m_aspectRatioActs[4];
|
||||||
|
QAction *m_smoothTransformAction;
|
||||||
|
|
||||||
QLabel *m_subPageLabel, *m_cursorPositionLabel;
|
QLabel *m_subPageLabel, *m_cursorPositionLabel;
|
||||||
QToolButton *m_previousSubPageButton, *m_nextSubPageButton;
|
QToolButton *m_previousSubPageButton, *m_nextSubPageButton;
|
||||||
QPushButton *m_insertModePushButton;
|
QPushButton *m_insertModePushButton;
|
||||||
QRadioButton *m_levelRadioButton[4];
|
QRadioButton *m_levelRadioButton[4];
|
||||||
|
|
||||||
QString m_curFile;
|
QString m_curFile, m_exportAutoFileName;
|
||||||
bool m_isUntitled;
|
bool m_isUntitled;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
|
|||||||
16
pagebase.h
16
pagebase.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -35,17 +35,17 @@ public:
|
|||||||
|
|
||||||
virtual bool isEmpty() const;
|
virtual bool isEmpty() const;
|
||||||
|
|
||||||
virtual QByteArray packet(int) const;
|
virtual QByteArray packet(int i) const;
|
||||||
virtual QByteArray packet(int, int) const;
|
virtual QByteArray packet(int i, int j) const;
|
||||||
virtual bool packetNeeded(int i) const { return m_displayPackets[i] != nullptr; }
|
virtual bool packetExists(int i) const { return m_displayPackets[i] != nullptr; }
|
||||||
virtual bool packetNeeded(int i, int j) const { return m_designationPackets[i-26][j] != nullptr; }
|
virtual bool packetExists(int i, int j) const { return m_designationPackets[i-26][j] != nullptr; }
|
||||||
virtual bool setPacket(int, QByteArray);
|
virtual bool setPacket(int i, QByteArray packetContents);
|
||||||
virtual bool setPacket(int, int, QByteArray);
|
virtual bool setPacket(int i, int j, QByteArray packetContents);
|
||||||
// bool deletePacket(int);
|
// bool deletePacket(int);
|
||||||
// bool deletePacket(int, int);
|
// bool deletePacket(int, int);
|
||||||
|
|
||||||
virtual bool controlBit(int bitNumber) const { return m_controlBits[bitNumber]; }
|
virtual bool controlBit(int bitNumber) const { return m_controlBits[bitNumber]; }
|
||||||
virtual bool setControlBit(int, bool);
|
virtual bool setControlBit(int bitNumber, bool active);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool m_controlBits[11];
|
bool m_controlBits[11];
|
||||||
|
|||||||
222
pagecomposelinksdockwidget.cpp
Normal file
222
pagecomposelinksdockwidget.cpp
Normal file
@@ -0,0 +1,222 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
|
*
|
||||||
|
* This file is part of QTeletextMaker.
|
||||||
|
*
|
||||||
|
* QTeletextMaker 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.
|
||||||
|
*
|
||||||
|
* QTeletextMaker 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 QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <QCheckBox>
|
||||||
|
#include <QComboBox>
|
||||||
|
#include <QGridLayout>
|
||||||
|
#include <QLabel>
|
||||||
|
#include <QLineEdit>
|
||||||
|
#include <QMap>
|
||||||
|
#include <QPair>
|
||||||
|
#include <QRegularExpressionValidator>
|
||||||
|
#include <QString>
|
||||||
|
|
||||||
|
#include "pagecomposelinksdockwidget.h"
|
||||||
|
|
||||||
|
PageComposeLinksDockWidget::PageComposeLinksDockWidget(TeletextWidget *parent): QDockWidget(parent)
|
||||||
|
{
|
||||||
|
QVBoxLayout *pageComposeLinksLayout = new QVBoxLayout;
|
||||||
|
QWidget *pageComposeLinksWidget = new QWidget;
|
||||||
|
|
||||||
|
m_parentMainWidget = parent;
|
||||||
|
|
||||||
|
this->setObjectName("PageComposeLinksDockWidget");
|
||||||
|
this->setWindowTitle("Compositional links");
|
||||||
|
|
||||||
|
QGridLayout *x27Layout = new QGridLayout;
|
||||||
|
|
||||||
|
QLabel *levelAllLabel = new QLabel(tr("Levels 2.5 and 3.5"));
|
||||||
|
levelAllLabel->setAlignment(Qt::AlignCenter);
|
||||||
|
x27Layout->addWidget(levelAllLabel, 0, 0, 1, 5);
|
||||||
|
|
||||||
|
// Link functions
|
||||||
|
x27Layout->addWidget(new QLabel(tr("GPOP")), 2, 0, 1, 1);
|
||||||
|
x27Layout->addWidget(new QLabel(tr("POP")), 3, 0, 1, 1);
|
||||||
|
x27Layout->addWidget(new QLabel(tr("GDRCS")), 4, 0, 1, 1);
|
||||||
|
x27Layout->addWidget(new QLabel(tr("DRCS")), 5, 0, 1, 1);
|
||||||
|
|
||||||
|
// Labels across the top
|
||||||
|
x27Layout->addWidget(new QLabel(tr("L2.5")), 1, 1, 1, 1);
|
||||||
|
x27Layout->addWidget(new QLabel(tr("L3.5")), 1, 2, 1, 1);
|
||||||
|
x27Layout->addWidget(new QLabel(tr("Page no")), 1, 3, 1, 1);
|
||||||
|
x27Layout->addWidget(new QLabel(tr("Sub pages required")), 1, 4, 1, 1);
|
||||||
|
|
||||||
|
QLabel *level3p5OnlyLabel = new QLabel(tr("Level 3.5 only"));
|
||||||
|
level3p5OnlyLabel->setAlignment(Qt::AlignCenter);
|
||||||
|
x27Layout->addWidget(level3p5OnlyLabel, 6, 0, 1, 5);
|
||||||
|
|
||||||
|
m_pageNumberValidator = new QRegularExpressionValidator(QRegularExpression("[1-8][0-9A-Fa-f][0-9A-Fa-f]"), this);
|
||||||
|
|
||||||
|
for (int i=0; i<8; i++) {
|
||||||
|
if (i < 4) {
|
||||||
|
// Required at which Levels
|
||||||
|
m_composeLinkLevelCheckbox[i][0] = new QCheckBox(this);
|
||||||
|
x27Layout->addWidget(m_composeLinkLevelCheckbox[i][0], i+2, 1, 1, 1);
|
||||||
|
connect(m_composeLinkLevelCheckbox[i][0], &QCheckBox::stateChanged, [=](bool active) { m_parentMainWidget->document()->currentSubPage()->setComposeLinkLevel2p5(i, active); });
|
||||||
|
m_composeLinkLevelCheckbox[i][1] = new QCheckBox(this);
|
||||||
|
x27Layout->addWidget(m_composeLinkLevelCheckbox[i][1], i+2, 2, 1, 1);
|
||||||
|
connect(m_composeLinkLevelCheckbox[i][1], &QCheckBox::stateChanged, [=](bool active) { m_parentMainWidget->document()->currentSubPage()->setComposeLinkLevel3p5(i, active); });
|
||||||
|
} else {
|
||||||
|
// Selectable link functions for Level 3.5
|
||||||
|
m_composeLinkFunctionComboBox[i-4] = new QComboBox;
|
||||||
|
m_composeLinkFunctionComboBox[i-4]->addItem("GPOP");
|
||||||
|
m_composeLinkFunctionComboBox[i-4]->addItem("POP");
|
||||||
|
m_composeLinkFunctionComboBox[i-4]->addItem("GDRCS");
|
||||||
|
m_composeLinkFunctionComboBox[i-4]->addItem("DRCS");
|
||||||
|
x27Layout->addWidget(m_composeLinkFunctionComboBox[i-4], i+3, 0, 1, 1, Qt::AlignTop);
|
||||||
|
connect(m_composeLinkFunctionComboBox[i-4], QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index) { m_parentMainWidget->document()->currentSubPage()->setComposeLinkFunction(i, index); } );
|
||||||
|
}
|
||||||
|
// Page link
|
||||||
|
m_composeLinkPageNumberLineEdit[i] = new QLineEdit("100");
|
||||||
|
m_composeLinkPageNumberLineEdit[i]->setMaxLength(3);
|
||||||
|
m_composeLinkPageNumberLineEdit[i]->setInputMask(">DHH");
|
||||||
|
m_composeLinkPageNumberLineEdit[i]->setValidator(m_pageNumberValidator);
|
||||||
|
x27Layout->addWidget(m_composeLinkPageNumberLineEdit[i], i+(i<4 ? 2 : 3), 3, 1, 1);
|
||||||
|
connect(m_composeLinkPageNumberLineEdit[i], &QLineEdit::textEdited, [=](QString value) { setComposeLinkPageNumber(i, value); } );
|
||||||
|
|
||||||
|
// Sub page numbers required
|
||||||
|
m_composeLinkSubPageNumbersLineEdit[i] = new QLineEdit(this);
|
||||||
|
x27Layout->addWidget(m_composeLinkSubPageNumbersLineEdit[i], i+(i<4 ? 2 : 3), 4, 1, 1);
|
||||||
|
connect(m_composeLinkSubPageNumbersLineEdit[i], &QLineEdit::textEdited, [=](QString value) { setComposeLinkSubPageNumbers(i, value); } );
|
||||||
|
}
|
||||||
|
|
||||||
|
pageComposeLinksLayout->addLayout(x27Layout);
|
||||||
|
|
||||||
|
pageComposeLinksLayout->addStretch(1);
|
||||||
|
|
||||||
|
pageComposeLinksWidget->setLayout(pageComposeLinksLayout);
|
||||||
|
this->setWidget(pageComposeLinksWidget);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PageComposeLinksDockWidget::setComposeLinkPageNumber(int linkNumber, const QString &newPageNumberString)
|
||||||
|
{
|
||||||
|
// The LineEdit should check if a valid hex number was entered, but just in case...
|
||||||
|
bool newPageNumberOk;
|
||||||
|
int newPageNumberRead = newPageNumberString.toInt(&newPageNumberOk, 16);
|
||||||
|
if ((!newPageNumberOk) || newPageNumberRead < 0x100 || newPageNumberRead > 0x8ff)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Stored as page link with relative magazine number, convert from absolute page number that was entered
|
||||||
|
newPageNumberRead ^= (m_parentMainWidget->document()->pageNumber() & 0x700);
|
||||||
|
m_parentMainWidget->document()->currentSubPage()->setComposeLinkPageNumber(linkNumber, newPageNumberRead);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PageComposeLinksDockWidget::setComposeLinkSubPageNumbers(int linkNumber, const QString &newSubPagesString)
|
||||||
|
{
|
||||||
|
int newSubPageCodes = 0x0000;
|
||||||
|
|
||||||
|
if (!newSubPagesString.isEmpty()) {
|
||||||
|
// Turn user-entered comma separated subpages and subpage-ranges into bits
|
||||||
|
// First we do the "comma separated"
|
||||||
|
const QStringList items = newSubPagesString.split(QLatin1Char(','), Qt::SkipEmptyParts);
|
||||||
|
// Now see if there's valid single numbers or ranges between the commas
|
||||||
|
for (const QString &item : items) {
|
||||||
|
if (item.isEmpty())
|
||||||
|
return;
|
||||||
|
if (item.contains(QLatin1Char('-'))) {
|
||||||
|
// Dash found, should be a range of subpages: check for two things either side of dash
|
||||||
|
const QStringList rangeItems = item.split('-');
|
||||||
|
if (rangeItems.count() != 2)
|
||||||
|
return;
|
||||||
|
// Now check if the two things are valid numbers 0-15, with first number less than second
|
||||||
|
bool ok1, ok2;
|
||||||
|
const int number1 = rangeItems[0].toInt(&ok1);
|
||||||
|
const int number2 = rangeItems[1].toInt(&ok2);
|
||||||
|
if (!ok1 || !ok2 || number1 < 0 || number2 < 0 || number1 > 15 || number2 > 15 || number2 < number1)
|
||||||
|
return;
|
||||||
|
// Yes, it is a valid range. Apply it to the result
|
||||||
|
for (int i=number1; i<=number2; i++)
|
||||||
|
newSubPageCodes |= (1 << i);
|
||||||
|
} else {
|
||||||
|
// A single thing (maybe extracted between commas): check if the thing is a number 0-15
|
||||||
|
bool ok;
|
||||||
|
const int number = item.toInt(&ok);
|
||||||
|
if (!ok || number < 0 || number > 15)
|
||||||
|
return;
|
||||||
|
// Yes, it is a valid number. Apply it to the result
|
||||||
|
newSubPageCodes |= (1 << number);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m_parentMainWidget->document()->currentSubPage()->setComposeLinkSubPageCodes(linkNumber, newSubPageCodes);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PageComposeLinksDockWidget::updateWidgets()
|
||||||
|
{
|
||||||
|
for (int i=0; i<8; i++) {
|
||||||
|
if (i >= 4) {
|
||||||
|
m_composeLinkFunctionComboBox[i-4]->blockSignals(true);
|
||||||
|
m_composeLinkFunctionComboBox[i-4]->setCurrentIndex(m_parentMainWidget->document()->currentSubPage()->composeLinkFunction(i));
|
||||||
|
m_composeLinkFunctionComboBox[i-4]->blockSignals(false);
|
||||||
|
} else {
|
||||||
|
m_composeLinkLevelCheckbox[i][0]->blockSignals(true);
|
||||||
|
m_composeLinkLevelCheckbox[i][0]->setChecked(m_parentMainWidget->document()->currentSubPage()->composeLinkLevel2p5(i));
|
||||||
|
m_composeLinkLevelCheckbox[i][0]->blockSignals(false);
|
||||||
|
m_composeLinkLevelCheckbox[i][1]->blockSignals(true);
|
||||||
|
m_composeLinkLevelCheckbox[i][1]->setChecked(m_parentMainWidget->document()->currentSubPage()->composeLinkLevel3p5(i));
|
||||||
|
m_composeLinkLevelCheckbox[i][1]->blockSignals(false);
|
||||||
|
}
|
||||||
|
// Stored as page link with relative magazine number, convert to absolute page number for display
|
||||||
|
int absoluteLinkPageNumber = m_parentMainWidget->document()->currentSubPage()->composeLinkPageNumber(i) ^ (m_parentMainWidget->document()->pageNumber() & 0x700);
|
||||||
|
// Fix magazine 0 to 8
|
||||||
|
if ((absoluteLinkPageNumber & 0x700) == 0x000)
|
||||||
|
absoluteLinkPageNumber |= 0x800;
|
||||||
|
m_composeLinkPageNumberLineEdit[i]->blockSignals(true);
|
||||||
|
m_composeLinkPageNumberLineEdit[i]->setText(QString::number(absoluteLinkPageNumber, 16).toUpper());
|
||||||
|
m_composeLinkPageNumberLineEdit[i]->blockSignals(false);
|
||||||
|
|
||||||
|
// Turn subpage bits into user-friendly comma separated numbers and number-ranges
|
||||||
|
QString rangeString;
|
||||||
|
|
||||||
|
if (m_parentMainWidget->document()->currentSubPage()->composeLinkSubPageCodes(i) != 0x0000) {
|
||||||
|
// First build a list of consecutive ranges seen
|
||||||
|
// The "b-index" is based on https://codereview.stackexchange.com/a/90074
|
||||||
|
QMap<int, QPair<int, int>> ranges;
|
||||||
|
int index = 0;
|
||||||
|
for (int b=0; b<16; b++)
|
||||||
|
if ((m_parentMainWidget->document()->currentSubPage()->composeLinkSubPageCodes(i) >> b) & 0x01) {
|
||||||
|
if (!ranges.contains(b-index))
|
||||||
|
ranges.insert(b-index, qMakePair(b, b));
|
||||||
|
else
|
||||||
|
ranges[b-index].second = b;
|
||||||
|
index++;
|
||||||
|
}
|
||||||
|
// Now go through the list and add single numbers or ranges as appropriate
|
||||||
|
QPair<int, int> range;
|
||||||
|
foreach (range, ranges) {
|
||||||
|
// For second and subsequent entries only, append a comma first
|
||||||
|
if (!rangeString.isEmpty())
|
||||||
|
rangeString.append(',');
|
||||||
|
// Append the single number or the first number of the range
|
||||||
|
rangeString.append(QString("%1").arg(range.first));
|
||||||
|
// If that was part of a range and not a single orphaned number
|
||||||
|
if (range.first != range.second) {
|
||||||
|
// Ranges of 3 or more use a dash. A range of 2 can make do with a comma
|
||||||
|
rangeString.append((range.first+1 == range.second) ? ',' : '-');
|
||||||
|
// Append the second number of the range
|
||||||
|
rangeString.append(QString("%1").arg(range.second));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m_composeLinkSubPageNumbersLineEdit[i]->blockSignals(true);
|
||||||
|
m_composeLinkSubPageNumbersLineEdit[i]->setText(rangeString);
|
||||||
|
m_composeLinkSubPageNumbersLineEdit[i]->blockSignals(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
52
pagecomposelinksdockwidget.h
Normal file
52
pagecomposelinksdockwidget.h
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
|
*
|
||||||
|
* This file is part of QTeletextMaker.
|
||||||
|
*
|
||||||
|
* QTeletextMaker 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.
|
||||||
|
*
|
||||||
|
* QTeletextMaker 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 QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PAGECOMPOSELINKSDOCKWIDGET_H
|
||||||
|
#define PAGECOMPOSELINKSDOCKWIDGET_H
|
||||||
|
|
||||||
|
#include <QCheckBox>
|
||||||
|
#include <QComboBox>
|
||||||
|
#include <QDockWidget>
|
||||||
|
#include <QLineEdit>
|
||||||
|
#include <QRegularExpressionValidator>
|
||||||
|
#include <QString>
|
||||||
|
|
||||||
|
#include "mainwidget.h"
|
||||||
|
|
||||||
|
class PageComposeLinksDockWidget : public QDockWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
PageComposeLinksDockWidget(TeletextWidget *parent);
|
||||||
|
void updateWidgets();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void setComposeLinkPageNumber(int linkNumber, const QString &newPageNumberString);
|
||||||
|
void setComposeLinkSubPageNumbers(int linkNumber, const QString &newSubPagesString);
|
||||||
|
|
||||||
|
TeletextWidget *m_parentMainWidget;
|
||||||
|
QCheckBox *m_composeLinkLevelCheckbox[4][2]; // For links 0-3
|
||||||
|
QComboBox *m_composeLinkFunctionComboBox[4]; // For links 4-7; remember to subtract 4!
|
||||||
|
QLineEdit *m_composeLinkPageNumberLineEdit[8], *m_composeLinkSubPageNumbersLineEdit[8];
|
||||||
|
|
||||||
|
QRegularExpressionValidator *m_pageNumberValidator;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -22,14 +22,12 @@
|
|||||||
#include <QGridLayout>
|
#include <QGridLayout>
|
||||||
#include <QGroupBox>
|
#include <QGroupBox>
|
||||||
#include <QLabel>
|
#include <QLabel>
|
||||||
#include <QLineEdit>
|
|
||||||
#include <QMap>
|
|
||||||
#include <QPair>
|
|
||||||
#include <QSpinBox>
|
#include <QSpinBox>
|
||||||
#include <QString>
|
|
||||||
|
|
||||||
#include "pageenhancementsdockwidget.h"
|
#include "pageenhancementsdockwidget.h"
|
||||||
|
|
||||||
|
#include "levelonecommands.h"
|
||||||
|
|
||||||
PageEnhancementsDockWidget::PageEnhancementsDockWidget(TeletextWidget *parent): QDockWidget(parent)
|
PageEnhancementsDockWidget::PageEnhancementsDockWidget(TeletextWidget *parent): QDockWidget(parent)
|
||||||
{
|
{
|
||||||
QVBoxLayout *pageEnhancementsLayout = new QVBoxLayout;
|
QVBoxLayout *pageEnhancementsLayout = new QVBoxLayout;
|
||||||
@@ -38,28 +36,26 @@ PageEnhancementsDockWidget::PageEnhancementsDockWidget(TeletextWidget *parent):
|
|||||||
m_parentMainWidget = parent;
|
m_parentMainWidget = parent;
|
||||||
|
|
||||||
this->setObjectName("PageEnhancementsDockWidget");
|
this->setObjectName("PageEnhancementsDockWidget");
|
||||||
this->setWindowTitle("Page enhancements");
|
this->setWindowTitle("X/28 page enhancements");
|
||||||
|
|
||||||
QGroupBox *x28GroupBox = new QGroupBox(tr("X/28 enhancements"));
|
// Colour group box and layout
|
||||||
QGridLayout *x28Layout = new QGridLayout;
|
QGroupBox *colourGroupBox = new QGroupBox(tr("Colours"));
|
||||||
|
QGridLayout *colourLayout = new QGridLayout;
|
||||||
|
|
||||||
// Default screen and default row colours
|
// Default screen and default row colours
|
||||||
x28Layout->addWidget(new QLabel(tr("Default screen colour")), 0, 0, 1, 1);
|
colourLayout->addWidget(new QLabel(tr("Default screen colour")), 0, 0, 1, 1);
|
||||||
x28Layout->addWidget(new QLabel(tr("Default row colour")), 1, 0, 1, 1);
|
colourLayout->addWidget(new QLabel(tr("Default row colour")), 1, 0, 1, 1);
|
||||||
m_defaultScreenColourCombo = new QComboBox;
|
m_defaultScreenColourCombo = new QComboBox;
|
||||||
|
m_defaultScreenColourCombo->setModel(m_parentMainWidget->document()->clutModel());
|
||||||
m_defaultRowColourCombo = new QComboBox;
|
m_defaultRowColourCombo = new QComboBox;
|
||||||
for (int r=0; r<=3; r++)
|
m_defaultRowColourCombo->setModel(m_parentMainWidget->document()->clutModel());
|
||||||
for (int c=0; c<=7; c++) {
|
colourLayout->addWidget(m_defaultScreenColourCombo, 0, 1, 1, 1, Qt::AlignTop);
|
||||||
m_defaultScreenColourCombo->addItem(tr("CLUT %1:%2").arg(r).arg(c));
|
connect(m_defaultScreenColourCombo, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index){ m_parentMainWidget->document()->undoStack()->push(new SetFullScreenColourCommand(m_parentMainWidget->document(), index)); });
|
||||||
m_defaultRowColourCombo->addItem(tr("CLUT %1:%2").arg(r).arg(c));
|
colourLayout->addWidget(m_defaultRowColourCombo, 1, 1, 1, 1, Qt::AlignTop);
|
||||||
}
|
connect(m_defaultRowColourCombo, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index){ m_parentMainWidget->document()->undoStack()->push(new SetFullRowColourCommand(m_parentMainWidget->document(), index)); });
|
||||||
x28Layout->addWidget(m_defaultScreenColourCombo, 0, 1, 1, 1, Qt::AlignTop);
|
|
||||||
connect(m_defaultScreenColourCombo, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index){ m_parentMainWidget->setDefaultScreenColour(index); });
|
|
||||||
x28Layout->addWidget(m_defaultRowColourCombo, 1, 1, 1, 1, Qt::AlignTop);
|
|
||||||
connect(m_defaultRowColourCombo, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index){ m_parentMainWidget->setDefaultRowColour(index); });
|
|
||||||
|
|
||||||
// CLUT remapping
|
// CLUT remapping
|
||||||
x28Layout->addWidget(new QLabel(tr("CLUT remapping")), 2, 0, 1, 1);
|
colourLayout->addWidget(new QLabel(tr("CLUT remapping")), 2, 0, 1, 1);
|
||||||
m_colourTableCombo = new QComboBox;
|
m_colourTableCombo = new QComboBox;
|
||||||
m_colourTableCombo->addItem("Fore 0 Back 0");
|
m_colourTableCombo->addItem("Fore 0 Back 0");
|
||||||
m_colourTableCombo->addItem("Fore 0 Back 1");
|
m_colourTableCombo->addItem("Fore 0 Back 1");
|
||||||
@@ -69,89 +65,44 @@ PageEnhancementsDockWidget::PageEnhancementsDockWidget(TeletextWidget *parent):
|
|||||||
m_colourTableCombo->addItem("Fore 2 Back 1");
|
m_colourTableCombo->addItem("Fore 2 Back 1");
|
||||||
m_colourTableCombo->addItem("Fore 2 Back 2");
|
m_colourTableCombo->addItem("Fore 2 Back 2");
|
||||||
m_colourTableCombo->addItem("Fore 2 Back 3");
|
m_colourTableCombo->addItem("Fore 2 Back 3");
|
||||||
x28Layout->addWidget(m_colourTableCombo, 2, 1, 1, 1, Qt::AlignTop);
|
colourLayout->addWidget(m_colourTableCombo, 2, 1, 1, 1, Qt::AlignTop);
|
||||||
connect(m_colourTableCombo, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index){ m_parentMainWidget->setColourTableRemap(index); });
|
connect(m_colourTableCombo, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index){ m_parentMainWidget->document()->undoStack()->push(new SetCLUTRemapCommand(m_parentMainWidget->document(), index)); });
|
||||||
|
|
||||||
// Black background colour substitution
|
// Black background colour substitution
|
||||||
m_blackBackgroundSubstAct = new QCheckBox("Black background colour substitution");
|
m_blackBackgroundSubstAct = new QCheckBox("Black background colour substitution");
|
||||||
x28Layout->addWidget(m_blackBackgroundSubstAct, 3, 0, 1, 2, Qt::AlignTop);
|
colourLayout->addWidget(m_blackBackgroundSubstAct, 3, 0, 1, 2, Qt::AlignTop);
|
||||||
connect(m_blackBackgroundSubstAct, &QCheckBox::stateChanged, m_parentMainWidget, &TeletextWidget::setBlackBackgroundSubst);
|
connect(m_blackBackgroundSubstAct, &QCheckBox::stateChanged, [=](int state){ m_parentMainWidget->document()->undoStack()->push(new SetBlackBackgroundSubstCommand(m_parentMainWidget->document(), state)); });
|
||||||
|
|
||||||
|
// Add group box to the main layout
|
||||||
|
colourGroupBox->setLayout(colourLayout);
|
||||||
|
pageEnhancementsLayout->addWidget(colourGroupBox);
|
||||||
|
|
||||||
|
|
||||||
|
// Side panels group box and layout
|
||||||
|
QGroupBox *sidePanelsGroupBox = new QGroupBox(tr("Side panels"));
|
||||||
|
QGridLayout *sidePanelsLayout = new QGridLayout;
|
||||||
|
|
||||||
// Side panel columns
|
// Side panel columns
|
||||||
x28Layout->addWidget(new QLabel(tr("Left side panel columns")), 4, 0, 1, 1);
|
sidePanelsLayout->addWidget(new QLabel(tr("Left side panel columns")), 0, 0, 1, 1);
|
||||||
m_leftSidePanelSpinBox = new QSpinBox(this);
|
m_leftSidePanelSpinBox = new QSpinBox(this);
|
||||||
m_leftSidePanelSpinBox->setMaximum(16);
|
m_leftSidePanelSpinBox->setMaximum(16);
|
||||||
x28Layout->addWidget(m_leftSidePanelSpinBox, 4, 1, 1, 1, Qt::AlignTop);
|
sidePanelsLayout->addWidget(m_leftSidePanelSpinBox, 0, 1, 1, 1, Qt::AlignTop);
|
||||||
connect(m_leftSidePanelSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), [=](int index){ setLeftSidePanelWidth(index); });
|
connect(m_leftSidePanelSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), [=](int index){ setLeftSidePanelWidth(index); });
|
||||||
|
|
||||||
x28Layout->addWidget(new QLabel(tr("Right side panel columns")), 5, 0, 1, 1);
|
sidePanelsLayout->addWidget(new QLabel(tr("Right side panel columns")), 1, 0, 1, 1);
|
||||||
m_rightSidePanelSpinBox = new QSpinBox(this);
|
m_rightSidePanelSpinBox = new QSpinBox(this);
|
||||||
m_rightSidePanelSpinBox->setMaximum(16);
|
m_rightSidePanelSpinBox->setMaximum(16);
|
||||||
x28Layout->addWidget(m_rightSidePanelSpinBox, 5, 1, 1, 1, Qt::AlignTop);
|
sidePanelsLayout->addWidget(m_rightSidePanelSpinBox, 1, 1, 1, 1, Qt::AlignTop);
|
||||||
connect(m_rightSidePanelSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), [=](int index){ setRightSidePanelWidth(index); });
|
connect(m_rightSidePanelSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), [=](int index){ setRightSidePanelWidth(index); });
|
||||||
|
|
||||||
// Side panels status
|
// Side panels status
|
||||||
m_sidePanelStatusAct = new QCheckBox("Side panels at level 3.5 only");
|
m_sidePanelStatusAct = new QCheckBox("Side panels at level 3.5 only");
|
||||||
x28Layout->addWidget(m_sidePanelStatusAct, 6, 0, 1, 2, Qt::AlignTop);
|
sidePanelsLayout->addWidget(m_sidePanelStatusAct, 2, 0, 1, 2, Qt::AlignTop);
|
||||||
connect(m_sidePanelStatusAct, &QCheckBox::stateChanged, m_parentMainWidget, &TeletextWidget::setSidePanelAtL35Only);
|
connect(m_sidePanelStatusAct, &QCheckBox::stateChanged, m_parentMainWidget, &TeletextWidget::setSidePanelAtL35Only);
|
||||||
|
|
||||||
x28GroupBox->setLayout(x28Layout);
|
// Add group box to the main layout
|
||||||
pageEnhancementsLayout->addWidget(x28GroupBox);
|
sidePanelsGroupBox->setLayout(sidePanelsLayout);
|
||||||
|
pageEnhancementsLayout->addWidget(sidePanelsGroupBox);
|
||||||
QGroupBox *x27GroupBox = new QGroupBox(tr("X/27 links - Level 2.5 and 3.5"));
|
|
||||||
QGridLayout *x27Layout = new QGridLayout;
|
|
||||||
|
|
||||||
// Link functions
|
|
||||||
x27Layout->addWidget(new QLabel(tr("GPOP")), 1, 0, 1, 1);
|
|
||||||
x27Layout->addWidget(new QLabel(tr("POP")), 2, 0, 1, 1);
|
|
||||||
x27Layout->addWidget(new QLabel(tr("GDRCS")), 3, 0, 1, 1);
|
|
||||||
x27Layout->addWidget(new QLabel(tr("DRCS")), 4, 0, 1, 1);
|
|
||||||
|
|
||||||
// Labels across the top
|
|
||||||
x27Layout->addWidget(new QLabel(tr("L2.5")), 0, 1, 1, 1);
|
|
||||||
x27Layout->addWidget(new QLabel(tr("L3.5")), 0, 2, 1, 1);
|
|
||||||
x27Layout->addWidget(new QLabel(tr("Page no")), 0, 3, 1, 1);
|
|
||||||
x27Layout->addWidget(new QLabel(tr("Sub pages required")), 0, 4, 1, 1);
|
|
||||||
|
|
||||||
QLabel *level3p5OnlyLabel = new QLabel(tr("Level 3.5 only"));
|
|
||||||
level3p5OnlyLabel->setAlignment(Qt::AlignCenter);
|
|
||||||
x27Layout->addWidget(level3p5OnlyLabel, 5, 0, 1, 5);
|
|
||||||
|
|
||||||
for (int i=0; i<8; i++) {
|
|
||||||
if (i < 4) {
|
|
||||||
// Required at which Levels
|
|
||||||
m_composeLinkLevelCheckbox[i][0] = new QCheckBox(this);
|
|
||||||
x27Layout->addWidget(m_composeLinkLevelCheckbox[i][0], i+1, 1, 1, 1);
|
|
||||||
connect(m_composeLinkLevelCheckbox[i][0], &QCheckBox::stateChanged, [=](bool active) { m_parentMainWidget->document()->currentSubPage()->setComposeLinkLevel2p5(i, active); });
|
|
||||||
m_composeLinkLevelCheckbox[i][1] = new QCheckBox(this);
|
|
||||||
x27Layout->addWidget(m_composeLinkLevelCheckbox[i][1], i+1, 2, 1, 1);
|
|
||||||
connect(m_composeLinkLevelCheckbox[i][1], &QCheckBox::stateChanged, [=](bool active) { m_parentMainWidget->document()->currentSubPage()->setComposeLinkLevel3p5(i, active); });
|
|
||||||
} else {
|
|
||||||
// Selectable link functions for Level 3.5
|
|
||||||
m_composeLinkFunctionComboBox[i-4] = new QComboBox;
|
|
||||||
m_composeLinkFunctionComboBox[i-4]->addItem("GPOP");
|
|
||||||
m_composeLinkFunctionComboBox[i-4]->addItem("POP");
|
|
||||||
m_composeLinkFunctionComboBox[i-4]->addItem("GDRCS");
|
|
||||||
m_composeLinkFunctionComboBox[i-4]->addItem("DRCS");
|
|
||||||
x27Layout->addWidget(m_composeLinkFunctionComboBox[i-4], i+2, 0, 1, 1, Qt::AlignTop);
|
|
||||||
connect(m_composeLinkFunctionComboBox[i-4], QOverload<int>::of(&QComboBox::currentIndexChanged), [=](int index) { m_parentMainWidget->document()->currentSubPage()->setComposeLinkFunction(i, index); } );
|
|
||||||
}
|
|
||||||
// Page link
|
|
||||||
m_composeLinkPageNumberLineEdit[i] = new QLineEdit("100");
|
|
||||||
m_composeLinkPageNumberLineEdit[i]->setMaxLength(3);
|
|
||||||
m_composeLinkPageNumberLineEdit[i]->setInputMask("DHH");
|
|
||||||
// TODO restrict first digit of page number to 1-8
|
|
||||||
x27Layout->addWidget(m_composeLinkPageNumberLineEdit[i], i+(i<4 ? 1 : 2), 3, 1, 1);
|
|
||||||
connect(m_composeLinkPageNumberLineEdit[i], &QLineEdit::textEdited, [=](QString value) { setComposeLinkPageNumber(i, value); } );
|
|
||||||
|
|
||||||
// Sub page numbers required
|
|
||||||
m_composeLinkSubPageNumbersLineEdit[i] = new QLineEdit(this);
|
|
||||||
x27Layout->addWidget(m_composeLinkSubPageNumbersLineEdit[i], i+(i<4 ? 1 : 2), 4, 1, 1);
|
|
||||||
connect(m_composeLinkSubPageNumbersLineEdit[i], &QLineEdit::textEdited, [=](QString value) { setComposeLinkSubPageNumbers(i, value); } );
|
|
||||||
}
|
|
||||||
|
|
||||||
x27GroupBox->setLayout(x27Layout);
|
|
||||||
pageEnhancementsLayout->addWidget(x27GroupBox);
|
|
||||||
|
|
||||||
pageEnhancementsLayout->addStretch(1);
|
pageEnhancementsLayout->addStretch(1);
|
||||||
|
|
||||||
@@ -181,60 +132,6 @@ void PageEnhancementsDockWidget::setRightSidePanelWidth(int newRightPanelSize)
|
|||||||
m_parentMainWidget->setSidePanelWidths(m_leftSidePanelSpinBox->value(), newRightPanelSize);
|
m_parentMainWidget->setSidePanelWidths(m_leftSidePanelSpinBox->value(), newRightPanelSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
void PageEnhancementsDockWidget::setComposeLinkPageNumber(int linkNumber, const QString &newPageNumberString)
|
|
||||||
{
|
|
||||||
// The LineEdit should check if a valid hex number was entered, but just in case...
|
|
||||||
bool newPageNumberOk;
|
|
||||||
int newPageNumberRead = newPageNumberString.toInt(&newPageNumberOk, 16);
|
|
||||||
if ((!newPageNumberOk) || newPageNumberRead < 0x100 || newPageNumberRead > 0x8ff)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Stored as page link with relative magazine number, convert from absolute page number that was entered
|
|
||||||
newPageNumberRead ^= (m_parentMainWidget->document()->pageNumber() & 0x700);
|
|
||||||
m_parentMainWidget->document()->currentSubPage()->setComposeLinkPageNumber(linkNumber, newPageNumberRead);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PageEnhancementsDockWidget::setComposeLinkSubPageNumbers(int linkNumber, const QString &newSubPagesString)
|
|
||||||
{
|
|
||||||
int newSubPageCodes = 0x0000;
|
|
||||||
|
|
||||||
if (!newSubPagesString.isEmpty()) {
|
|
||||||
// Turn user-entered comma separated subpages and subpage-ranges into bits
|
|
||||||
// First we do the "comma separated"
|
|
||||||
const QStringList items = newSubPagesString.split(QLatin1Char(','), QString::SkipEmptyParts);
|
|
||||||
// Now see if there's valid single numbers or ranges between the commas
|
|
||||||
for (const QString &item : items) {
|
|
||||||
if (item.isEmpty())
|
|
||||||
return;
|
|
||||||
if (item.contains(QLatin1Char('-'))) {
|
|
||||||
// Dash found, should be a range of subpages: check for two things either side of dash
|
|
||||||
const QStringList rangeItems = item.split('-');
|
|
||||||
if (rangeItems.count() != 2)
|
|
||||||
return;
|
|
||||||
// Now check if the two things are valid numbers 0-15, with first number less than second
|
|
||||||
bool ok1, ok2;
|
|
||||||
const int number1 = rangeItems[0].toInt(&ok1);
|
|
||||||
const int number2 = rangeItems[1].toInt(&ok2);
|
|
||||||
if (!ok1 || !ok2 || number1 < 0 || number2 < 0 || number1 > 15 || number2 > 15 || number2 < number1)
|
|
||||||
return;
|
|
||||||
// Yes, it is a valid range. Apply it to the result
|
|
||||||
for (int i=number1; i<=number2; i++)
|
|
||||||
newSubPageCodes |= (1 << i);
|
|
||||||
} else {
|
|
||||||
// A single thing (maybe extracted between commas): check if the thing is a number 0-15
|
|
||||||
bool ok;
|
|
||||||
const int number = item.toInt(&ok);
|
|
||||||
if (!ok || number < 0 || number > 15)
|
|
||||||
return;
|
|
||||||
// Yes, it is a valid number. Apply it to the result
|
|
||||||
newSubPageCodes |= (1 << number);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
m_parentMainWidget->document()->currentSubPage()->setComposeLinkSubPageCodes(linkNumber, newSubPageCodes);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PageEnhancementsDockWidget::updateWidgets()
|
void PageEnhancementsDockWidget::updateWidgets()
|
||||||
{
|
{
|
||||||
int leftSidePanelColumnsResult = 0;
|
int leftSidePanelColumnsResult = 0;
|
||||||
@@ -268,64 +165,4 @@ void PageEnhancementsDockWidget::updateWidgets()
|
|||||||
m_sidePanelStatusAct->blockSignals(true);
|
m_sidePanelStatusAct->blockSignals(true);
|
||||||
m_sidePanelStatusAct->setChecked(!m_parentMainWidget->document()->currentSubPage()->sidePanelStatusL25());
|
m_sidePanelStatusAct->setChecked(!m_parentMainWidget->document()->currentSubPage()->sidePanelStatusL25());
|
||||||
m_sidePanelStatusAct->blockSignals(false);
|
m_sidePanelStatusAct->blockSignals(false);
|
||||||
|
|
||||||
for (int i=0; i<8; i++) {
|
|
||||||
if (i >= 4) {
|
|
||||||
m_composeLinkFunctionComboBox[i-4]->blockSignals(true);
|
|
||||||
m_composeLinkFunctionComboBox[i-4]->setCurrentIndex(m_parentMainWidget->document()->currentSubPage()->composeLinkFunction(i));
|
|
||||||
m_composeLinkFunctionComboBox[i-4]->blockSignals(false);
|
|
||||||
} else {
|
|
||||||
m_composeLinkLevelCheckbox[i][0]->blockSignals(true);
|
|
||||||
m_composeLinkLevelCheckbox[i][0]->setChecked(m_parentMainWidget->document()->currentSubPage()->composeLinkLevel2p5(i));
|
|
||||||
m_composeLinkLevelCheckbox[i][0]->blockSignals(false);
|
|
||||||
m_composeLinkLevelCheckbox[i][1]->blockSignals(true);
|
|
||||||
m_composeLinkLevelCheckbox[i][1]->setChecked(m_parentMainWidget->document()->currentSubPage()->composeLinkLevel3p5(i));
|
|
||||||
m_composeLinkLevelCheckbox[i][1]->blockSignals(false);
|
|
||||||
}
|
|
||||||
// Stored as page link with relative magazine number, convert to absolute page number for display
|
|
||||||
int absoluteLinkPageNumber = m_parentMainWidget->document()->currentSubPage()->composeLinkPageNumber(i) ^ (m_parentMainWidget->document()->pageNumber() & 0x700);
|
|
||||||
// Fix magazine 0 to 8
|
|
||||||
if ((absoluteLinkPageNumber & 0x700) == 0x000)
|
|
||||||
absoluteLinkPageNumber |= 0x800;
|
|
||||||
m_composeLinkPageNumberLineEdit[i]->blockSignals(true);
|
|
||||||
m_composeLinkPageNumberLineEdit[i]->setText(QString::number(absoluteLinkPageNumber, 16).toUpper());
|
|
||||||
m_composeLinkPageNumberLineEdit[i]->blockSignals(false);
|
|
||||||
|
|
||||||
// Turn subpage bits into user-friendly comma separated numbers and number-ranges
|
|
||||||
QString rangeString;
|
|
||||||
|
|
||||||
if (m_parentMainWidget->document()->currentSubPage()->composeLinkSubPageCodes(i) != 0x0000) {
|
|
||||||
// First build a list of consecutive ranges seen
|
|
||||||
// The "b-index" is based on https://codereview.stackexchange.com/a/90074
|
|
||||||
QMap<int, QPair<int, int>> ranges;
|
|
||||||
int index = 0;
|
|
||||||
for (int b=0; b<16; b++)
|
|
||||||
if ((m_parentMainWidget->document()->currentSubPage()->composeLinkSubPageCodes(i) >> b) & 0x01) {
|
|
||||||
if (!ranges.contains(b-index))
|
|
||||||
ranges.insert(b-index, qMakePair(b, b));
|
|
||||||
else
|
|
||||||
ranges[b-index].second = b;
|
|
||||||
index++;
|
|
||||||
}
|
|
||||||
// Now go through the list and add single numbers or ranges as appropriate
|
|
||||||
QPair<int, int> range;
|
|
||||||
foreach (range, ranges) {
|
|
||||||
// For second and subsequent entries only, append a comma first
|
|
||||||
if (!rangeString.isEmpty())
|
|
||||||
rangeString.append(',');
|
|
||||||
// Append the single number or the first number of the range
|
|
||||||
rangeString.append(QString("%1").arg(range.first));
|
|
||||||
// If that was part of a range and not a single orphaned number
|
|
||||||
if (range.first != range.second) {
|
|
||||||
// Ranges of 3 or more use a dash. A range of 2 can make do with a comma
|
|
||||||
rangeString.append((range.first+1 == range.second) ? ',' : '-');
|
|
||||||
// Append the second number of the range
|
|
||||||
rangeString.append(QString("%1").arg(range.second));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
m_composeLinkSubPageNumbersLineEdit[i]->blockSignals(true);
|
|
||||||
m_composeLinkSubPageNumbersLineEdit[i]->setText(rangeString);
|
|
||||||
m_composeLinkSubPageNumbersLineEdit[i]->blockSignals(false);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -23,9 +23,7 @@
|
|||||||
#include <QCheckBox>
|
#include <QCheckBox>
|
||||||
#include <QComboBox>
|
#include <QComboBox>
|
||||||
#include <QDockWidget>
|
#include <QDockWidget>
|
||||||
#include <QLineEdit>
|
|
||||||
#include <QSpinBox>
|
#include <QSpinBox>
|
||||||
#include <QString>
|
|
||||||
|
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
|
|
||||||
@@ -38,19 +36,13 @@ public:
|
|||||||
void updateWidgets();
|
void updateWidgets();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void setLeftSidePanelWidth(int);
|
void setLeftSidePanelWidth(int newLeftPanelSize);
|
||||||
void setRightSidePanelWidth(int);
|
void setRightSidePanelWidth(int newRightPanelSize);
|
||||||
void setComposeLinkPageNumber(int, const QString &);
|
|
||||||
void setComposeLinkSubPageNumbers(int, const QString &);
|
|
||||||
|
|
||||||
TeletextWidget *m_parentMainWidget;
|
TeletextWidget *m_parentMainWidget;
|
||||||
QComboBox *m_defaultScreenColourCombo, *m_defaultRowColourCombo, *m_colourTableCombo;
|
QComboBox *m_defaultScreenColourCombo, *m_defaultRowColourCombo, *m_colourTableCombo;
|
||||||
QCheckBox *m_blackBackgroundSubstAct, *m_sidePanelStatusAct;
|
QCheckBox *m_blackBackgroundSubstAct, *m_sidePanelStatusAct;
|
||||||
QSpinBox *m_leftSidePanelSpinBox, *m_rightSidePanelSpinBox;
|
QSpinBox *m_leftSidePanelSpinBox, *m_rightSidePanelSpinBox;
|
||||||
|
|
||||||
QCheckBox *m_composeLinkLevelCheckbox[4][2]; // For links 0-3
|
|
||||||
QComboBox *m_composeLinkFunctionComboBox[4]; // For links 4-7; remember to subtract 4!
|
|
||||||
QLineEdit *m_composeLinkPageNumberLineEdit[8], *m_composeLinkSubPageNumbersLineEdit[8];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -24,6 +24,7 @@
|
|||||||
#include <QHBoxLayout>
|
#include <QHBoxLayout>
|
||||||
#include <QLabel>
|
#include <QLabel>
|
||||||
#include <QLineEdit>
|
#include <QLineEdit>
|
||||||
|
#include <QRegularExpressionValidator>
|
||||||
#include <QSpinBox>
|
#include <QSpinBox>
|
||||||
#include <QVBoxLayout>
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
@@ -40,14 +41,16 @@ PageOptionsDockWidget::PageOptionsDockWidget(TeletextWidget *parent): QDockWidge
|
|||||||
this->setWindowTitle("Page options");
|
this->setWindowTitle("Page options");
|
||||||
|
|
||||||
// Page number
|
// Page number
|
||||||
|
m_pageNumberValidator = new QRegularExpressionValidator(QRegularExpression("[1-8][0-9A-Fa-f][0-9A-Fa-f]"), this);
|
||||||
|
|
||||||
QHBoxLayout *pageNumberLayout = new QHBoxLayout;
|
QHBoxLayout *pageNumberLayout = new QHBoxLayout;
|
||||||
pageNumberLayout->addWidget(new QLabel(tr("Page number")));
|
pageNumberLayout->addWidget(new QLabel(tr("Page number")));
|
||||||
m_pageNumberEdit = new QLineEdit("100");
|
m_pageNumberEdit = new QLineEdit("100");
|
||||||
m_pageNumberEdit->setMaxLength(3);
|
m_pageNumberEdit->setMaxLength(3);
|
||||||
m_pageNumberEdit->setInputMask("DHH");
|
m_pageNumberEdit->setInputMask(">DHH");
|
||||||
//TODO restrict first digit of page number to 1-8
|
m_pageNumberEdit->setValidator(m_pageNumberValidator);
|
||||||
pageNumberLayout->addWidget(m_pageNumberEdit);
|
pageNumberLayout->addWidget(m_pageNumberEdit);
|
||||||
connect(m_pageNumberEdit, &QLineEdit::textEdited, m_parentMainWidget->document(), &TeletextDocument::setPageNumber);
|
connect(m_pageNumberEdit, &QLineEdit::textEdited, m_parentMainWidget->document(), &TeletextDocument::setPageNumberFromString);
|
||||||
|
|
||||||
pageOptionsLayout->addLayout(pageNumberLayout);
|
pageOptionsLayout->addLayout(pageNumberLayout);
|
||||||
|
|
||||||
@@ -66,8 +69,8 @@ PageOptionsDockWidget::PageOptionsDockWidget(TeletextWidget *parent): QDockWidge
|
|||||||
fastTextLayout->addWidget(new QLabel(fastTextLabel[i]), 0, i, 1, 1, Qt::AlignCenter);
|
fastTextLayout->addWidget(new QLabel(fastTextLabel[i]), 0, i, 1, 1, Qt::AlignCenter);
|
||||||
m_fastTextEdit[i] = new QLineEdit;
|
m_fastTextEdit[i] = new QLineEdit;
|
||||||
m_fastTextEdit[i]->setMaxLength(3);
|
m_fastTextEdit[i]->setMaxLength(3);
|
||||||
m_fastTextEdit[i]->setInputMask("DHH");
|
m_fastTextEdit[i]->setInputMask(">DHH");
|
||||||
//TODO restrict first digit of page number to 1-8
|
m_fastTextEdit[i]->setValidator(m_pageNumberValidator);
|
||||||
fastTextLayout->addWidget(m_fastTextEdit[i], 1, i, 1, 1);
|
fastTextLayout->addWidget(m_fastTextEdit[i], 1, i, 1, 1);
|
||||||
connect(m_fastTextEdit[i], &QLineEdit::textEdited, [=](QString value) { setFastTextLinkPageNumber(i, value); } );
|
connect(m_fastTextEdit[i], &QLineEdit::textEdited, [=](QString value) { setFastTextLinkPageNumber(i, value); } );
|
||||||
}
|
}
|
||||||
@@ -189,6 +192,9 @@ void PageOptionsDockWidget::updateWidgets()
|
|||||||
m_defaultNOSCombo->setCurrentIndex(m_defaultNOSCombo->findData((m_parentMainWidget->document()->currentSubPage()->defaultCharSet() << 3) | m_parentMainWidget->document()->currentSubPage()->defaultNOS()));
|
m_defaultNOSCombo->setCurrentIndex(m_defaultNOSCombo->findData((m_parentMainWidget->document()->currentSubPage()->defaultCharSet() << 3) | m_parentMainWidget->document()->currentSubPage()->defaultNOS()));
|
||||||
m_defaultNOSCombo->blockSignals(false);
|
m_defaultNOSCombo->blockSignals(false);
|
||||||
m_secondRegionCombo->blockSignals(true);
|
m_secondRegionCombo->blockSignals(true);
|
||||||
|
if (m_parentMainWidget->document()->currentSubPage()->secondCharSet() == 15)
|
||||||
|
m_secondRegionCombo->setCurrentIndex(0);
|
||||||
|
else
|
||||||
m_secondRegionCombo->setCurrentText(QString::number(m_parentMainWidget->document()->currentSubPage()->secondCharSet()));
|
m_secondRegionCombo->setCurrentText(QString::number(m_parentMainWidget->document()->currentSubPage()->secondCharSet()));
|
||||||
m_secondRegionCombo->blockSignals(false);
|
m_secondRegionCombo->blockSignals(false);
|
||||||
m_secondNOSCombo->blockSignals(true);
|
m_secondNOSCombo->blockSignals(true);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -24,6 +24,7 @@
|
|||||||
#include <QComboBox>
|
#include <QComboBox>
|
||||||
#include <QDockWidget>
|
#include <QDockWidget>
|
||||||
#include <QLineEdit>
|
#include <QLineEdit>
|
||||||
|
#include <QRegularExpressionValidator>
|
||||||
#include <QSpinBox>
|
#include <QSpinBox>
|
||||||
|
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
@@ -45,8 +46,10 @@ private:
|
|||||||
QComboBox *m_defaultRegionCombo, *m_defaultNOSCombo, *m_secondRegionCombo, *m_secondNOSCombo;
|
QComboBox *m_defaultRegionCombo, *m_defaultNOSCombo, *m_secondRegionCombo, *m_secondNOSCombo;
|
||||||
QLineEdit *m_fastTextEdit[6];
|
QLineEdit *m_fastTextEdit[6];
|
||||||
|
|
||||||
void addRegionList(QComboBox *);
|
QRegularExpressionValidator *m_pageNumberValidator;
|
||||||
void setFastTextLinkPageNumber(int, const QString &);
|
|
||||||
|
void addRegionList(QComboBox *regionCombo);
|
||||||
|
void setFastTextLinkPageNumber(int linkNumber, const QString &pageNumberString);
|
||||||
void setDefaultRegion();
|
void setDefaultRegion();
|
||||||
void setDefaultNOS();
|
void setDefaultNOS();
|
||||||
void setSecondRegion();
|
void setSecondRegion();
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -74,7 +74,7 @@ void PageX26Base::setEnhancementListFromPacket(int packetNumber, QByteArray pack
|
|||||||
newX26Triplet.setAddress(packetContents.at(i*3+1) & 0x3f);
|
newX26Triplet.setAddress(packetContents.at(i*3+1) & 0x3f);
|
||||||
newX26Triplet.setMode(packetContents.at(i*3+2) & 0x1f);
|
newX26Triplet.setMode(packetContents.at(i*3+2) & 0x1f);
|
||||||
newX26Triplet.setData(((packetContents.at(i*3+3) & 0x3f) << 1) | ((packetContents.at(i*3+2) & 0x20) >> 5));
|
newX26Triplet.setData(((packetContents.at(i*3+3) & 0x3f) << 1) | ((packetContents.at(i*3+2) & 0x20) >> 5));
|
||||||
m_enhancements[enhanceListPointer] = newX26Triplet;
|
m_enhancements.replace(enhanceListPointer, newX26Triplet);
|
||||||
}
|
}
|
||||||
if (newX26Triplet.mode() == 0x1f && newX26Triplet.address() == 0x3f && newX26Triplet.data() & 0x01)
|
if (newX26Triplet.mode() == 0x1f && newX26Triplet.address() == 0x3f && newX26Triplet.data() & 0x01)
|
||||||
// Last triplet was a Termination Marker (without ..follows) so clean up the repeated ones
|
// Last triplet was a Termination Marker (without ..follows) so clean up the repeated ones
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -31,14 +31,15 @@ class PageX26Base : public PageBase //: public QObject
|
|||||||
//Q_OBJECT
|
//Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
QList<X26Triplet> *enhancements() { return &m_enhancements; };
|
X26TripletList *enhancements() { return &m_enhancements; };
|
||||||
|
virtual int maxEnhancements() const =0;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
QByteArray packetFromEnhancementList(int) const;
|
QByteArray packetFromEnhancementList(int packetNumber) const;
|
||||||
void setEnhancementListFromPacket(int, QByteArray);
|
void setEnhancementListFromPacket(int packetNumber, QByteArray packetContents);
|
||||||
bool packetFromEnhancementListNeeded(int n) const { return ((m_enhancements.size()+12) / 13) > n; };
|
bool packetFromEnhancementListNeeded(int n) const { return ((m_enhancements.size()+12) / 13) > n; };
|
||||||
|
|
||||||
QList<X26Triplet> m_enhancements;
|
X26TripletList m_enhancements;
|
||||||
const X26Triplet m_paddingX26Triplet { 41, 0x1e, 0 };
|
const X26Triplet m_paddingX26Triplet { 41, 0x1e, 0 };
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -39,6 +39,9 @@ PaletteDockWidget::PaletteDockWidget(TeletextWidget *parent): QDockWidget(parent
|
|||||||
|
|
||||||
m_parentMainWidget = parent;
|
m_parentMainWidget = parent;
|
||||||
|
|
||||||
|
m_level3p5Accepted = false; // true when Level 3.5 radio button is checked in main window
|
||||||
|
m_level3p5Seen = false; // true when CLUT 0 or 1 is redefined
|
||||||
|
|
||||||
this->setObjectName("PaletteDockWidget");
|
this->setObjectName("PaletteDockWidget");
|
||||||
this->setWindowTitle("Palette");
|
this->setWindowTitle("Palette");
|
||||||
for (int c=0; c<=7; c++)
|
for (int c=0; c<=7; c++)
|
||||||
@@ -57,6 +60,9 @@ PaletteDockWidget::PaletteDockWidget(TeletextWidget *parent): QDockWidget(parent
|
|||||||
connect(m_colourButton[i], &QAbstractButton::clicked, [=]() { selectColour(i); });
|
connect(m_colourButton[i], &QAbstractButton::clicked, [=]() { selectColour(i); });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
updateLevel3p5Cursor();
|
||||||
|
|
||||||
m_showHexValuesCheckBox = new QCheckBox(tr("Show colour hex values"));
|
m_showHexValuesCheckBox = new QCheckBox(tr("Show colour hex values"));
|
||||||
paletteGridLayout->addWidget(m_showHexValuesCheckBox, 5, 1, 1, 8);
|
paletteGridLayout->addWidget(m_showHexValuesCheckBox, 5, 1, 1, 8);
|
||||||
connect(m_showHexValuesCheckBox, &QCheckBox::stateChanged, this, &PaletteDockWidget::updateAllColourButtons);
|
connect(m_showHexValuesCheckBox, &QCheckBox::stateChanged, this, &PaletteDockWidget::updateAllColourButtons);
|
||||||
@@ -70,26 +76,34 @@ void PaletteDockWidget::updateColourButton(int colourIndex)
|
|||||||
{
|
{
|
||||||
if (colourIndex == 8)
|
if (colourIndex == 8)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
QString colourString = QString("%1").arg(m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex), 3, 16, QChar('0'));
|
QString colourString = QString("%1").arg(m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex), 3, 16, QChar('0'));
|
||||||
if (m_showHexValuesCheckBox->isChecked())
|
if (m_showHexValuesCheckBox->isChecked())
|
||||||
m_colourButton[colourIndex]->setText(colourString);
|
m_colourButton[colourIndex]->setText(colourString);
|
||||||
else
|
else
|
||||||
m_colourButton[colourIndex]->setText(nullptr);
|
m_colourButton[colourIndex]->setText(nullptr);
|
||||||
|
|
||||||
// Set text itself to black or white so it can be seen over background colour - http://alienryderflex.com/hsp.html
|
|
||||||
int r = m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex) >> 8;
|
int r = m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex) >> 8;
|
||||||
int g = (m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex) >> 4) & 0xf;
|
int g = (m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex) >> 4) & 0xf;
|
||||||
int b = m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex) & 0xf;
|
int b = m_parentMainWidget->document()->currentSubPage()->CLUT(colourIndex) & 0xf;
|
||||||
|
// Set text itself to black or white so it can be seen over background colour - http://alienryderflex.com/hsp.html
|
||||||
char blackOrWhite = (sqrt(r*r*0.299 + g*g*0.587 + b*b*0.114) > 7.647) ? '0' : 'f';
|
char blackOrWhite = (sqrt(r*r*0.299 + g*g*0.587 + b*b*0.114) > 7.647) ? '0' : 'f';
|
||||||
|
|
||||||
QString qss = QString("background-color: #%1; color: #%2%2%2; border: none").arg(colourString).arg(blackOrWhite);
|
QString qss = QString("background-color: #%1; color: #%2%2%2; border: none").arg(colourString).arg(blackOrWhite);
|
||||||
m_colourButton[colourIndex]->setStyleSheet(qss);
|
m_colourButton[colourIndex]->setStyleSheet(qss);
|
||||||
|
|
||||||
if (m_parentMainWidget->document()->currentSubPage()->isPaletteDefault(colourIndex))
|
if (m_parentMainWidget->document()->currentSubPage()->isPaletteDefault(colourIndex)) {
|
||||||
// Default colour was set, disable Reset button if all colours in row are default as well
|
// Default colour was set, disable Reset button if all colours in a CLUT row are default as well
|
||||||
m_resetButton[colourIndex>>3]->setEnabled(!m_parentMainWidget->document()->currentSubPage()->isPaletteDefault(colourIndex & 0x18, colourIndex | 0x07));
|
m_resetButton[colourIndex >> 3]->setEnabled(!m_parentMainWidget->document()->currentSubPage()->isPaletteDefault(colourIndex & 0x18, colourIndex | 0x07));
|
||||||
else
|
// Check if CLUTs 0 and 1 are all default if necessary
|
||||||
m_resetButton[colourIndex>>3]->setEnabled(true);
|
if (colourIndex < 16 && m_parentMainWidget->document()->currentSubPage()->isPaletteDefault(0, 15))
|
||||||
|
setLevel3p5Seen(false);
|
||||||
|
} else {
|
||||||
|
// Custom colour was set, enable Reset button for that CLUT
|
||||||
|
m_resetButton[colourIndex >> 3]->setEnabled(true);
|
||||||
|
if (colourIndex < 16)
|
||||||
|
setLevel3p5Seen(true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void PaletteDockWidget::updateAllColourButtons()
|
void PaletteDockWidget::updateAllColourButtons()
|
||||||
@@ -98,6 +112,41 @@ void PaletteDockWidget::updateAllColourButtons()
|
|||||||
updateColourButton(i);
|
updateColourButton(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void PaletteDockWidget::setLevel3p5Accepted(bool b)
|
||||||
|
{
|
||||||
|
if (b == m_level3p5Accepted)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_level3p5Accepted = b;
|
||||||
|
|
||||||
|
updateLevel3p5Cursor();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PaletteDockWidget::setLevel3p5Seen(bool b)
|
||||||
|
{
|
||||||
|
if (b == m_level3p5Seen)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_level3p5Seen = b;
|
||||||
|
|
||||||
|
updateLevel3p5Cursor();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PaletteDockWidget::updateLevel3p5Cursor()
|
||||||
|
{
|
||||||
|
const Qt::CursorShape cursor = (m_level3p5Accepted || m_level3p5Seen) ? Qt::ArrowCursor : Qt::ForbiddenCursor;
|
||||||
|
|
||||||
|
if (m_colourButton[0]->cursor() == cursor)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (int i=0; i<16; i++) {
|
||||||
|
if (i == 8)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
m_colourButton[i]->setCursor(cursor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void PaletteDockWidget::showEvent(QShowEvent *event)
|
void PaletteDockWidget::showEvent(QShowEvent *event)
|
||||||
{
|
{
|
||||||
Q_UNUSED(event);
|
Q_UNUSED(event);
|
||||||
@@ -106,16 +155,16 @@ void PaletteDockWidget::showEvent(QShowEvent *event)
|
|||||||
|
|
||||||
void PaletteDockWidget::selectColour(int colourIndex)
|
void PaletteDockWidget::selectColour(int colourIndex)
|
||||||
{
|
{
|
||||||
|
if (colourIndex < 16 && !m_level3p5Accepted && !m_level3p5Seen)
|
||||||
|
return;
|
||||||
|
|
||||||
const QColor newColour = QColorDialog::getColor(m_parentMainWidget->document()->currentSubPage()->CLUTtoQColor(colourIndex), this, "Select Colour");
|
const QColor newColour = QColorDialog::getColor(m_parentMainWidget->document()->currentSubPage()->CLUTtoQColor(colourIndex), this, "Select Colour");
|
||||||
|
|
||||||
if (newColour.isValid()) {
|
if (newColour.isValid())
|
||||||
QUndoCommand *setColourCommand = new SetColourCommand(m_parentMainWidget->document(), colourIndex, ((newColour.red() & 0xf0) << 4) | (newColour.green() & 0xf0) | ((newColour.blue() & 0xf0) >> 4));
|
m_parentMainWidget->document()->undoStack()->push(new SetColourCommand(m_parentMainWidget->document(), colourIndex, ((newColour.red() & 0xf0) << 4) | (newColour.green() & 0xf0) | ((newColour.blue() & 0xf0) >> 4)));
|
||||||
m_parentMainWidget->document()->undoStack()->push(setColourCommand);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void PaletteDockWidget::resetCLUT(int colourTable)
|
void PaletteDockWidget::resetCLUT(int colourTable)
|
||||||
{
|
{
|
||||||
QUndoCommand *resetCLUTCommand = new ResetCLUTCommand(m_parentMainWidget->document(), colourTable);
|
m_parentMainWidget->document()->undoStack()->push(new ResetCLUTCommand(m_parentMainWidget->document(), colourTable));
|
||||||
m_parentMainWidget->document()->undoStack()->push(resetCLUTCommand);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -35,19 +35,23 @@ public:
|
|||||||
void updateAllColourButtons();
|
void updateAllColourButtons();
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void updateColourButton(int);
|
void updateColourButton(int colourIndex);
|
||||||
|
void setLevel3p5Accepted(bool b);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void showEvent(QShowEvent *);
|
void showEvent(QShowEvent *event);
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void selectColour(int);
|
void selectColour(int colourIndex);
|
||||||
|
void setLevel3p5Seen(bool b);
|
||||||
|
void updateLevel3p5Cursor();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void resetCLUT(int);
|
void resetCLUT(int colourTable);
|
||||||
|
|
||||||
QPushButton *m_colourButton[32], *m_resetButton[4];
|
QPushButton *m_colourButton[32], *m_resetButton[4];
|
||||||
QCheckBox *m_showHexValuesCheckBox;
|
QCheckBox *m_showHexValuesCheckBox;
|
||||||
|
bool m_level3p5Accepted, m_level3p5Seen;
|
||||||
|
|
||||||
TeletextWidget *m_parentMainWidget;
|
TeletextWidget *m_parentMainWidget;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
QT += widgets
|
QT += widgets
|
||||||
requires(qtConfig(filedialog))
|
requires(qtConfig(filedialog))
|
||||||
|
|
||||||
HEADERS = document.h \
|
HEADERS = decode.h \
|
||||||
|
document.h \
|
||||||
|
hamming.h \
|
||||||
keymap.h \
|
keymap.h \
|
||||||
levelonecommands.h \
|
levelonecommands.h \
|
||||||
levelonepage.h \
|
levelonepage.h \
|
||||||
@@ -10,6 +12,7 @@ HEADERS = document.h \
|
|||||||
mainwindow.h \
|
mainwindow.h \
|
||||||
pagebase.h \
|
pagebase.h \
|
||||||
pagex26base.h \
|
pagex26base.h \
|
||||||
|
pagecomposelinksdockwidget.h \
|
||||||
pageenhancementsdockwidget.h \
|
pageenhancementsdockwidget.h \
|
||||||
pageoptionsdockwidget.h \
|
pageoptionsdockwidget.h \
|
||||||
palettedockwidget.h \
|
palettedockwidget.h \
|
||||||
@@ -18,7 +21,8 @@ HEADERS = document.h \
|
|||||||
x26dockwidget.h \
|
x26dockwidget.h \
|
||||||
x26model.h \
|
x26model.h \
|
||||||
x26triplets.h
|
x26triplets.h
|
||||||
SOURCES = document.cpp \
|
SOURCES = decode.cpp \
|
||||||
|
document.cpp \
|
||||||
levelonecommands.cpp \
|
levelonecommands.cpp \
|
||||||
levelonepage.cpp \
|
levelonepage.cpp \
|
||||||
loadsave.cpp \
|
loadsave.cpp \
|
||||||
@@ -27,6 +31,7 @@ SOURCES = document.cpp \
|
|||||||
mainwindow.cpp \
|
mainwindow.cpp \
|
||||||
pagebase.cpp \
|
pagebase.cpp \
|
||||||
pagex26base.cpp \
|
pagex26base.cpp \
|
||||||
|
pagecomposelinksdockwidget.cpp \
|
||||||
pageenhancementsdockwidget.cpp \
|
pageenhancementsdockwidget.cpp \
|
||||||
pageoptionsdockwidget.cpp \
|
pageoptionsdockwidget.cpp \
|
||||||
palettedockwidget.cpp \
|
palettedockwidget.cpp \
|
||||||
|
|||||||
1370
render.cpp
1370
render.cpp
File diff suppressed because it is too large
Load Diff
237
render.h
237
render.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -21,158 +21,25 @@
|
|||||||
#define RENDER_H
|
#define RENDER_H
|
||||||
|
|
||||||
#include <QBitmap>
|
#include <QBitmap>
|
||||||
#include <QMap>
|
#include <QColor>
|
||||||
#include <QMultiMap>
|
#include <QImage>
|
||||||
#include <QPair>
|
#include <QPixmap>
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#include "levelonepage.h"
|
#include "decode.h"
|
||||||
|
|
||||||
|
class TeletextFontBitmap
|
||||||
struct textCharacter {
|
|
||||||
unsigned char code=0x20;
|
|
||||||
int set=0;
|
|
||||||
int diacritical=0;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct displayAttributes {
|
|
||||||
bool doubleHeight=false;
|
|
||||||
bool doubleWidth=false;
|
|
||||||
bool boxingWindow=false;
|
|
||||||
bool conceal=false;
|
|
||||||
bool invert=false;
|
|
||||||
bool underlineSeparated=false;
|
|
||||||
bool forceContiguous=false;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct textAttributes {
|
|
||||||
int foreColour=0x07;
|
|
||||||
int backColour=0x00;
|
|
||||||
struct flashFunctions {
|
|
||||||
int mode=0;
|
|
||||||
int ratePhase=0;
|
|
||||||
int phaseNumber=0;
|
|
||||||
} flash;
|
|
||||||
displayAttributes display;
|
|
||||||
/* font style */
|
|
||||||
};
|
|
||||||
|
|
||||||
struct textCell {
|
|
||||||
textCharacter character;
|
|
||||||
textAttributes attribute;
|
|
||||||
bool bottomHalf=false;
|
|
||||||
bool rightHalf=false;
|
|
||||||
bool level1Mosaic=false;
|
|
||||||
int level1CharSet=0;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct applyAttributes {
|
|
||||||
bool applyForeColour=false;
|
|
||||||
bool applyBackColour=false;
|
|
||||||
bool applyFlash=false;
|
|
||||||
bool applyDisplayAttributes=false;
|
|
||||||
bool applyTextSizeOnly=false;
|
|
||||||
bool applyBoxingOnly=false;
|
|
||||||
bool applyConcealOnly=false;
|
|
||||||
bool applyContiguousOnly=false;
|
|
||||||
bool copyAboveAttributes=false;
|
|
||||||
textAttributes attribute;
|
|
||||||
};
|
|
||||||
|
|
||||||
class ActivePosition
|
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
ActivePosition();
|
TeletextFontBitmap();
|
||||||
int row() const { return (m_row == -1) ? 0 : m_row; }
|
~TeletextFontBitmap();
|
||||||
int column() const { return (m_column == -1) ? 0 : m_column; }
|
|
||||||
bool isDeployed() const { return m_row != -1; }
|
QBitmap *rawBitmap() const { return s_fontBitmap; }
|
||||||
bool setRow(int);
|
QImage *image() const { return s_fontImage; }
|
||||||
bool setColumn(int);
|
|
||||||
// bool setRowAndColumn(int, int);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int m_row, m_column;
|
static int s_instances;
|
||||||
};
|
static QBitmap* s_fontBitmap;
|
||||||
|
static QImage* s_fontImage;
|
||||||
|
|
||||||
class TextLayer
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
// TextLayer(TeletextPage* thePage) : currentPage(thePage) { };
|
|
||||||
virtual ~TextLayer() = default;
|
|
||||||
void setTeletextPage(LevelOnePage *);
|
|
||||||
virtual textCharacter character(int, int) =0;
|
|
||||||
virtual void attributes(int, int, applyAttributes *) =0;
|
|
||||||
virtual int fullScreenColour() const =0;
|
|
||||||
virtual int fullRowColour(int) const =0;
|
|
||||||
virtual bool fullRowDownwards(int) const =0;
|
|
||||||
virtual int objectType() const =0;
|
|
||||||
void setFullScreenColour(int);
|
|
||||||
void setFullRowColour(int, int, bool);
|
|
||||||
|
|
||||||
// Key QPair is row and column, value QPair is triplet mode and data
|
|
||||||
QMultiMap<QPair<int, int>, QPair<int, int>> enhanceMap;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
LevelOnePage* m_levelOnePage;
|
|
||||||
int m_layerFullScreenColour=-1;
|
|
||||||
int m_layerFullRowColour[25];
|
|
||||||
bool m_layerFullRowDownwards[25];
|
|
||||||
applyAttributes m_applyAttributes;
|
|
||||||
};
|
|
||||||
|
|
||||||
class EnhanceLayer: public TextLayer
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
EnhanceLayer();
|
|
||||||
textCharacter character(int, int);
|
|
||||||
void attributes(int, int, applyAttributes *);
|
|
||||||
int fullScreenColour() const { return m_layerFullScreenColour; };
|
|
||||||
int fullRowColour(int r) const { return m_layerFullRowColour[r]; };
|
|
||||||
bool fullRowDownwards(int r) const { return m_layerFullRowDownwards[r]; };
|
|
||||||
int objectType() const { return m_objectType; };
|
|
||||||
void setObjectType(int);
|
|
||||||
void setOrigin(int, int);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
int m_objectType=0;
|
|
||||||
int m_originR=0;
|
|
||||||
int m_originC=0;
|
|
||||||
int m_rowCached=-1;
|
|
||||||
int m_rightMostColumn[25];
|
|
||||||
};
|
|
||||||
|
|
||||||
class Level1Layer: public TextLayer
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
// Level1Layer(TeletextPage *thePage) : TextLayer(thePage) { };
|
|
||||||
Level1Layer();
|
|
||||||
textCharacter character(int, int);
|
|
||||||
void attributes(int, int, applyAttributes *);
|
|
||||||
int fullScreenColour() const { return -1; };
|
|
||||||
int fullRowColour(int) const { return -1; };
|
|
||||||
bool fullRowDownwards(int) const { return false; };
|
|
||||||
int objectType() const { return 0; }
|
|
||||||
bool isRowBottomHalf(int r) const { return m_rowHeight[r]==RHbottomhalf; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
void updateRowCache(int);
|
|
||||||
|
|
||||||
struct level1CacheAttributes {
|
|
||||||
int foreColour=0x07;
|
|
||||||
int backColour=0x00;
|
|
||||||
unsigned char sizeCode=0x0c;
|
|
||||||
bool mosaics=false;
|
|
||||||
bool separated=false;
|
|
||||||
bool held=false;
|
|
||||||
bool escSwitch=false;
|
|
||||||
unsigned char holdChar=0x20;
|
|
||||||
bool holdSeparated=false;
|
|
||||||
};
|
|
||||||
level1CacheAttributes m_attributeCache[40];
|
|
||||||
int m_rowCached=-1;
|
|
||||||
bool m_rowHasDoubleHeightAttr[25];
|
|
||||||
enum rowHeightEnum { RHnormal=-1, RHtophalf, RHbottomhalf } m_rowHeight[25];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class TeletextPageRender : public QObject
|
class TeletextPageRender : public QObject
|
||||||
@@ -182,66 +49,40 @@ class TeletextPageRender : public QObject
|
|||||||
public:
|
public:
|
||||||
TeletextPageRender();
|
TeletextPageRender();
|
||||||
~TeletextPageRender();
|
~TeletextPageRender();
|
||||||
void decodePage();
|
|
||||||
void renderPage();
|
QImage* image(int i) const { return m_pageImage[i]; };
|
||||||
void renderPage(int r);
|
bool mix() const { return m_mix; };
|
||||||
void setTeletextPage(LevelOnePage *);
|
void setDecoder(TeletextPageDecode *decoder);
|
||||||
void updateSidePanels();
|
void renderPage(bool force=false);
|
||||||
void buildEnhanceMap(TextLayer *, int=0);
|
bool showControlCodes() const { return m_showControlCodes; };
|
||||||
QPixmap* pagePixmap(int i) const { return m_pagePixmap[i]; };
|
|
||||||
bool level1MosaicAttribute(int r, int c) const { return m_cell[r][c].level1Mosaic; };
|
|
||||||
int level1CharSet(int r, int c) const { return m_cell[r][c].level1CharSet; };
|
|
||||||
int leftSidePanelColumns() const { return m_leftSidePanelColumns; };
|
|
||||||
int rightSidePanelColumns() const { return m_rightSidePanelColumns; };
|
|
||||||
void setGrid(bool);
|
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void setReveal(bool);
|
void colourChanged(int index);
|
||||||
void setMix(bool);
|
void setReveal(bool reveal);
|
||||||
void setShowCodes(bool);
|
void setMix(bool mix);
|
||||||
void setRenderLevel(int);
|
void setShowControlCodes(bool showControlCodes);
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void fullScreenColourChanged(QColor);
|
void flashChanged(int newFlashHz);
|
||||||
void fullRowColourChanged(int, QColor);
|
|
||||||
void flashChanged(int);
|
|
||||||
void sidePanelsChanged();
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void updateFlashRequired(int);
|
TeletextFontBitmap m_fontBitmap;
|
||||||
inline void setFullScreenColour(int);
|
QImage* m_pageImage[6];
|
||||||
inline void setFullRowColour(int, int);
|
unsigned char m_controlCodeCache[25][40];
|
||||||
|
bool m_reveal, m_mix, m_showControlCodes;
|
||||||
QBitmap* m_fontBitmap;
|
int m_flashBuffersHz;
|
||||||
QPixmap* m_pagePixmap[6];
|
int m_flashingRow[25];
|
||||||
int m_finalFullScreenColour, m_renderLevel;
|
|
||||||
QColor m_finalFullScreenQColor;
|
|
||||||
int m_leftSidePanelColumns, m_rightSidePanelColumns;
|
|
||||||
bool m_reveal, m_mix, m_grid, m_showCodes;
|
|
||||||
Level1Layer m_level1Layer;
|
|
||||||
std::vector<TextLayer *> m_textLayer;
|
|
||||||
const int m_foregroundRemap[8] = { 0, 0, 0, 8, 8, 16, 16, 16 };
|
|
||||||
const int m_backgroundRemap[8] = { 0, 8, 16, 8, 16, 8, 16, 24 };
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
textCell m_cell[25][72];
|
inline void drawFromBitmap(QPainter &, int, int, const QImage, TeletextPageDecode::CharacterFragment);
|
||||||
LevelOnePage* m_levelOnePage;
|
inline void drawFromFontBitmap(QPainter &painter, int r, int c, unsigned char characterCode, int characterSet, TeletextPageDecode::CharacterFragment characterFragment);
|
||||||
int m_flashRequired;
|
inline void drawCharacter(QPainter &painter, int r, int c, unsigned char characterCode, int characterSet, int characterDiacritical, TeletextPageDecode::CharacterFragment characterFragment);
|
||||||
int m_fullRowColour[25];
|
inline void drawBoldOrItalicCharacter(QPainter &painter, int r, int c, unsigned char characterCode, int characterSet, TeletextPageDecode::CharacterFragment characterFragment);
|
||||||
QColor m_fullRowQColor[25];
|
void renderRow(int r, int ph, bool force=false);
|
||||||
int m_flashRow[25];
|
void setRowFlashStatus(int r, int rowFlashHz);
|
||||||
bool m_concealRow[25];
|
|
||||||
};
|
|
||||||
|
|
||||||
static const QMap<int, int> g0CharacterMap {
|
QColor m_foregroundQColor, m_backgroundQColor;
|
||||||
{ 0x00, 12 }, { 0x01, 15 }, { 0x02, 22 }, { 0x03, 16 }, { 0x04, 14 }, { 0x05, 19 }, { 0x06, 11 },
|
TeletextPageDecode *m_decoder;
|
||||||
{ 0x08, 18 }, { 0x09, 15 }, { 0x0a, 22 }, { 0x0b, 16 }, { 0x0c, 14 }, { 0x0e, 11 },
|
|
||||||
{ 0x10, 12 }, { 0x11, 15 }, { 0x12, 22 }, { 0x13, 16 }, { 0x14, 14 }, { 0x15, 19 }, { 0x16, 23 },
|
|
||||||
{ 0x1d, 21 }, { 0x1f, 20 },
|
|
||||||
{ 0x20, 1 }, { 0x21, 15 }, { 0x22, 13 }, { 0x23, 17 }, { 0x24, 2 }, { 0x25, 3 }, { 0x26, 11 },
|
|
||||||
{ 0x36, 23 }, { 0x37, 4 },
|
|
||||||
{ 0x40, 12 }, { 0x44, 14 }, { 0x47, 5 },
|
|
||||||
{ 0x55, 6 }, { 0x57, 5 }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -50,17 +50,30 @@ void InsertTripletCommand::redo()
|
|||||||
for (int i=0; i<m_count; i++)
|
for (int i=0; i<m_count; i++)
|
||||||
m_teletextDocument->currentSubPage()->enhancements()->insert(m_row+i, m_insertedTriplet);
|
m_teletextDocument->currentSubPage()->enhancements()->insert(m_row+i, m_insertedTriplet);
|
||||||
|
|
||||||
|
if (!changingSubPage)
|
||||||
|
m_x26Model->endInsertRows();
|
||||||
|
|
||||||
|
// Preserve pointers to local object definitions that have moved
|
||||||
|
for (int i=0; i<m_teletextDocument->currentSubPage()->enhancements()->size(); i++) {
|
||||||
|
X26Triplet triplet = m_teletextDocument->currentSubPage()->enhancements()->at(i);
|
||||||
|
|
||||||
|
if (triplet.modeExt() >= 0x11 && triplet.modeExt() <= 0x13 && ((triplet.address() & 0x18) == 0x08) && triplet.objectLocalIndex() >= m_row) {
|
||||||
|
triplet.setObjectLocalIndex(triplet.objectLocalIndex() + m_count);
|
||||||
|
m_teletextDocument->currentSubPage()->enhancements()->replace(i, triplet);
|
||||||
|
if (!changingSubPage)
|
||||||
|
m_x26Model->emit dataChanged(m_x26Model->createIndex(i, 0), m_x26Model->createIndex(i, 3));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (changingSubPage)
|
if (changingSubPage)
|
||||||
m_teletextDocument->emit subPageSelected();
|
m_teletextDocument->emit subPageSelected();
|
||||||
else {
|
else
|
||||||
m_x26Model->endInsertRows();
|
m_teletextDocument->emit contentsChanged();
|
||||||
m_teletextDocument->emit refreshNeeded();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (m_firstDo)
|
if (m_firstDo)
|
||||||
m_firstDo = false;
|
m_firstDo = false;
|
||||||
else
|
|
||||||
m_teletextDocument->emit tripletCommandHighlight(m_row+1);
|
m_teletextDocument->emit tripletCommandHighlight(m_row);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InsertTripletCommand::undo()
|
void InsertTripletCommand::undo()
|
||||||
@@ -76,12 +89,25 @@ void InsertTripletCommand::undo()
|
|||||||
for (int i=0; i<m_count; i++)
|
for (int i=0; i<m_count; i++)
|
||||||
m_teletextDocument->currentSubPage()->enhancements()->removeAt(m_row);
|
m_teletextDocument->currentSubPage()->enhancements()->removeAt(m_row);
|
||||||
|
|
||||||
|
if (!changingSubPage)
|
||||||
|
m_x26Model->endRemoveRows();
|
||||||
|
|
||||||
|
// Preserve pointers to local object definitions that have moved
|
||||||
|
for (int i=0; i<m_teletextDocument->currentSubPage()->enhancements()->size(); i++) {
|
||||||
|
X26Triplet triplet = m_teletextDocument->currentSubPage()->enhancements()->at(i);
|
||||||
|
|
||||||
|
if (triplet.modeExt() >= 0x11 && triplet.modeExt() <= 0x13 && ((triplet.address() & 0x18) == 0x08) && triplet.objectLocalIndex() >= m_row) {
|
||||||
|
triplet.setObjectLocalIndex(triplet.objectLocalIndex() - m_count);
|
||||||
|
m_teletextDocument->currentSubPage()->enhancements()->replace(i, triplet);
|
||||||
|
if (!changingSubPage)
|
||||||
|
m_x26Model->emit dataChanged(m_x26Model->createIndex(i, 0), m_x26Model->createIndex(i, 3));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (changingSubPage)
|
if (changingSubPage)
|
||||||
m_teletextDocument->emit subPageSelected();
|
m_teletextDocument->emit subPageSelected();
|
||||||
else {
|
else
|
||||||
m_x26Model->endRemoveRows();
|
m_teletextDocument->emit contentsChanged();
|
||||||
m_teletextDocument->emit refreshNeeded();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -101,15 +127,35 @@ DeleteTripletCommand::DeleteTripletCommand(TeletextDocument *teletextDocument, X
|
|||||||
|
|
||||||
void DeleteTripletCommand::redo()
|
void DeleteTripletCommand::redo()
|
||||||
{
|
{
|
||||||
|
bool changingSubPage = (m_teletextDocument->currentSubPageIndex() != m_subPageIndex);
|
||||||
|
|
||||||
|
if (changingSubPage) {
|
||||||
m_teletextDocument->emit aboutToChangeSubPage();
|
m_teletextDocument->emit aboutToChangeSubPage();
|
||||||
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex);
|
||||||
|
} else
|
||||||
m_x26Model->beginRemoveRows(QModelIndex(), m_row, m_row+m_count-1);
|
m_x26Model->beginRemoveRows(QModelIndex(), m_row, m_row+m_count-1);
|
||||||
|
|
||||||
for (int i=0; i<m_count; i++)
|
for (int i=0; i<m_count; i++)
|
||||||
m_teletextDocument->currentSubPage()->enhancements()->removeAt(m_row);
|
m_teletextDocument->currentSubPage()->enhancements()->removeAt(m_row);
|
||||||
|
|
||||||
|
if (!changingSubPage)
|
||||||
m_x26Model->endRemoveRows();
|
m_x26Model->endRemoveRows();
|
||||||
|
|
||||||
|
// Preserve pointers to local object definitions that have moved
|
||||||
|
for (int i=0; i<m_teletextDocument->currentSubPage()->enhancements()->size(); i++) {
|
||||||
|
X26Triplet triplet = m_teletextDocument->currentSubPage()->enhancements()->at(i);
|
||||||
|
|
||||||
|
if (triplet.modeExt() >= 0x11 && triplet.modeExt() <= 0x13 && ((triplet.address() & 0x18) == 0x08) && triplet.objectLocalIndex() >= m_row) {
|
||||||
|
triplet.setObjectLocalIndex(triplet.objectLocalIndex() - m_count);
|
||||||
|
m_teletextDocument->currentSubPage()->enhancements()->replace(i, triplet);
|
||||||
|
m_x26Model->emit dataChanged(m_x26Model->createIndex(i, 0), m_x26Model->createIndex(i, 3));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (changingSubPage)
|
||||||
m_teletextDocument->emit subPageSelected();
|
m_teletextDocument->emit subPageSelected();
|
||||||
|
else
|
||||||
|
m_teletextDocument->emit contentsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeleteTripletCommand::undo()
|
void DeleteTripletCommand::undo()
|
||||||
@@ -125,12 +171,25 @@ void DeleteTripletCommand::undo()
|
|||||||
for (int i=0; i<m_count; i++)
|
for (int i=0; i<m_count; i++)
|
||||||
m_teletextDocument->currentSubPage()->enhancements()->insert(m_row+i, m_deletedTriplet);
|
m_teletextDocument->currentSubPage()->enhancements()->insert(m_row+i, m_deletedTriplet);
|
||||||
|
|
||||||
|
if (!changingSubPage)
|
||||||
|
m_x26Model->endInsertRows();
|
||||||
|
|
||||||
|
// Preserve pointers to local object definitions that have moved
|
||||||
|
for (int i=0; i<m_teletextDocument->currentSubPage()->enhancements()->size(); i++) {
|
||||||
|
X26Triplet triplet = m_teletextDocument->currentSubPage()->enhancements()->at(i);
|
||||||
|
|
||||||
|
if (triplet.modeExt() >= 0x11 && triplet.modeExt() <= 0x13 && ((triplet.address() & 0x18) == 0x08) && triplet.objectLocalIndex() >= m_row) {
|
||||||
|
triplet.setObjectLocalIndex(triplet.objectLocalIndex() + m_count);
|
||||||
|
m_teletextDocument->currentSubPage()->enhancements()->replace(i, triplet);
|
||||||
|
if (!changingSubPage)
|
||||||
|
m_x26Model->emit dataChanged(m_x26Model->createIndex(i, 0), m_x26Model->createIndex(i, 3));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (changingSubPage)
|
if (changingSubPage)
|
||||||
m_teletextDocument->emit subPageSelected();
|
m_teletextDocument->emit subPageSelected();
|
||||||
else {
|
else
|
||||||
m_x26Model->endInsertRows();
|
m_teletextDocument->emit contentsChanged();
|
||||||
m_teletextDocument->emit refreshNeeded();
|
|
||||||
}
|
|
||||||
|
|
||||||
m_teletextDocument->emit tripletCommandHighlight(m_row);
|
m_teletextDocument->emit tripletCommandHighlight(m_row);
|
||||||
}
|
}
|
||||||
@@ -164,9 +223,9 @@ void EditTripletCommand::redo()
|
|||||||
if (m_teletextDocument->currentSubPageIndex() != m_subPageIndex)
|
if (m_teletextDocument->currentSubPageIndex() != m_subPageIndex)
|
||||||
m_teletextDocument->selectSubPageIndex(m_subPageIndex, true);
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex, true);
|
||||||
|
|
||||||
m_teletextDocument->currentSubPage()->enhancements()->operator[](m_row) = m_newTriplet;
|
m_teletextDocument->currentSubPage()->enhancements()->replace(m_row, m_newTriplet);
|
||||||
m_x26Model->emit dataChanged(m_x26Model->createIndex(m_row, 0), m_x26Model->createIndex(m_row, 3), {m_role});
|
m_x26Model->emit dataChanged(m_x26Model->createIndex(m_row, 0), m_x26Model->createIndex(m_row, 3), {m_role});
|
||||||
m_teletextDocument->emit refreshNeeded();
|
m_teletextDocument->emit contentsChanged();
|
||||||
|
|
||||||
if (m_firstDo)
|
if (m_firstDo)
|
||||||
m_firstDo = false;
|
m_firstDo = false;
|
||||||
@@ -179,9 +238,9 @@ void EditTripletCommand::undo()
|
|||||||
if (m_teletextDocument->currentSubPageIndex() != m_subPageIndex)
|
if (m_teletextDocument->currentSubPageIndex() != m_subPageIndex)
|
||||||
m_teletextDocument->selectSubPageIndex(m_subPageIndex, true);
|
m_teletextDocument->selectSubPageIndex(m_subPageIndex, true);
|
||||||
|
|
||||||
m_teletextDocument->currentSubPage()->enhancements()->operator[](m_row) = m_oldTriplet;
|
m_teletextDocument->currentSubPage()->enhancements()->replace(m_row, m_oldTriplet);
|
||||||
m_x26Model->emit dataChanged(m_x26Model->createIndex(m_row, 0), m_x26Model->createIndex(m_row, 3), {m_role});
|
m_x26Model->emit dataChanged(m_x26Model->createIndex(m_row, 0), m_x26Model->createIndex(m_row, 3), {m_role});
|
||||||
m_teletextDocument->emit refreshNeeded();
|
m_teletextDocument->emit contentsChanged();
|
||||||
m_teletextDocument->emit tripletCommandHighlight(m_row);
|
m_teletextDocument->emit tripletCommandHighlight(m_row);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -29,7 +29,7 @@
|
|||||||
class InsertTripletCommand : public QUndoCommand
|
class InsertTripletCommand : public QUndoCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
InsertTripletCommand(TeletextDocument *, X26Model *, int, int, const X26Triplet, QUndoCommand *parent = 0);
|
InsertTripletCommand(TeletextDocument *teletextDocument, X26Model *x26Model, int row, int count, const X26Triplet newTriplet, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
@@ -45,7 +45,7 @@ private:
|
|||||||
class DeleteTripletCommand : public QUndoCommand
|
class DeleteTripletCommand : public QUndoCommand
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
DeleteTripletCommand(TeletextDocument *, X26Model *, int, int, QUndoCommand *parent = 0);
|
DeleteTripletCommand(TeletextDocument *teletextDocument, X26Model *x26Model, int row, int count, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
@@ -63,7 +63,7 @@ public:
|
|||||||
enum { Id = 201 };
|
enum { Id = 201 };
|
||||||
enum EditTripletEnum { ETaddress, ETmode, ETdata };
|
enum EditTripletEnum { ETaddress, ETmode, ETdata };
|
||||||
|
|
||||||
EditTripletCommand(TeletextDocument *, X26Model *, int, int, int, int, int, QUndoCommand *parent = 0);
|
EditTripletCommand(TeletextDocument *teletextDocument, X26Model *x26Model, int row, int tripletPart, int bitsToKeep, int newValue, int role, QUndoCommand *parent = 0);
|
||||||
|
|
||||||
void redo() override;
|
void redo() override;
|
||||||
void undo() override;
|
void undo() override;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -17,6 +17,7 @@
|
|||||||
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
* along with QTeletextMaker. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
#include <QActionGroup>
|
#include <QActionGroup>
|
||||||
#include <QButtonGroup>
|
#include <QButtonGroup>
|
||||||
#include <QCheckBox>
|
#include <QCheckBox>
|
||||||
@@ -34,8 +35,42 @@
|
|||||||
#include <QToolButton>
|
#include <QToolButton>
|
||||||
#include <QVBoxLayout>
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
|
#include "render.h"
|
||||||
#include "x26dockwidget.h"
|
#include "x26dockwidget.h"
|
||||||
|
|
||||||
|
CharacterListModel::CharacterListModel(QObject *parent): QAbstractListModel(parent)
|
||||||
|
{
|
||||||
|
m_characterSet = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int CharacterListModel::rowCount(const QModelIndex & /*parent*/) const
|
||||||
|
{
|
||||||
|
return 96;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant CharacterListModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (!index.isValid())
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
if (role == Qt::DisplayRole)
|
||||||
|
return QString("0x%1").arg(index.row()+0x20, 2, 16);
|
||||||
|
|
||||||
|
if (role == Qt::DecorationRole)
|
||||||
|
return m_fontBitmap.rawBitmap()->copy(index.row()*12, m_characterSet*10, 12, 10);
|
||||||
|
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CharacterListModel::setCharacterSet(int characterSet)
|
||||||
|
{
|
||||||
|
if (characterSet != m_characterSet) {
|
||||||
|
m_characterSet = characterSet;
|
||||||
|
emit dataChanged(createIndex(0, 0), createIndex(95, 0), QVector<int>(Qt::DecorationRole));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
||||||
{
|
{
|
||||||
QVBoxLayout *x26Layout = new QVBoxLayout;
|
QVBoxLayout *x26Layout = new QVBoxLayout;
|
||||||
@@ -70,16 +105,6 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
// "Cooked" or user-friendly triplet type, row and column selection
|
// "Cooked" or user-friendly triplet type, row and column selection
|
||||||
QHBoxLayout *cookedTripletLayout = new QHBoxLayout;
|
QHBoxLayout *cookedTripletLayout = new QHBoxLayout;
|
||||||
|
|
||||||
m_cookedModeTypeComboBox = new QComboBox;
|
|
||||||
m_cookedModeTypeComboBox->addItem("Set Active Position");
|
|
||||||
m_cookedModeTypeComboBox->addItem("Row triplet");
|
|
||||||
m_cookedModeTypeComboBox->addItem("Column triplet");
|
|
||||||
m_cookedModeTypeComboBox->addItem("Object");
|
|
||||||
m_cookedModeTypeComboBox->addItem("Terminator");
|
|
||||||
m_cookedModeTypeComboBox->addItem("PDC/reserved");
|
|
||||||
cookedTripletLayout->addWidget(m_cookedModeTypeComboBox);
|
|
||||||
connect(m_cookedModeTypeComboBox, QOverload<int>::of(&QComboBox::activated), this, &X26DockWidget::updateCookedModeFromCookedType);
|
|
||||||
|
|
||||||
// Cooked row spinbox
|
// Cooked row spinbox
|
||||||
QLabel *rowLabel = new QLabel(tr("Row"));
|
QLabel *rowLabel = new QLabel(tr("Row"));
|
||||||
rowLabel->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Preferred);
|
rowLabel->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Preferred);
|
||||||
@@ -102,10 +127,101 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
connect(m_cookedColumnSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), this, &X26DockWidget::cookedColumnSpinBoxChanged);
|
connect(m_cookedColumnSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), this, &X26DockWidget::cookedColumnSpinBoxChanged);
|
||||||
|
|
||||||
// Cooked triplet mode
|
// Cooked triplet mode
|
||||||
m_cookedModeComboBox = new QComboBox;
|
QLabel *modeLabel = new QLabel(tr("Mode"));
|
||||||
m_cookedModeComboBox->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
|
modeLabel->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Preferred);
|
||||||
cookedTripletLayout->addWidget(m_cookedModeComboBox);
|
cookedTripletLayout->addWidget(modeLabel);
|
||||||
connect(m_cookedModeComboBox, QOverload<int>::of(&QComboBox::activated), this, &X26DockWidget::cookedModeComboBoxChanged);
|
m_cookedModePushButton = new QPushButton;
|
||||||
|
cookedTripletLayout->addWidget(m_cookedModePushButton);
|
||||||
|
|
||||||
|
// Cooked triplet menu
|
||||||
|
// We build the menus for both "Insert" buttons at the same time
|
||||||
|
m_cookedModeMenu = new QMenu(this);
|
||||||
|
m_insertBeforeMenu = new QMenu(this);
|
||||||
|
m_insertAfterMenu = new QMenu(this);
|
||||||
|
|
||||||
|
for (int m=0; m<3; m++) {
|
||||||
|
QMenu *menuToBuild;
|
||||||
|
|
||||||
|
if (m == 0)
|
||||||
|
menuToBuild = m_cookedModeMenu;
|
||||||
|
else if (m == 1)
|
||||||
|
menuToBuild = m_insertBeforeMenu;
|
||||||
|
else // if (m == 2)
|
||||||
|
menuToBuild = m_insertAfterMenu;
|
||||||
|
|
||||||
|
auto newModeMenuAction=[&](QMenu *menu, int mode)
|
||||||
|
{
|
||||||
|
QAction *action = menu->addAction(m_x26Model->modeTripletName(mode));
|
||||||
|
if (m == 0)
|
||||||
|
connect(action, &QAction::triggered, [=]() { cookedModeMenuSelected(mode); });
|
||||||
|
else if (m == 1)
|
||||||
|
connect(action, &QAction::triggered, [=]() { insertTriplet(mode, false); });
|
||||||
|
else // if (m == 2)
|
||||||
|
connect(action, &QAction::triggered, [=]() { insertTriplet(mode, true); });
|
||||||
|
};
|
||||||
|
|
||||||
|
newModeMenuAction(menuToBuild, 0x04);
|
||||||
|
QMenu *rowTripletSubMenu = menuToBuild->addMenu(tr("Row triplet"));
|
||||||
|
newModeMenuAction(rowTripletSubMenu, 0x00);
|
||||||
|
newModeMenuAction(rowTripletSubMenu, 0x01);
|
||||||
|
newModeMenuAction(rowTripletSubMenu, 0x07);
|
||||||
|
newModeMenuAction(rowTripletSubMenu, 0x18);
|
||||||
|
QMenu *columnTripletSubMenu = menuToBuild->addMenu(tr("Column triplet"));
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x20);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x23);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x27);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x2c);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x2e);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x28);
|
||||||
|
columnTripletSubMenu->addSeparator();
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x29);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x2f);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x21);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x22);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x2b);
|
||||||
|
newModeMenuAction(columnTripletSubMenu, 0x2d);
|
||||||
|
QMenu *diacriticalSubMenu = columnTripletSubMenu->addMenu(tr("G0 diacritical"));
|
||||||
|
for (int i=0; i<16; i++)
|
||||||
|
newModeMenuAction(diacriticalSubMenu, 0x30 + i);
|
||||||
|
QMenu *objectSubMenu = menuToBuild->addMenu(tr("Object"));
|
||||||
|
newModeMenuAction(objectSubMenu, 0x10);
|
||||||
|
newModeMenuAction(objectSubMenu, 0x11);
|
||||||
|
newModeMenuAction(objectSubMenu, 0x12);
|
||||||
|
newModeMenuAction(objectSubMenu, 0x13);
|
||||||
|
newModeMenuAction(objectSubMenu, 0x15);
|
||||||
|
newModeMenuAction(objectSubMenu, 0x16);
|
||||||
|
newModeMenuAction(objectSubMenu, 0x17);
|
||||||
|
newModeMenuAction(menuToBuild, 0x1f);
|
||||||
|
menuToBuild->addSeparator();
|
||||||
|
QMenu *pdcSubMenu = menuToBuild->addMenu(tr("PDC/reserved"));
|
||||||
|
newModeMenuAction(pdcSubMenu, 0x08);
|
||||||
|
newModeMenuAction(pdcSubMenu, 0x09);
|
||||||
|
newModeMenuAction(pdcSubMenu, 0x0a);
|
||||||
|
newModeMenuAction(pdcSubMenu, 0x0b);
|
||||||
|
newModeMenuAction(pdcSubMenu, 0x0c);
|
||||||
|
newModeMenuAction(pdcSubMenu, 0x0d);
|
||||||
|
newModeMenuAction(pdcSubMenu, 0x26);
|
||||||
|
QMenu *reservedRowSubMenu = pdcSubMenu->addMenu(tr("Reserved row"));
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x02);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x03);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x05);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x06);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x0e);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x0f);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x14);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x19);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x1a);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x1b);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x1c);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x1d);
|
||||||
|
newModeMenuAction(reservedRowSubMenu, 0x1e);
|
||||||
|
QMenu *reservedColumnSubMenu = pdcSubMenu->addMenu(tr("Reserved column"));
|
||||||
|
newModeMenuAction(reservedColumnSubMenu, 0x24);
|
||||||
|
newModeMenuAction(reservedColumnSubMenu, 0x25);
|
||||||
|
newModeMenuAction(reservedColumnSubMenu, 0x2a);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_cookedModePushButton->setMenu(m_cookedModeMenu);
|
||||||
|
|
||||||
// Raw triplet values
|
// Raw triplet values
|
||||||
QHBoxLayout *rawTripletLayout = new QHBoxLayout;
|
QHBoxLayout *rawTripletLayout = new QHBoxLayout;
|
||||||
@@ -161,9 +277,7 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
QHBoxLayout *colourAndRowLayout = new QHBoxLayout;
|
QHBoxLayout *colourAndRowLayout = new QHBoxLayout;
|
||||||
|
|
||||||
m_colourComboBox = new QComboBox;
|
m_colourComboBox = new QComboBox;
|
||||||
for (int c=0; c<=3; c++)
|
m_colourComboBox->setModel(m_parentMainWidget->document()->clutModel());
|
||||||
for (int e=0; e<=7; e++)
|
|
||||||
m_colourComboBox->addItem(tr("CLUT %1:%2").arg(c).arg(e));
|
|
||||||
colourAndRowLayout->addWidget(m_colourComboBox);
|
colourAndRowLayout->addWidget(m_colourComboBox);
|
||||||
connect(m_colourComboBox, QOverload<int>::of(&QComboBox::activated), this, [=](const int value) { updateModelFromCookedWidget(value, Qt::UserRole+1); } );
|
connect(m_colourComboBox, QOverload<int>::of(&QComboBox::activated), this, [=](const int value) { updateModelFromCookedWidget(value, Qt::UserRole+1); } );
|
||||||
|
|
||||||
@@ -181,8 +295,7 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
QHBoxLayout *characterCodeLayout = new QHBoxLayout;
|
QHBoxLayout *characterCodeLayout = new QHBoxLayout;
|
||||||
|
|
||||||
m_characterCodeComboBox = new QComboBox;
|
m_characterCodeComboBox = new QComboBox;
|
||||||
for (int i=32; i<128; i++)
|
m_characterCodeComboBox->setModel(&m_characterListModel);
|
||||||
m_characterCodeComboBox->addItem(QString("0x%1").arg(i, 2, 16), i);
|
|
||||||
characterCodeLayout->addWidget(m_characterCodeComboBox);
|
characterCodeLayout->addWidget(m_characterCodeComboBox);
|
||||||
connect(m_characterCodeComboBox, QOverload<int>::of(&QComboBox::activated), this, [=](const int value) { updateModelFromCookedWidget(value+32, Qt::UserRole+1); } );
|
connect(m_characterCodeComboBox, QOverload<int>::of(&QComboBox::activated), this, [=](const int value) { updateModelFromCookedWidget(value+32, Qt::UserRole+1); } );
|
||||||
|
|
||||||
@@ -240,7 +353,7 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
m_objectSourceComboBox->addItem("POP");
|
m_objectSourceComboBox->addItem("POP");
|
||||||
m_objectSourceComboBox->addItem("GPOP");
|
m_objectSourceComboBox->addItem("GPOP");
|
||||||
invokeObjectLayout->addWidget(m_objectSourceComboBox);
|
invokeObjectLayout->addWidget(m_objectSourceComboBox);
|
||||||
connect(m_objectSourceComboBox, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](const int value) { updateModelFromCookedWidget(value, Qt::UserRole+1); updateCookedTripletParameters(m_x26View->currentIndex()); } );
|
connect(m_objectSourceComboBox, QOverload<int>::of(&QComboBox::currentIndexChanged), [=](const int value) { updateModelFromCookedWidget(value, Qt::UserRole+1); updateAllCookedTripletWidgets(m_x26View->currentIndex()); } );
|
||||||
|
|
||||||
// Object required at which levels
|
// Object required at which levels
|
||||||
m_objectRequiredAtLevelsComboBox = new QComboBox;
|
m_objectRequiredAtLevelsComboBox = new QComboBox;
|
||||||
@@ -253,13 +366,15 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
// Invoke Local Objects
|
// Invoke Local Objects
|
||||||
QHBoxLayout *invokeLocalObjectLayout = new QHBoxLayout;
|
QHBoxLayout *invokeLocalObjectLayout = new QHBoxLayout;
|
||||||
|
|
||||||
invokeLocalObjectLayout->addWidget(new QLabel(tr("Designation")));
|
m_invokeLocalObjectDesignationCodeLabel = new QLabel(tr("Designation"));
|
||||||
|
invokeLocalObjectLayout->addWidget(m_invokeLocalObjectDesignationCodeLabel);
|
||||||
m_invokeLocalObjectDesignationCodeSpinBox = new QSpinBox;
|
m_invokeLocalObjectDesignationCodeSpinBox = new QSpinBox;
|
||||||
m_invokeLocalObjectDesignationCodeSpinBox->setMaximum(15);
|
m_invokeLocalObjectDesignationCodeSpinBox->setMaximum(15);
|
||||||
invokeLocalObjectLayout->addWidget(m_invokeLocalObjectDesignationCodeSpinBox);
|
invokeLocalObjectLayout->addWidget(m_invokeLocalObjectDesignationCodeSpinBox);
|
||||||
connect(m_invokeLocalObjectDesignationCodeSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), this, [=](const int value) { updateModelFromCookedWidget(value, Qt::UserRole+2); } );
|
connect(m_invokeLocalObjectDesignationCodeSpinBox, QOverload<int>::of(&QSpinBox::valueChanged), this, [=](const int value) { updateModelFromCookedWidget(value, Qt::UserRole+2); } );
|
||||||
|
|
||||||
invokeLocalObjectLayout->addWidget(new QLabel(tr("Triplet")));
|
m_invokeLocalObjectTripletNumberLabel = new QLabel(tr("Triplet"));
|
||||||
|
invokeLocalObjectLayout->addWidget(m_invokeLocalObjectTripletNumberLabel);
|
||||||
m_invokeLocalObjectTripletNumberSpinBox = new QSpinBox;
|
m_invokeLocalObjectTripletNumberSpinBox = new QSpinBox;
|
||||||
m_invokeLocalObjectTripletNumberSpinBox->setMaximum(12);
|
m_invokeLocalObjectTripletNumberSpinBox->setMaximum(12);
|
||||||
invokeLocalObjectLayout->addWidget(m_invokeLocalObjectTripletNumberSpinBox);
|
invokeLocalObjectLayout->addWidget(m_invokeLocalObjectTripletNumberSpinBox);
|
||||||
@@ -410,60 +525,70 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
m_tripletParameterStackedLayout->addWidget(blankWidget);
|
m_tripletParameterStackedLayout->addWidget(blankWidget);
|
||||||
|
|
||||||
// Index 1
|
// Index 1
|
||||||
|
colourAndRowLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *colourAndRowWidget = new QWidget;
|
QWidget *colourAndRowWidget = new QWidget;
|
||||||
colourAndRowWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
colourAndRowWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
colourAndRowWidget->setLayout(colourAndRowLayout);
|
colourAndRowWidget->setLayout(colourAndRowLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(colourAndRowWidget);
|
m_tripletParameterStackedLayout->addWidget(colourAndRowWidget);
|
||||||
|
|
||||||
// Index 2
|
// Index 2
|
||||||
|
characterCodeLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *characterCodeWidget = new QWidget;
|
QWidget *characterCodeWidget = new QWidget;
|
||||||
characterCodeWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
characterCodeWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
characterCodeWidget->setLayout(characterCodeLayout);
|
characterCodeWidget->setLayout(characterCodeLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(characterCodeWidget);
|
m_tripletParameterStackedLayout->addWidget(characterCodeWidget);
|
||||||
|
|
||||||
// Index 3
|
// Index 3
|
||||||
|
flashModeRateLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *flashModeRateWidget = new QWidget;
|
QWidget *flashModeRateWidget = new QWidget;
|
||||||
flashModeRateWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
flashModeRateWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
flashModeRateWidget->setLayout(flashModeRateLayout);
|
flashModeRateWidget->setLayout(flashModeRateLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(flashModeRateWidget);
|
m_tripletParameterStackedLayout->addWidget(flashModeRateWidget);
|
||||||
|
|
||||||
// Index 4
|
// Index 4
|
||||||
|
displayAttributesLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *displayAttributesWidget = new QWidget;
|
QWidget *displayAttributesWidget = new QWidget;
|
||||||
displayAttributesWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
displayAttributesWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
displayAttributesWidget->setLayout(displayAttributesLayout);
|
displayAttributesWidget->setLayout(displayAttributesLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(displayAttributesWidget);
|
m_tripletParameterStackedLayout->addWidget(displayAttributesWidget);
|
||||||
|
|
||||||
// Index 5
|
// Index 5
|
||||||
|
invokeObjectLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *invokeObjectWidget = new QWidget;
|
QWidget *invokeObjectWidget = new QWidget;
|
||||||
invokeObjectWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
invokeObjectWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
invokeObjectWidget->setLayout(invokeObjectLayout);
|
invokeObjectWidget->setLayout(invokeObjectLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(invokeObjectWidget);
|
m_tripletParameterStackedLayout->addWidget(invokeObjectWidget);
|
||||||
|
|
||||||
// Index 6
|
// Index 6
|
||||||
|
DRCSModeLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *DRCSModeWidget = new QWidget;
|
QWidget *DRCSModeWidget = new QWidget;
|
||||||
DRCSModeWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
DRCSModeWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
DRCSModeWidget->setLayout(DRCSModeLayout);
|
DRCSModeWidget->setLayout(DRCSModeLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(DRCSModeWidget);
|
m_tripletParameterStackedLayout->addWidget(DRCSModeWidget);
|
||||||
|
|
||||||
// Index 7
|
// Index 7
|
||||||
|
DRCSCharacterLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *DRCSCharacterWidget = new QWidget;
|
QWidget *DRCSCharacterWidget = new QWidget;
|
||||||
DRCSCharacterWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
DRCSCharacterWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
DRCSCharacterWidget->setLayout(DRCSCharacterLayout);
|
DRCSCharacterWidget->setLayout(DRCSCharacterLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(DRCSCharacterWidget);
|
m_tripletParameterStackedLayout->addWidget(DRCSCharacterWidget);
|
||||||
|
|
||||||
// Index 8
|
// Index 8
|
||||||
|
fontStyleLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *fontStyleWidget = new QWidget;
|
QWidget *fontStyleWidget = new QWidget;
|
||||||
fontStyleWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
fontStyleWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
fontStyleWidget->setLayout(fontStyleLayout);
|
fontStyleWidget->setLayout(fontStyleLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(fontStyleWidget);
|
m_tripletParameterStackedLayout->addWidget(fontStyleWidget);
|
||||||
|
|
||||||
// Index 9
|
// Index 9
|
||||||
|
reservedPDCLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *reservedPDCWidget = new QWidget;
|
QWidget *reservedPDCWidget = new QWidget;
|
||||||
reservedPDCWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
reservedPDCWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
reservedPDCWidget->setLayout(reservedPDCLayout);
|
reservedPDCWidget->setLayout(reservedPDCLayout);
|
||||||
m_tripletParameterStackedLayout->addWidget(reservedPDCWidget);
|
m_tripletParameterStackedLayout->addWidget(reservedPDCWidget);
|
||||||
|
|
||||||
// Index 10
|
// Index 10
|
||||||
|
terminationMarkerLayout->setContentsMargins(0, 0, 0, 0);
|
||||||
QWidget *terminationMarkerWidget = new QWidget;
|
QWidget *terminationMarkerWidget = new QWidget;
|
||||||
terminationMarkerWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
terminationMarkerWidget->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||||||
terminationMarkerWidget->setLayout(terminationMarkerLayout);
|
terminationMarkerWidget->setLayout(terminationMarkerLayout);
|
||||||
@@ -479,30 +604,41 @@ X26DockWidget::X26DockWidget(TeletextWidget *parent): QDockWidget(parent)
|
|||||||
// Insert and delete widgets
|
// Insert and delete widgets
|
||||||
QHBoxLayout *insertDeleteLayout = new QHBoxLayout;
|
QHBoxLayout *insertDeleteLayout = new QHBoxLayout;
|
||||||
|
|
||||||
m_insertPushButton = new QPushButton(tr("Insert triplet"));
|
m_insertBeforePushButton = new QPushButton(tr("Insert before"));
|
||||||
insertDeleteLayout->addWidget(m_insertPushButton);
|
insertDeleteLayout->addWidget(m_insertBeforePushButton);
|
||||||
m_deletePushButton = new QPushButton(tr("Delete triplet"));
|
m_insertBeforePushButton->setMenu(m_insertBeforeMenu);
|
||||||
|
|
||||||
|
m_insertAfterPushButton = new QPushButton(tr("Insert after"));
|
||||||
|
insertDeleteLayout->addWidget(m_insertAfterPushButton);
|
||||||
|
m_insertAfterPushButton->setMenu(m_insertAfterMenu);
|
||||||
|
|
||||||
|
m_insertCopyPushButton = new QPushButton(tr("Insert copy"));
|
||||||
|
insertDeleteLayout->addWidget(m_insertCopyPushButton);
|
||||||
|
|
||||||
|
m_deletePushButton = new QPushButton(tr("Delete"));
|
||||||
insertDeleteLayout->addWidget(m_deletePushButton);
|
insertDeleteLayout->addWidget(m_deletePushButton);
|
||||||
|
|
||||||
connect(m_insertPushButton, &QPushButton::clicked, this, &X26DockWidget::insertTriplet);
|
connect(m_insertCopyPushButton, &QPushButton::clicked, this, &X26DockWidget::insertTripletCopy);
|
||||||
connect(m_deletePushButton, &QPushButton::clicked, this, &X26DockWidget::deleteTriplet);
|
connect(m_deletePushButton, &QPushButton::clicked, this, &X26DockWidget::deleteTriplet);
|
||||||
|
|
||||||
x26Layout->addLayout(insertDeleteLayout);
|
x26Layout->addLayout(insertDeleteLayout);
|
||||||
|
|
||||||
|
disableTripletWidgets();
|
||||||
|
|
||||||
x26Widget->setLayout(x26Layout);
|
x26Widget->setLayout(x26Layout);
|
||||||
this->setWidget(x26Widget);
|
this->setWidget(x26Widget);
|
||||||
|
|
||||||
m_x26View->setContextMenuPolicy(Qt::CustomContextMenu);
|
m_x26View->setContextMenuPolicy(Qt::CustomContextMenu);
|
||||||
connect(m_x26View, &QWidget::customContextMenuRequested, this, &X26DockWidget::customMenuRequested);
|
connect(m_x26View, &QWidget::customContextMenuRequested, this, &X26DockWidget::customMenuRequested);
|
||||||
|
|
||||||
connect(m_x26View, &QAbstractItemView::clicked, this, &X26DockWidget::rowClicked);
|
connect(m_x26View->selectionModel(), &QItemSelectionModel::currentRowChanged, this, &X26DockWidget::rowSelected);
|
||||||
}
|
}
|
||||||
|
|
||||||
void X26DockWidget::keyPressEvent(QKeyEvent *event)
|
void X26DockWidget::keyPressEvent(QKeyEvent *event)
|
||||||
{
|
{
|
||||||
switch (event->key()) {
|
switch (event->key()) {
|
||||||
case Qt::Key_Insert:
|
case Qt::Key_Insert:
|
||||||
insertTriplet();
|
insertTripletCopy();
|
||||||
break;
|
break;
|
||||||
case Qt::Key_Delete:
|
case Qt::Key_Delete:
|
||||||
deleteTriplet();
|
deleteTriplet();
|
||||||
@@ -520,7 +656,7 @@ void X26DockWidget::selectX26ListRow(int row)
|
|||||||
row = m_x26Model->rowCount() - 1;
|
row = m_x26Model->rowCount() - 1;
|
||||||
|
|
||||||
m_x26View->selectRow(row);
|
m_x26View->selectRow(row);
|
||||||
rowClicked(m_x26View->currentIndex());
|
rowSelected(m_x26View->currentIndex(), QModelIndex());
|
||||||
}
|
}
|
||||||
|
|
||||||
void X26DockWidget::loadX26List()
|
void X26DockWidget::loadX26List()
|
||||||
@@ -536,145 +672,53 @@ void X26DockWidget::unloadX26List()
|
|||||||
m_rawTripletModeSpinBox->setEnabled(false);
|
m_rawTripletModeSpinBox->setEnabled(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void X26DockWidget::rowClicked(const QModelIndex &index)
|
void X26DockWidget::rowSelected(const QModelIndex ¤t, const QModelIndex &previous)
|
||||||
{
|
{
|
||||||
updateAllRawTripletSpinBoxes(index);
|
Q_UNUSED(previous);
|
||||||
updateAllCookedTripletWidgets(index);
|
|
||||||
|
if (current.isValid()) {
|
||||||
|
updateAllRawTripletSpinBoxes(current);
|
||||||
|
updateAllCookedTripletWidgets(current);
|
||||||
|
} else
|
||||||
|
disableTripletWidgets();
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26DockWidget::disableTripletWidgets()
|
||||||
|
{
|
||||||
|
m_rawTripletAddressSpinBox->setEnabled(false);
|
||||||
|
m_rawTripletDataSpinBox->setEnabled(false);
|
||||||
|
m_rawTripletModeSpinBox->setEnabled(false);
|
||||||
|
m_rawTripletAddressSpinBox->blockSignals(true);
|
||||||
|
m_rawTripletModeSpinBox->blockSignals(true);
|
||||||
|
m_rawTripletDataSpinBox->blockSignals(true);
|
||||||
|
m_rawTripletAddressSpinBox->setValue(0);
|
||||||
|
m_rawTripletModeSpinBox->setValue(0);
|
||||||
|
m_rawTripletDataSpinBox->setValue(0);
|
||||||
|
m_rawTripletAddressSpinBox->blockSignals(false);
|
||||||
|
m_rawTripletModeSpinBox->blockSignals(false);
|
||||||
|
m_rawTripletDataSpinBox->blockSignals(false);
|
||||||
|
|
||||||
|
m_cookedRowSpinBox->setEnabled(false);
|
||||||
|
m_cookedColumnSpinBox->setEnabled(false);
|
||||||
|
m_cookedRowSpinBox->blockSignals(true);
|
||||||
|
m_cookedColumnSpinBox->blockSignals(true);
|
||||||
|
m_cookedRowSpinBox->setValue(1);
|
||||||
|
m_cookedColumnSpinBox->setValue(0);
|
||||||
|
m_cookedRowSpinBox->blockSignals(false);
|
||||||
|
m_cookedColumnSpinBox->blockSignals(false);
|
||||||
|
|
||||||
|
m_cookedModePushButton->setEnabled(false);
|
||||||
|
m_cookedModePushButton->setText(QString());
|
||||||
|
|
||||||
|
m_tripletParameterStackedLayout->setCurrentIndex(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void X26DockWidget::updateAllCookedTripletWidgets(const QModelIndex &index)
|
void X26DockWidget::updateAllCookedTripletWidgets(const QModelIndex &index)
|
||||||
{
|
{
|
||||||
const int modeExt = index.model()->data(index.model()->index(index.row(), 2), Qt::EditRole).toInt();
|
const int modeExt = index.model()->data(index.model()->index(index.row(), 2), Qt::EditRole).toInt();
|
||||||
|
|
||||||
// Find which triplettype the triplet is
|
m_cookedModePushButton->setEnabled(true);
|
||||||
const int oldCookedModeType = m_cookedModeTypeComboBox->currentIndex();
|
m_cookedModePushButton->setText(m_x26Model->modeTripletName(modeExt));
|
||||||
switch (modeExt) {
|
|
||||||
case 0x04:
|
|
||||||
m_cookedModeTypeComboBox->setCurrentIndex(0);
|
|
||||||
break;
|
|
||||||
case 0x00:
|
|
||||||
case 0x01:
|
|
||||||
case 0x07:
|
|
||||||
case 0x18:
|
|
||||||
m_cookedModeTypeComboBox->setCurrentIndex(1);
|
|
||||||
break;
|
|
||||||
case 0x20 ... 0x23:
|
|
||||||
case 0x27 ... 0x29:
|
|
||||||
case 0x2b ... 0x3f:
|
|
||||||
m_cookedModeTypeComboBox->setCurrentIndex(2);
|
|
||||||
break;
|
|
||||||
case 0x10 ... 0x13:
|
|
||||||
case 0x15 ... 0x17:
|
|
||||||
m_cookedModeTypeComboBox->setCurrentIndex(3);
|
|
||||||
break;
|
|
||||||
case 0x1f:
|
|
||||||
m_cookedModeTypeComboBox->setCurrentIndex(4);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
m_cookedModeTypeComboBox->setCurrentIndex(5);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the triplettype has changed, update the triplet mode combobox
|
|
||||||
if (oldCookedModeType != m_cookedModeTypeComboBox->currentIndex())
|
|
||||||
updateCookedModeFromCookedType(-1);
|
|
||||||
for (int i=0; i<m_cookedModeComboBox->count(); i++)
|
|
||||||
if (m_cookedModeComboBox->itemData(i) == modeExt) {
|
|
||||||
m_cookedModeComboBox->blockSignals(true);
|
|
||||||
m_cookedModeComboBox->setCurrentIndex(i);
|
|
||||||
m_cookedModeComboBox->blockSignals(false);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
updateCookedTripletParameters(index);
|
|
||||||
}
|
|
||||||
|
|
||||||
void X26DockWidget::updateCookedModeFromCookedType(const int value)
|
|
||||||
{
|
|
||||||
while (m_cookedModeComboBox->count() > 0)
|
|
||||||
m_cookedModeComboBox->removeItem(0);
|
|
||||||
// When called as a slot, "value" parameter would be the same as this currentIndex
|
|
||||||
switch (m_cookedModeTypeComboBox->currentIndex()) {
|
|
||||||
case 1:
|
|
||||||
m_cookedModeComboBox->addItem("select...", -1);
|
|
||||||
m_cookedModeComboBox->addItem("Full screen colour", 0x00);
|
|
||||||
m_cookedModeComboBox->addItem("Full row colour", 0x01);
|
|
||||||
m_cookedModeComboBox->addItem("Address row 0", 0x07);
|
|
||||||
m_cookedModeComboBox->addItem("DRCS mode", 0x18);
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
m_cookedModeComboBox->addItem("select...", -1);
|
|
||||||
m_cookedModeComboBox->addItem("Foreground colour", 0x20);
|
|
||||||
m_cookedModeComboBox->addItem("Background colour", 0x23);
|
|
||||||
m_cookedModeComboBox->addItem("Flash functions", 0x27);
|
|
||||||
m_cookedModeComboBox->addItem("Display attrs", 0x2c);
|
|
||||||
m_cookedModeComboBox->addItem("Font style L 3.5", 0x2e);
|
|
||||||
m_cookedModeComboBox->addItem("Mod G0 and G2", 0x28);
|
|
||||||
m_cookedModeComboBox->addItem("G0 character", 0x29);
|
|
||||||
m_cookedModeComboBox->addItem("G2 character", 0x2f);
|
|
||||||
m_cookedModeComboBox->addItem("G1 block mosaic", 0x21);
|
|
||||||
m_cookedModeComboBox->addItem("G3 at L 1.5", 0x22);
|
|
||||||
m_cookedModeComboBox->addItem("G3 at L 2.5", 0x2b);
|
|
||||||
m_cookedModeComboBox->addItem("DRCS character", 0x2d);
|
|
||||||
for (int i=0; i<16; i++)
|
|
||||||
m_cookedModeComboBox->addItem(QString("G0 diactricial ")+QString("%1").arg(i, 1, 16).toUpper(), 0x30 | i);
|
|
||||||
break;
|
|
||||||
case 3:
|
|
||||||
m_cookedModeComboBox->addItem("select...", -1);
|
|
||||||
m_cookedModeComboBox->addItem("Origin modifier", 0x10);
|
|
||||||
m_cookedModeComboBox->addItem("Invoke active obj", 0x11);
|
|
||||||
m_cookedModeComboBox->addItem("Invoke adaptive obj", 0x12);
|
|
||||||
m_cookedModeComboBox->addItem("Invoke passive obj", 0x13);
|
|
||||||
m_cookedModeComboBox->addItem("Define active obj", 0x15);
|
|
||||||
m_cookedModeComboBox->addItem("Define adaptive obj", 0x16);
|
|
||||||
m_cookedModeComboBox->addItem("Define passive obj", 0x17);
|
|
||||||
break;
|
|
||||||
case 5:
|
|
||||||
m_cookedModeComboBox->addItem("select...", -1);
|
|
||||||
m_cookedModeComboBox->addItem("Origin and Source", 0x08);
|
|
||||||
m_cookedModeComboBox->addItem("Month and day", 0x09);
|
|
||||||
m_cookedModeComboBox->addItem("Row + start hours", 0x0a);
|
|
||||||
m_cookedModeComboBox->addItem("Row + end hours", 0x0b);
|
|
||||||
m_cookedModeComboBox->addItem("Row + time offset", 0x0c);
|
|
||||||
m_cookedModeComboBox->addItem("Series ID and code", 0x0d);
|
|
||||||
m_cookedModeComboBox->addItem("Col + start/end mins", 0x26);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x02", 0x02);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x03", 0x03);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x05", 0x05);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x06", 0x06);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x0e", 0x0e);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x0f", 0x0f);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x14", 0x14);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x19", 0x19);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x1a", 0x1a);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x1b", 0x1b);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x1c", 0x1c);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x1d", 0x1d);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved row 0x1e", 0x1e);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved col 0x04", 0x04);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved col 0x05", 0x05);
|
|
||||||
m_cookedModeComboBox->addItem("Reserved col 0x0a", 0x0a);
|
|
||||||
break;
|
|
||||||
case 0:
|
|
||||||
// When called as a slot the user set the combobox themself, so set the triplet mode immediately
|
|
||||||
if (value != -1) {
|
|
||||||
m_x26Model->setData(m_x26Model->index(m_x26View->currentIndex().row(), 2), 4, Qt::EditRole);
|
|
||||||
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
|
||||||
updateCookedTripletParameters(m_x26View->currentIndex());
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
if (value != -1) {
|
|
||||||
m_x26Model->setData(m_x26Model->index(m_x26View->currentIndex().row(), 2), 31, Qt::EditRole);
|
|
||||||
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
|
||||||
updateCookedTripletParameters(m_x26View->currentIndex());
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void X26DockWidget::updateCookedTripletParameters(const QModelIndex &index)
|
|
||||||
{
|
|
||||||
const int modeExt = index.model()->data(index.model()->index(index.row(), 2), Qt::EditRole).toInt();
|
|
||||||
|
|
||||||
switch (modeExt) {
|
switch (modeExt) {
|
||||||
case 0x04: // Set active position
|
case 0x04: // Set active position
|
||||||
@@ -704,8 +748,30 @@ void X26DockWidget::updateCookedTripletParameters(const QModelIndex &index)
|
|||||||
case 0x22: // G3 character at Level 1.5
|
case 0x22: // G3 character at Level 1.5
|
||||||
case 0x29: // G0 character
|
case 0x29: // G0 character
|
||||||
case 0x2b: // G3 character at Level 2.5
|
case 0x2b: // G3 character at Level 2.5
|
||||||
case 0x2f ... 0x3f: // G2 character, G0 character with diacritical
|
case 0x2f: // G2 character
|
||||||
|
case 0x30:
|
||||||
|
case 0x31:
|
||||||
|
case 0x32:
|
||||||
|
case 0x33:
|
||||||
|
case 0x34:
|
||||||
|
case 0x35:
|
||||||
|
case 0x36:
|
||||||
|
case 0x37:
|
||||||
|
case 0x38:
|
||||||
|
case 0x39:
|
||||||
|
case 0x3a:
|
||||||
|
case 0x3b:
|
||||||
|
case 0x3c:
|
||||||
|
case 0x3d:
|
||||||
|
case 0x3e:
|
||||||
|
case 0x3f: // G0 character with diacritical
|
||||||
m_characterCodeComboBox->blockSignals(true);
|
m_characterCodeComboBox->blockSignals(true);
|
||||||
|
if (modeExt == 0x21)
|
||||||
|
m_characterListModel.setCharacterSet(24);
|
||||||
|
else if (modeExt == 0x22 || modeExt == 0x2b)
|
||||||
|
m_characterListModel.setCharacterSet(26);
|
||||||
|
else
|
||||||
|
m_characterListModel.setCharacterSet(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+2).toInt());
|
||||||
m_characterCodeComboBox->setCurrentIndex(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+1).toInt()-32);
|
m_characterCodeComboBox->setCurrentIndex(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+1).toInt()-32);
|
||||||
m_characterCodeComboBox->blockSignals(false);
|
m_characterCodeComboBox->blockSignals(false);
|
||||||
m_tripletParameterStackedLayout->setCurrentIndex(2);
|
m_tripletParameterStackedLayout->setCurrentIndex(2);
|
||||||
@@ -738,8 +804,12 @@ void X26DockWidget::updateCookedTripletParameters(const QModelIndex &index)
|
|||||||
m_displayAttributeUnderlineCheckBox->blockSignals(false);
|
m_displayAttributeUnderlineCheckBox->blockSignals(false);
|
||||||
m_tripletParameterStackedLayout->setCurrentIndex(4);
|
m_tripletParameterStackedLayout->setCurrentIndex(4);
|
||||||
break;
|
break;
|
||||||
case 0x11 ... 0x13: // Invoke object
|
case 0x11: // Invoke Active Object
|
||||||
case 0x15 ... 0x17: // Define object
|
case 0x12: // Invoke Adaptive Object
|
||||||
|
case 0x13: // Invoke Passive Object
|
||||||
|
case 0x15: // Define Active Object
|
||||||
|
case 0x16: // Define Adaptive Object
|
||||||
|
case 0x17: // Define Passive Object
|
||||||
if (index.model()->data(index.model()->index(index.row(), 1), Qt::UserRole).toInt() & 0x04) {
|
if (index.model()->data(index.model()->index(index.row(), 1), Qt::UserRole).toInt() & 0x04) {
|
||||||
// Define object
|
// Define object
|
||||||
m_objectSourceComboBox->setVisible(false);
|
m_objectSourceComboBox->setVisible(false);
|
||||||
@@ -756,14 +826,22 @@ void X26DockWidget::updateCookedTripletParameters(const QModelIndex &index)
|
|||||||
// BUG we're only dealing with Local Object Definitions at the moment!
|
// BUG we're only dealing with Local Object Definitions at the moment!
|
||||||
if (index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+1).toInt() == 0 || (index.model()->data(index.model()->index(index.row(), 1), Qt::UserRole).toInt() & 0x04)) {
|
if (index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+1).toInt() == 0 || (index.model()->data(index.model()->index(index.row(), 1), Qt::UserRole).toInt() & 0x04)) {
|
||||||
// if (triplet.objectSource() == X26Triplet::LocalObjectSource) {
|
// if (triplet.objectSource() == X26Triplet::LocalObjectSource) {
|
||||||
|
const bool tripletLocationWidgetsVisible = (modeExt & 0x04) != 0x04;
|
||||||
|
|
||||||
|
m_invokeLocalObjectDesignationCodeLabel->setVisible(tripletLocationWidgetsVisible);
|
||||||
|
m_invokeLocalObjectDesignationCodeSpinBox->setVisible(tripletLocationWidgetsVisible);
|
||||||
|
m_invokeLocalObjectTripletNumberLabel->setVisible(tripletLocationWidgetsVisible);
|
||||||
|
m_invokeLocalObjectTripletNumberSpinBox->setVisible(tripletLocationWidgetsVisible);
|
||||||
m_objectSourceComboBox->setCurrentIndex(0);
|
m_objectSourceComboBox->setCurrentIndex(0);
|
||||||
m_invokeObjectSourceStackedLayout->setCurrentIndex(0);
|
m_invokeObjectSourceStackedLayout->setCurrentIndex(0);
|
||||||
|
if (tripletLocationWidgetsVisible) {
|
||||||
m_invokeLocalObjectDesignationCodeSpinBox->blockSignals(true);
|
m_invokeLocalObjectDesignationCodeSpinBox->blockSignals(true);
|
||||||
m_invokeLocalObjectTripletNumberSpinBox->blockSignals(true);
|
m_invokeLocalObjectTripletNumberSpinBox->blockSignals(true);
|
||||||
m_invokeLocalObjectDesignationCodeSpinBox->setValue(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+2).toInt());
|
m_invokeLocalObjectDesignationCodeSpinBox->setValue(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+2).toInt());
|
||||||
m_invokeLocalObjectTripletNumberSpinBox->setValue(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+3).toInt());
|
m_invokeLocalObjectTripletNumberSpinBox->setValue(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+3).toInt());
|
||||||
m_invokeLocalObjectDesignationCodeSpinBox->blockSignals(false);
|
m_invokeLocalObjectDesignationCodeSpinBox->blockSignals(false);
|
||||||
m_invokeLocalObjectTripletNumberSpinBox->blockSignals(false);
|
m_invokeLocalObjectTripletNumberSpinBox->blockSignals(false);
|
||||||
|
}
|
||||||
} else { // if (triplet.objectSource() != X26Triplet::IllegalObjectSource) {
|
} else { // if (triplet.objectSource() != X26Triplet::IllegalObjectSource) {
|
||||||
m_objectSourceComboBox->setCurrentIndex(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+1).toInt());
|
m_objectSourceComboBox->setCurrentIndex(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole+1).toInt());
|
||||||
m_invokeObjectSourceStackedLayout->setCurrentIndex(1);
|
m_invokeObjectSourceStackedLayout->setCurrentIndex(1);
|
||||||
@@ -849,28 +927,36 @@ void X26DockWidget::updateCookedTripletParameters(const QModelIndex &index)
|
|||||||
// Now deal with cooked row and column spinboxes
|
// Now deal with cooked row and column spinboxes
|
||||||
m_cookedRowSpinBox->blockSignals(true);
|
m_cookedRowSpinBox->blockSignals(true);
|
||||||
m_cookedColumnSpinBox->blockSignals(true);
|
m_cookedColumnSpinBox->blockSignals(true);
|
||||||
QVariant rowVariant = index.model()->data(index.model()->index(index.row(), 0), Qt::EditRole);
|
const QVariant rowVariant = index.model()->data(index.model()->index(index.row(), 0), Qt::EditRole);
|
||||||
if (rowVariant.isNull()) {
|
if (rowVariant.isNull()) {
|
||||||
m_cookedRowSpinBox->setEnabled(false);
|
m_cookedRowSpinBox->setEnabled(false);
|
||||||
m_cookedRowSpinBox->setValue(0);
|
m_cookedRowSpinBox->setValue(0);
|
||||||
|
m_cookedRowSpinBox->setPrefix("");
|
||||||
} else {
|
} else {
|
||||||
m_cookedRowSpinBox->setEnabled(true);
|
m_cookedRowSpinBox->setEnabled(true);
|
||||||
if (index.model()->data(index.model()->index(index.row(), 2), Qt::EditRole) == 0x10)
|
if (modeExt == 0x10) {
|
||||||
m_cookedRowSpinBox->setRange(0, 23);
|
m_cookedRowSpinBox->setRange(0, 23);
|
||||||
else
|
m_cookedRowSpinBox->setPrefix("+");
|
||||||
|
} else {
|
||||||
m_cookedRowSpinBox->setRange(1, 24);
|
m_cookedRowSpinBox->setRange(1, 24);
|
||||||
|
m_cookedRowSpinBox->setPrefix("");
|
||||||
|
}
|
||||||
m_cookedRowSpinBox->setValue(rowVariant.toInt());
|
m_cookedRowSpinBox->setValue(rowVariant.toInt());
|
||||||
}
|
}
|
||||||
QVariant columnVariant = index.model()->data(index.model()->index(index.row(), 1), Qt::EditRole);
|
const QVariant columnVariant = index.model()->data(index.model()->index(index.row(), 1), Qt::EditRole);
|
||||||
if (columnVariant.isNull()) {
|
if (columnVariant.isNull()) {
|
||||||
m_cookedColumnSpinBox->setEnabled(false);
|
m_cookedColumnSpinBox->setEnabled(false);
|
||||||
m_cookedColumnSpinBox->setValue(0);
|
m_cookedColumnSpinBox->setValue(0);
|
||||||
|
m_cookedColumnSpinBox->setPrefix("");
|
||||||
} else {
|
} else {
|
||||||
m_cookedColumnSpinBox->setEnabled(true);
|
m_cookedColumnSpinBox->setEnabled(true);
|
||||||
if (index.model()->data(index.model()->index(index.row(), 2), Qt::EditRole) == 0x10)
|
if (modeExt == 0x10) {
|
||||||
m_cookedColumnSpinBox->setMaximum(71);
|
m_cookedColumnSpinBox->setMaximum(71);
|
||||||
else
|
m_cookedColumnSpinBox->setPrefix("+");
|
||||||
|
} else {
|
||||||
m_cookedColumnSpinBox->setMaximum(39);
|
m_cookedColumnSpinBox->setMaximum(39);
|
||||||
|
m_cookedColumnSpinBox->setPrefix("");
|
||||||
|
}
|
||||||
m_cookedColumnSpinBox->setValue(columnVariant.toInt());
|
m_cookedColumnSpinBox->setValue(columnVariant.toInt());
|
||||||
}
|
}
|
||||||
m_cookedRowSpinBox->blockSignals(false);
|
m_cookedRowSpinBox->blockSignals(false);
|
||||||
@@ -945,35 +1031,87 @@ void X26DockWidget::cookedColumnSpinBoxChanged(const int value)
|
|||||||
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
||||||
}
|
}
|
||||||
|
|
||||||
void X26DockWidget::cookedModeComboBoxChanged(const int value)
|
void X26DockWidget::cookedModeMenuSelected(const int value)
|
||||||
{
|
{
|
||||||
if (!m_x26View->currentIndex().isValid())
|
if (!m_x26View->currentIndex().isValid())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Avoid "select..."
|
m_x26Model->setData(m_x26Model->index(m_x26View->currentIndex().row(), 2), value, Qt::EditRole);
|
||||||
if (m_cookedModeComboBox->itemData(value) == -1)
|
|
||||||
return;
|
|
||||||
|
|
||||||
m_x26Model->setData(m_x26Model->index(m_x26View->currentIndex().row(), 2), m_cookedModeComboBox->itemData(value).toInt(), Qt::EditRole);
|
|
||||||
|
|
||||||
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
||||||
updateCookedTripletParameters(m_x26View->currentIndex());
|
updateAllCookedTripletWidgets(m_x26View->currentIndex());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void X26DockWidget::updateModelFromCookedWidget(const int value, const int role)
|
void X26DockWidget::updateModelFromCookedWidget(const int value, const int role)
|
||||||
{
|
{
|
||||||
m_x26Model->setData(m_x26Model->index(m_x26View->currentIndex().row(), 0), value, role);
|
m_x26Model->setData(m_x26Model->index(m_x26View->currentIndex().row(), 0), value, role);
|
||||||
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
updateAllRawTripletSpinBoxes(m_x26View->currentIndex());
|
||||||
}
|
}
|
||||||
|
|
||||||
void X26DockWidget::insertTriplet()
|
void X26DockWidget::insertTriplet(int modeExt, bool after)
|
||||||
|
{
|
||||||
|
QModelIndex index = m_x26View->currentIndex();
|
||||||
|
X26Triplet newTriplet(modeExt < 0x20 ? 41 : 0, modeExt & 0x1f, 0);
|
||||||
|
int newListRow;
|
||||||
|
|
||||||
|
if (index.isValid()) {
|
||||||
|
newListRow = index.row()+after;
|
||||||
|
|
||||||
|
// If we're inserting a column triplet next to another column triplet,
|
||||||
|
// duplicate the column number
|
||||||
|
// Avoid the PDC and reserved mode triplets
|
||||||
|
if (modeExt >= 0x20 && modeExt != 0x24 && modeExt != 0x25 && modeExt != 0x26 && modeExt != 0x2a) {
|
||||||
|
const int existingTripletModeExt = index.model()->data(index.model()->index(index.row(), 2), Qt::EditRole).toInt();
|
||||||
|
|
||||||
|
if (existingTripletModeExt >= 0x20 && existingTripletModeExt != 0x24 && existingTripletModeExt != 0x25 && existingTripletModeExt != 0x26 && existingTripletModeExt != 0x2a)
|
||||||
|
newTriplet.setAddress(index.model()->data(index.model()->index(index.row(), 0), Qt::UserRole).toInt());
|
||||||
|
}
|
||||||
|
// If we're inserting a Set Active Position or Full Row Colour triplet,
|
||||||
|
// look for a previous row setting triplet and set this one to the row after
|
||||||
|
if (modeExt == 0x04 || modeExt == 0x01) {
|
||||||
|
for (int i=newListRow-1; i>=0; i--) {
|
||||||
|
const int scanTripletModeExt = index.model()->data(index.model()->index(i, 2), Qt::EditRole).toInt();
|
||||||
|
|
||||||
|
if (scanTripletModeExt == 0x04 || scanTripletModeExt == 0x01) {
|
||||||
|
const int scanActivePositionRow = index.model()->data(index.model()->index(i, 0), Qt::EditRole).toInt()+1;
|
||||||
|
|
||||||
|
if (scanActivePositionRow < 25)
|
||||||
|
newTriplet.setAddressRow(scanActivePositionRow);
|
||||||
|
else
|
||||||
|
newTriplet.setAddressRow(24);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else
|
||||||
|
newListRow = 0;
|
||||||
|
|
||||||
|
// For character triplets, ensure Data is not reserved
|
||||||
|
if (modeExt == 0x21 || modeExt == 0x22 || modeExt == 0x29 || modeExt == 0x2b || modeExt >= 0x2f)
|
||||||
|
newTriplet.setData(0x20);
|
||||||
|
// For Address Row 0, set Address
|
||||||
|
if (modeExt == 0x07)
|
||||||
|
newTriplet.setAddress(63);
|
||||||
|
// For Termination Marker, set Address and Mode
|
||||||
|
if (modeExt == 0x1f) {
|
||||||
|
newTriplet.setAddress(63);
|
||||||
|
newTriplet.setData(7);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_x26Model->insertRows(newListRow, 1, QModelIndex(), newTriplet);
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26DockWidget::insertTripletCopy()
|
||||||
{
|
{
|
||||||
QModelIndex index = m_x26View->currentIndex();
|
QModelIndex index = m_x26View->currentIndex();
|
||||||
|
|
||||||
if (index.isValid())
|
if (index.isValid())
|
||||||
m_x26Model->insertRow(index.row(), QModelIndex());
|
m_x26Model->insertRow(index.row(), QModelIndex());
|
||||||
else
|
else
|
||||||
m_x26Model->insertFirstRow();
|
// No existing triplet to copy, so insert a Termination Marker
|
||||||
|
m_x26Model->insertRows(0, 1, QModelIndex(), X26Triplet(63, 31, 7));
|
||||||
}
|
}
|
||||||
|
|
||||||
void X26DockWidget::deleteTriplet()
|
void X26DockWidget::deleteTriplet()
|
||||||
@@ -989,9 +1127,9 @@ void X26DockWidget::customMenuRequested(QPoint pos)
|
|||||||
|
|
||||||
QMenu *menu = new QMenu(this);
|
QMenu *menu = new QMenu(this);
|
||||||
|
|
||||||
QAction *insertAct = new QAction("Insert triplet", this);
|
QAction *insertAct = new QAction("Insert triplet copy", this);
|
||||||
menu->addAction(insertAct);
|
menu->addAction(insertAct);
|
||||||
connect(insertAct, &QAction::triggered, this, &X26DockWidget::insertTriplet);
|
connect(insertAct, &QAction::triggered, this, &X26DockWidget::insertTripletCopy);
|
||||||
if (index.isValid()) {
|
if (index.isValid()) {
|
||||||
QAction *deleteAct = new QAction("Delete triplet", this);
|
QAction *deleteAct = new QAction("Delete triplet", this);
|
||||||
menu->addAction(deleteAct);
|
menu->addAction(deleteAct);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -20,10 +20,13 @@
|
|||||||
#ifndef X26DOCKWIDGET_H
|
#ifndef X26DOCKWIDGET_H
|
||||||
#define X26DOCKWIDGET_H
|
#define X26DOCKWIDGET_H
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
#include <QCheckBox>
|
#include <QCheckBox>
|
||||||
#include <QComboBox>
|
#include <QComboBox>
|
||||||
#include <QDockWidget>
|
#include <QDockWidget>
|
||||||
#include <QGroupBox>
|
#include <QGroupBox>
|
||||||
|
#include <QLabel>
|
||||||
|
#include <QMenu>
|
||||||
#include <QPushButton>
|
#include <QPushButton>
|
||||||
#include <QRadioButton>
|
#include <QRadioButton>
|
||||||
#include <QSpinBox>
|
#include <QSpinBox>
|
||||||
@@ -31,8 +34,25 @@
|
|||||||
#include <QTableView>
|
#include <QTableView>
|
||||||
|
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
|
#include "render.h"
|
||||||
#include "x26model.h"
|
#include "x26model.h"
|
||||||
|
|
||||||
|
class CharacterListModel : public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
CharacterListModel(QObject *parent = 0);
|
||||||
|
|
||||||
|
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
|
void setCharacterSet(int characterSet);
|
||||||
|
|
||||||
|
private:
|
||||||
|
TeletextFontBitmap m_fontBitmap;
|
||||||
|
int m_characterSet;
|
||||||
|
};
|
||||||
|
|
||||||
class X26DockWidget : public QDockWidget
|
class X26DockWidget : public QDockWidget
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@@ -41,35 +61,35 @@ public:
|
|||||||
X26DockWidget(TeletextWidget *parent);
|
X26DockWidget(TeletextWidget *parent);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void insertTriplet();
|
void insertTriplet(int modeExt, bool after);
|
||||||
|
void insertTripletCopy();
|
||||||
void deleteTriplet();
|
void deleteTriplet();
|
||||||
void customMenuRequested(QPoint pos);
|
void customMenuRequested(QPoint pos);
|
||||||
void loadX26List();
|
void loadX26List();
|
||||||
void unloadX26List();
|
void unloadX26List();
|
||||||
void rowClicked(const QModelIndex &);
|
void rowSelected(const QModelIndex ¤t, const QModelIndex &previous);
|
||||||
void updateAllRawTripletSpinBoxes(const QModelIndex &);
|
void updateAllRawTripletSpinBoxes(const QModelIndex &index);
|
||||||
void updateRawTripletDataSpinBox(const QModelIndex &);
|
void updateRawTripletDataSpinBox(const QModelIndex &index);
|
||||||
void updateAllCookedTripletWidgets(const QModelIndex &);
|
void updateAllCookedTripletWidgets(const QModelIndex & index);
|
||||||
void updateCookedModeFromCookedType(const int);
|
void rawTripletAddressSpinBoxChanged(int value);
|
||||||
void updateCookedTripletParameters(const QModelIndex &);
|
void rawTripletModeSpinBoxChanged(int value);
|
||||||
void rawTripletAddressSpinBoxChanged(int);
|
void rawTripletDataSpinBoxChanged(int value);
|
||||||
void rawTripletModeSpinBoxChanged(int);
|
void cookedRowSpinBoxChanged(const int value);
|
||||||
void rawTripletDataSpinBoxChanged(int);
|
void cookedColumnSpinBoxChanged(const int value);
|
||||||
void cookedRowSpinBoxChanged(const int);
|
void cookedModeMenuSelected(const int value);
|
||||||
void cookedColumnSpinBoxChanged(const int);
|
void updateModelFromCookedWidget(const int value, const int role);
|
||||||
void cookedModeComboBoxChanged(const int);
|
void selectX26ListRow(int row);
|
||||||
void updateModelFromCookedWidget(const int, const int);
|
|
||||||
void selectX26ListRow(int);
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void keyPressEvent(QKeyEvent *event) override;
|
void keyPressEvent(QKeyEvent *event) override;
|
||||||
|
CharacterListModel m_characterListModel;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QTableView *m_x26View;
|
QTableView *m_x26View;
|
||||||
X26Model *m_x26Model;
|
X26Model *m_x26Model;
|
||||||
QComboBox *m_cookedModeTypeComboBox;
|
|
||||||
QSpinBox *m_cookedRowSpinBox, *m_cookedColumnSpinBox;
|
QSpinBox *m_cookedRowSpinBox, *m_cookedColumnSpinBox;
|
||||||
QComboBox *m_cookedModeComboBox;
|
QMenu *m_cookedModeMenu, *m_insertBeforeMenu, *m_insertAfterMenu;
|
||||||
|
QPushButton *m_cookedModePushButton;
|
||||||
QSpinBox *m_rawTripletAddressSpinBox, *m_rawTripletModeSpinBox, *m_rawTripletDataSpinBox;
|
QSpinBox *m_rawTripletAddressSpinBox, *m_rawTripletModeSpinBox, *m_rawTripletDataSpinBox;
|
||||||
QStackedLayout *m_rawOrCookedStackedLayout;
|
QStackedLayout *m_rawOrCookedStackedLayout;
|
||||||
QComboBox *m_colourComboBox;
|
QComboBox *m_colourComboBox;
|
||||||
@@ -79,6 +99,7 @@ private:
|
|||||||
QComboBox *m_textSizeComboBox;
|
QComboBox *m_textSizeComboBox;
|
||||||
QCheckBox *m_displayAttributeBoxingCheckBox, *m_displayAttributeConcealCheckBox, *m_displayAttributeInvertCheckBox, *m_displayAttributeUnderlineCheckBox;
|
QCheckBox *m_displayAttributeBoxingCheckBox, *m_displayAttributeConcealCheckBox, *m_displayAttributeInvertCheckBox, *m_displayAttributeUnderlineCheckBox;
|
||||||
QComboBox *m_objectSourceComboBox, *m_objectRequiredAtLevelsComboBox;
|
QComboBox *m_objectSourceComboBox, *m_objectRequiredAtLevelsComboBox;
|
||||||
|
QLabel *m_invokeLocalObjectDesignationCodeLabel, *m_invokeLocalObjectTripletNumberLabel;
|
||||||
QSpinBox *m_invokeLocalObjectDesignationCodeSpinBox, *m_invokeLocalObjectTripletNumberSpinBox;
|
QSpinBox *m_invokeLocalObjectDesignationCodeSpinBox, *m_invokeLocalObjectTripletNumberSpinBox;
|
||||||
QSpinBox *m_invokePOPSubPageSpinBox, *m_invokePOPPacketNumberSpinBox;
|
QSpinBox *m_invokePOPSubPageSpinBox, *m_invokePOPPacketNumberSpinBox;
|
||||||
QComboBox *m_invokePOPTripletNumberComboBox, *m_invokePOPPointerBitsComboBox;
|
QComboBox *m_invokePOPTripletNumberComboBox, *m_invokePOPPointerBitsComboBox;
|
||||||
@@ -93,9 +114,12 @@ private:
|
|||||||
QSpinBox *m_reservedPDCSpinBox;
|
QSpinBox *m_reservedPDCSpinBox;
|
||||||
QComboBox *m_terminationMarkerPageTypeComboBox;
|
QComboBox *m_terminationMarkerPageTypeComboBox;
|
||||||
QCheckBox *m_terminationMarkerMoreFollowsCheckBox;
|
QCheckBox *m_terminationMarkerMoreFollowsCheckBox;
|
||||||
QPushButton *m_insertPushButton, *m_deletePushButton;
|
QPushButton *m_insertBeforePushButton, *m_insertAfterPushButton, *m_insertCopyPushButton, *m_deletePushButton;
|
||||||
|
|
||||||
TeletextWidget *m_parentMainWidget;
|
TeletextWidget *m_parentMainWidget;
|
||||||
|
|
||||||
|
void disableTripletWidgets();
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
853
x26model.cpp
853
x26model.cpp
File diff suppressed because it is too large
Load Diff
38
x26model.h
38
x26model.h
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -29,17 +29,19 @@ class X26Model : public QAbstractListModel
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
X26Model(TeletextWidget *parent);
|
X26Model(TeletextWidget *parent);
|
||||||
void setX26ListLoaded(bool);
|
void setX26ListLoaded(bool newListLoaded);
|
||||||
int rowCount(const QModelIndex &parent = QModelIndex()) const override ;
|
int rowCount(const QModelIndex &parent = QModelIndex()) const override ;
|
||||||
int columnCount(const QModelIndex &parent = QModelIndex()) const override;
|
int columnCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
bool setData(const QModelIndex &index, const QVariant &value, int role);
|
bool setData(const QModelIndex &index, const QVariant &value, int role);
|
||||||
QVariant headerData(int section, Qt::Orientation orientation, int role) const;
|
QVariant headerData(int section, Qt::Orientation orientation, int role) const;
|
||||||
bool insertFirstRow();
|
|
||||||
bool insertRows(int position, int rows, const QModelIndex &parent);
|
bool insertRows(int position, int rows, const QModelIndex &parent);
|
||||||
|
bool insertRows(int position, int rows, const QModelIndex &parent, X26Triplet triplet);
|
||||||
bool removeRows(int position, int rows, const QModelIndex &index);
|
bool removeRows(int position, int rows, const QModelIndex &index);
|
||||||
// Qt::ItemFlags flags(const QModelIndex &index) const;
|
// Qt::ItemFlags flags(const QModelIndex &index) const;
|
||||||
|
|
||||||
|
const QString modeTripletName(int i) const { return m_modeTripletName[i]; }
|
||||||
|
|
||||||
// The x26commands classes manipulate the model but beginInsertRows and endInsertRows
|
// The x26commands classes manipulate the model but beginInsertRows and endInsertRows
|
||||||
// are protected methods, so we need to friend them
|
// are protected methods, so we need to friend them
|
||||||
friend class InsertTripletCommand;
|
friend class InsertTripletCommand;
|
||||||
@@ -49,9 +51,9 @@ public:
|
|||||||
private:
|
private:
|
||||||
TeletextWidget *m_parentMainWidget;
|
TeletextWidget *m_parentMainWidget;
|
||||||
bool m_listLoaded;
|
bool m_listLoaded;
|
||||||
};
|
TeletextFontBitmap m_fontBitmap;
|
||||||
|
|
||||||
static const QString modeTripletName[64] {
|
const QString m_modeTripletName[64] {
|
||||||
// Row triplet modes
|
// Row triplet modes
|
||||||
"Full screen colour",
|
"Full screen colour",
|
||||||
"Full row colour",
|
"Full row colour",
|
||||||
@@ -95,8 +97,8 @@ static const QString modeTripletName[64] {
|
|||||||
|
|
||||||
// Column triplet modes
|
// Column triplet modes
|
||||||
"Foreground colour",
|
"Foreground colour",
|
||||||
"G1 character",
|
"G1 block mosaic",
|
||||||
"G3 character, level 1.5",
|
"G3 smooth mosaic, level 1.5",
|
||||||
"Background colour",
|
"Background colour",
|
||||||
|
|
||||||
"Reserved 0x04",
|
"Reserved 0x04",
|
||||||
@@ -107,12 +109,12 @@ static const QString modeTripletName[64] {
|
|||||||
"Modified G0/G2 character set",
|
"Modified G0/G2 character set",
|
||||||
"G0 character",
|
"G0 character",
|
||||||
"Reserved 0x0a",
|
"Reserved 0x0a",
|
||||||
"G3 character, level 2.5",
|
"G3 smooth mosaic, level 2.5",
|
||||||
|
|
||||||
"Display attributes",
|
"Display attributes",
|
||||||
"DRCS character",
|
"DRCS character",
|
||||||
"Font style",
|
"Font style, level 3.5",
|
||||||
"G2 character",
|
"G2 supplementary character",
|
||||||
|
|
||||||
"G0 character no diacritical",
|
"G0 character no diacritical",
|
||||||
"G0 character diacritical 1",
|
"G0 character diacritical 1",
|
||||||
@@ -130,6 +132,22 @@ static const QString modeTripletName[64] {
|
|||||||
"G0 character diacritical D",
|
"G0 character diacritical D",
|
||||||
"G0 character diacritical E",
|
"G0 character diacritical E",
|
||||||
"G0 character diacritical F"
|
"G0 character diacritical F"
|
||||||
|
};
|
||||||
|
|
||||||
|
struct tripletErrorShow {
|
||||||
|
QString message;
|
||||||
|
int columnHighlight;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Needs to be in the same order as enum X26TripletError in x26triplets.h
|
||||||
|
const tripletErrorShow m_tripletErrors[6] {
|
||||||
|
{ "", 0 }, // No error
|
||||||
|
{ "Active Position can't move up", 0 },
|
||||||
|
{ "Active Position can't move left within row", 1 },
|
||||||
|
{ "Invocation not pointing to Object Definition", 3 },
|
||||||
|
{ "Invoked and Defined Object types don't match", 2 },
|
||||||
|
{ "Origin Modifier not followed by Object Invocation", 2 }
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
261
x26triplets.cpp
261
x26triplets.cpp
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -50,3 +50,262 @@ void X26Triplet::setAddressColumn(int addressColumn)
|
|||||||
{
|
{
|
||||||
m_address = addressColumn;
|
m_address = addressColumn;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void X26Triplet::setObjectLocalDesignationCode(int i)
|
||||||
|
{
|
||||||
|
m_address = (m_address & 0x38) | (i >> 3);
|
||||||
|
m_data = (m_data & 0x0f) | ((i & 0x07) << 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26Triplet::setObjectLocalTripletNumber(int i)
|
||||||
|
{
|
||||||
|
m_data = (m_data & 0x70) | i;
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26Triplet::setObjectLocalIndex(int i)
|
||||||
|
{
|
||||||
|
m_address = (m_address & 0x38) + (i >= 104); // Set bit 0 of address if triplet >= 8
|
||||||
|
m_data = (((i / 13) & 0x07) << 4) | (i % 13);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void X26TripletList::updateInternalData()
|
||||||
|
{
|
||||||
|
ActivePosition activePosition;
|
||||||
|
X26Triplet *triplet;
|
||||||
|
|
||||||
|
// Check for errors, and fill in where the Active Position goes for Level 2.5 and above
|
||||||
|
for (int i=0; i < m_list.size(); i++) {
|
||||||
|
triplet = &m_list[i];
|
||||||
|
triplet->m_error = X26Triplet::NoError;
|
||||||
|
triplet->m_reservedMode = false;
|
||||||
|
triplet->m_reservedData = false;
|
||||||
|
|
||||||
|
if (triplet->isRowTriplet()) {
|
||||||
|
switch (triplet->modeExt()) {
|
||||||
|
case 0x00: // Full screen colour
|
||||||
|
if (activePosition.isDeployed())
|
||||||
|
// TODO more specific error needed
|
||||||
|
triplet->m_error = X26Triplet::ActivePositionMovedUp;
|
||||||
|
if (triplet->m_data & 0x60)
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
break;
|
||||||
|
case 0x01: // Full row colour
|
||||||
|
if (!activePosition.setRow(triplet->addressRow()))
|
||||||
|
triplet->m_error = X26Triplet::ActivePositionMovedUp;
|
||||||
|
if ((triplet->m_data & 0x60) != 0x00 && (triplet->m_data & 0x60) != 0x60)
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
break;
|
||||||
|
case 0x04: // Set Active Position;
|
||||||
|
if (!activePosition.setRow(triplet->addressRow()))
|
||||||
|
triplet->m_error = X26Triplet::ActivePositionMovedUp;
|
||||||
|
else if (triplet->data() >= 40)
|
||||||
|
// FIXME data column highlighted?
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
else if (!activePosition.setColumn(triplet->data()))
|
||||||
|
triplet->m_error = X26Triplet::ActivePositionMovedLeft;
|
||||||
|
break;
|
||||||
|
case 0x07: // Address row 0
|
||||||
|
if (triplet->m_address != 63)
|
||||||
|
// FIXME data column highlighted?
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
else if (activePosition.isDeployed())
|
||||||
|
triplet->m_error = X26Triplet::ActivePositionMovedUp;
|
||||||
|
else {
|
||||||
|
activePosition.setRow(0);
|
||||||
|
activePosition.setColumn(8);
|
||||||
|
}
|
||||||
|
if ((triplet->m_data & 0x60) != 0x00 && (triplet->m_data & 0x60) != 0x60)
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
break;
|
||||||
|
case 0x10: // Origin Modifier
|
||||||
|
if (i == m_list.size()-1 ||
|
||||||
|
m_list.at(i+1).modeExt() < 0x11 ||
|
||||||
|
m_list.at(i+1).modeExt() > 0x13)
|
||||||
|
triplet->m_error = X26Triplet::OriginModifierAlone;
|
||||||
|
break;
|
||||||
|
case 0x11: // Invoke Active Object
|
||||||
|
case 0x12: // Invoke Adaptive Object
|
||||||
|
case 0x13: // Invoke Passive Object
|
||||||
|
if (triplet->objectSource() == X26Triplet::LocalObject) {
|
||||||
|
if (triplet->objectLocalTripletNumber() > 12 ||
|
||||||
|
triplet->objectLocalIndex() > (m_list.size()-1) ||
|
||||||
|
m_list.at(triplet->objectLocalIndex()).modeExt() < 0x15 ||
|
||||||
|
m_list.at(triplet->objectLocalIndex()).modeExt() > 0x17)
|
||||||
|
triplet->m_error = X26Triplet::InvokePointerInvalid;
|
||||||
|
else if ((triplet->modeExt() | 0x04) != m_list.at(triplet->objectLocalIndex()).modeExt())
|
||||||
|
triplet->m_error = X26Triplet::InvokeTypeMismatch;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 0x15: // Define Active Object
|
||||||
|
case 0x16: // Define Adaptive Object
|
||||||
|
case 0x17: // Define Passive Object
|
||||||
|
activePosition.reset();
|
||||||
|
// Make sure data field holds correct place of triplet
|
||||||
|
// otherwise the object won't appear
|
||||||
|
triplet->setObjectLocalIndex(i);
|
||||||
|
break;
|
||||||
|
case 0x18: // DRCS mode
|
||||||
|
if ((triplet->m_data & 0x30) == 0x00)
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
case 0x1f: // Termination marker
|
||||||
|
case 0x08: // PDC country of origin & programme source
|
||||||
|
case 0x09: // PDC month & day
|
||||||
|
case 0x0a: // PDC cursor row & announced start hour
|
||||||
|
case 0x0b: // PDC cursor row & announced finish hour
|
||||||
|
case 0x0c: // PDC cursor row & local time offset
|
||||||
|
case 0x0d: // PDC series ID & series code
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
triplet->m_reservedMode = true;
|
||||||
|
};
|
||||||
|
// Column triplet: all triplets modes except PDC and reserved move the Active Position
|
||||||
|
} else if (triplet->modeExt() == 0x24 || triplet->modeExt() == 0x25 || triplet->modeExt() == 0x2a)
|
||||||
|
triplet->m_reservedMode = true;
|
||||||
|
else if (triplet->modeExt() != 0x26 && !activePosition.setColumn(triplet->addressColumn()))
|
||||||
|
triplet->m_error = X26Triplet::ActivePositionMovedLeft;
|
||||||
|
else
|
||||||
|
switch (triplet->modeExt()) {
|
||||||
|
case 0x20: // Foreground colour
|
||||||
|
case 0x23: // Background colour
|
||||||
|
if (triplet->m_data & 0x60)
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
break;
|
||||||
|
case 0x27: // Additional flash functions
|
||||||
|
if (triplet->m_data >= 0x18)
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
break;
|
||||||
|
case 0x28: // Modified G0 and G2 character set
|
||||||
|
if (triplet->m_data > 0x26)
|
||||||
|
switch (triplet->m_data) {
|
||||||
|
case 0x36:
|
||||||
|
case 0x37:
|
||||||
|
case 0x40:
|
||||||
|
case 0x44:
|
||||||
|
case 0x47:
|
||||||
|
case 0x55:
|
||||||
|
case 0x57:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 0x2d: // DRCS character
|
||||||
|
if ((triplet->m_data & 0x3f) >= 48)
|
||||||
|
// Should really be an error?
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
break;
|
||||||
|
case 0x21: // G1 mosaic character
|
||||||
|
case 0x22: // G3 mosaic character at level 1.5
|
||||||
|
case 0x29: // G0 character
|
||||||
|
case 0x2b: // G3 mosaic character at level >=2.5
|
||||||
|
case 0x2f: // G2 character
|
||||||
|
if (triplet->m_data < 0x20)
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (triplet->modeExt() >= 0x30 && triplet->modeExt() <= 0x3f && triplet->m_data < 0x20)
|
||||||
|
// G0 diacritical mark
|
||||||
|
triplet->m_reservedData = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
triplet->m_activePositionRow = activePosition.row();
|
||||||
|
triplet->m_activePositionColumn = activePosition.column();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now work out where the Active Position goes on a Level 1.5 decoder
|
||||||
|
activePosition.reset();
|
||||||
|
|
||||||
|
for (int i=0; i < m_list.size(); i++) {
|
||||||
|
triplet = &m_list[i];
|
||||||
|
|
||||||
|
if (triplet->modeExt() == 0x1f) // Termination marker
|
||||||
|
break;
|
||||||
|
|
||||||
|
switch (triplet->modeExt()) {
|
||||||
|
case 0x04: // Set Active Position;
|
||||||
|
activePosition.setRow(triplet->addressRow());
|
||||||
|
break;
|
||||||
|
case 0x07: // Address row 0
|
||||||
|
if (triplet->m_address == 63 && activePosition.setRow(0))
|
||||||
|
activePosition.setColumn(8);
|
||||||
|
break;
|
||||||
|
case 0x22: // G3 mosaic character at level 1.5
|
||||||
|
case 0x2f: // G2 character
|
||||||
|
activePosition.setColumn(triplet->addressColumn());
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (triplet->modeExt() >= 0x30 && triplet->modeExt() <= 0x3f)
|
||||||
|
// G0 diacritical mark
|
||||||
|
activePosition.setColumn(triplet->addressColumn());
|
||||||
|
}
|
||||||
|
|
||||||
|
triplet->m_activePositionRow1p5 = activePosition.row();
|
||||||
|
triplet->m_activePositionColumn1p5 = activePosition.column();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26TripletList::append(const X26Triplet &value)
|
||||||
|
{
|
||||||
|
m_list.append(value);
|
||||||
|
updateInternalData();
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26TripletList::insert(int i, const X26Triplet &value)
|
||||||
|
{
|
||||||
|
m_list.insert(i, value);
|
||||||
|
updateInternalData();
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26TripletList::removeAt(int i)
|
||||||
|
{
|
||||||
|
m_list.removeAt(i);
|
||||||
|
if (m_list.size() != 0 && i < m_list.size())
|
||||||
|
updateInternalData();
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26TripletList::replace(int i, const X26Triplet &value)
|
||||||
|
{
|
||||||
|
m_list.replace(i, value);
|
||||||
|
updateInternalData();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
X26TripletList::ActivePosition::ActivePosition()
|
||||||
|
{
|
||||||
|
m_row = m_column = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void X26TripletList::ActivePosition::reset()
|
||||||
|
{
|
||||||
|
m_row = m_column = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool X26TripletList::ActivePosition::setRow(int row)
|
||||||
|
{
|
||||||
|
if (row < m_row)
|
||||||
|
return false;
|
||||||
|
if (row > m_row) {
|
||||||
|
m_row = row;
|
||||||
|
m_column = -1;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool X26TripletList::ActivePosition::setColumn(int column)
|
||||||
|
{
|
||||||
|
if (column < m_column)
|
||||||
|
return false;
|
||||||
|
if (m_row == -1 and column >= 0)
|
||||||
|
m_row = 0;
|
||||||
|
m_column = column;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
bool X26TripletList::ActivePosition::setRowAndColumn(int newRow, int newColumn)
|
||||||
|
{
|
||||||
|
if (!setRow(newRow))
|
||||||
|
return false;
|
||||||
|
return setColumn(newColumn);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2020, 2021 Gavin MacGregor
|
* Copyright (C) 2020-2024 Gavin MacGregor
|
||||||
*
|
*
|
||||||
* This file is part of QTeletextMaker.
|
* This file is part of QTeletextMaker.
|
||||||
*
|
*
|
||||||
@@ -20,15 +20,21 @@
|
|||||||
#ifndef X26TRIPLETS_H
|
#ifndef X26TRIPLETS_H
|
||||||
#define X26TRIPLETS_H
|
#define X26TRIPLETS_H
|
||||||
|
|
||||||
|
#include <QList>
|
||||||
|
|
||||||
class X26Triplet
|
class X26Triplet
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
// x26model.h has the Plain English descriptions of these errors
|
||||||
|
enum X26TripletError { NoError, ActivePositionMovedUp, ActivePositionMovedLeft, InvokePointerInvalid, InvokeTypeMismatch, OriginModifierAlone };
|
||||||
|
enum ObjectSource { InvalidObjectSource, LocalObject, POPObject, GPOPObject };
|
||||||
|
|
||||||
X26Triplet() {}
|
X26Triplet() {}
|
||||||
// X26Triplet(const X26Triplet &other);
|
// X26Triplet(const X26Triplet &other);
|
||||||
|
|
||||||
// X26Triplet &operator=(const X26Triplet &other);
|
// X26Triplet &operator=(const X26Triplet &other);
|
||||||
|
|
||||||
X26Triplet(int, int, int);
|
X26Triplet(int address, int mode, int data);
|
||||||
|
|
||||||
int address() const { return m_address; }
|
int address() const { return m_address; }
|
||||||
int mode() const { return m_mode; }
|
int mode() const { return m_mode; }
|
||||||
@@ -38,15 +44,81 @@ public:
|
|||||||
int addressColumn() const { return (m_address); }
|
int addressColumn() const { return (m_address); }
|
||||||
bool isRowTriplet() const { return (m_address >= 40); }
|
bool isRowTriplet() const { return (m_address >= 40); }
|
||||||
|
|
||||||
void setAddress(int);
|
void setAddress(int address);
|
||||||
void setMode(int);
|
void setMode(int mode);
|
||||||
void setData(int);
|
void setData(int data);
|
||||||
void setAddressRow(int);
|
void setAddressRow(int addressRow);
|
||||||
void setAddressColumn(int);
|
void setAddressColumn(int addressColumn);
|
||||||
|
|
||||||
|
int objectSource() const { return (m_address & 0x18) >> 3; }
|
||||||
|
|
||||||
|
int objectLocalDesignationCode() const { return (((m_address & 0x01) << 3) | (m_data >> 4)); }
|
||||||
|
int objectLocalTripletNumber() const { return m_data & 0x0f; }
|
||||||
|
int objectLocalIndex() const { return objectLocalDesignationCode() * 13 + objectLocalTripletNumber(); }
|
||||||
|
void setObjectLocalDesignationCode(int i);
|
||||||
|
void setObjectLocalTripletNumber(int i);
|
||||||
|
void setObjectLocalIndex(int i);
|
||||||
|
|
||||||
|
int activePositionRow() const { return m_activePositionRow; }
|
||||||
|
int activePositionColumn() const { return m_activePositionColumn; }
|
||||||
|
int activePositionRow1p5() const { return m_activePositionRow1p5; }
|
||||||
|
int activePositionColumn1p5() const { return m_activePositionColumn1p5; }
|
||||||
|
X26TripletError error() const { return m_error; }
|
||||||
|
bool reservedMode() const { return m_reservedMode; }
|
||||||
|
bool reservedData() const { return m_reservedData; }
|
||||||
|
|
||||||
|
friend class X26TripletList;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
// Only these variables are manipulated by the X26Triplet class
|
||||||
int m_address, m_mode, m_data;
|
int m_address, m_mode, m_data;
|
||||||
|
// and the following are filled in by X26TripletList::updateInternalData()
|
||||||
|
int m_activePositionRow = -1;
|
||||||
|
int m_activePositionColumn = -1;
|
||||||
|
int m_activePositionRow1p5 = -1;
|
||||||
|
int m_activePositionColumn1p5 = -1;
|
||||||
|
X26TripletError m_error = NoError;
|
||||||
|
bool m_reservedMode = false;
|
||||||
|
bool m_reservedData = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
class X26TripletList
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void append(const X26Triplet &value);
|
||||||
|
void insert(int i, const X26Triplet &value);
|
||||||
|
void removeAt(int i);
|
||||||
|
void replace(int i, const X26Triplet &value);
|
||||||
|
|
||||||
|
void removeLast() { m_list.removeLast(); }
|
||||||
|
|
||||||
|
const X26Triplet &at(int i) const { return m_list.at(i); }
|
||||||
|
bool isEmpty() const { return m_list.isEmpty(); }
|
||||||
|
void reserve(int alloc) { m_list.reserve(alloc); }
|
||||||
|
int size() const { return m_list.size(); }
|
||||||
|
|
||||||
|
private:
|
||||||
|
void updateInternalData();
|
||||||
|
|
||||||
|
QList<X26Triplet> m_list;
|
||||||
|
|
||||||
|
class ActivePosition
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
ActivePosition();
|
||||||
|
void reset();
|
||||||
|
// int row() const { return (m_row == -1) ? 0 : m_row; }
|
||||||
|
// int column() const { return (m_column == -1) ? 0 : m_column; }
|
||||||
|
int row() const { return m_row; }
|
||||||
|
int column() const { return m_column; }
|
||||||
|
bool isDeployed() const { return m_row != -1; }
|
||||||
|
bool setRow(int);
|
||||||
|
bool setColumn(int);
|
||||||
|
// bool setRowAndColumn(int, int);
|
||||||
|
|
||||||
|
private:
|
||||||
|
int m_row, m_column;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
Reference in New Issue
Block a user