This commit is contained in:
Adam D. Ruppe 2021-04-11 22:22:39 -04:00
parent d5c3539293
commit 3d396dfaa6
5 changed files with 288 additions and 30 deletions

7
cgi.d
View File

@ -2114,7 +2114,7 @@ class Cgi {
None None
} }
/+ /++
Sets an HTTP cookie, automatically encoding the data to the correct string. Sets an HTTP cookie, automatically encoding the data to the correct string.
expiresIn is how many milliseconds in the future the cookie will expire. expiresIn is how many milliseconds in the future the cookie will expire.
TIP: to make a cookie accessible from subdomains, set the domain to .yourdomain.com. TIP: to make a cookie accessible from subdomains, set the domain to .yourdomain.com.
@ -3354,15 +3354,10 @@ bool tryAddonServers(string[] args) {
printf("Add-on servers not compiled in.\n"); printf("Add-on servers not compiled in.\n");
return true; return true;
case "--timer-server": case "--timer-server":
try {
version(with_addon_servers) version(with_addon_servers)
runTimerServer(); runTimerServer();
else else
printf("Add-on servers not compiled in.\n"); printf("Add-on servers not compiled in.\n");
} catch(Throwable t) {
import std.file;
std.file.write("/tmp/timer-exception", t.toString);
}
return true; return true;
case "--timed-jobs": case "--timed-jobs":
import core.demangle; import core.demangle;

View File

@ -343,7 +343,8 @@ struct HttpResponse {
// ignore // ignore
} }
header = header[1 .. $]; if(header.length)
header = header[1 .. $];
} }
ret ~= current; ret ~= current;
@ -1529,8 +1530,11 @@ class HttpRequest {
bodyReadingState.chunkedState = 0; bodyReadingState.chunkedState = 0;
while(data[a] != 10) while(data[a] != 10) {
a++; a++;
if(a == data.length)
return stillAlive; // in the footer state we're just discarding everything until we're done so this should be ok
}
data = data[a + 1 .. $]; data = data[a + 1 .. $];
if(bodyReadingState.isGzipped || bodyReadingState.isDeflated) { if(bodyReadingState.isGzipped || bodyReadingState.isDeflated) {

View File

@ -4,6 +4,9 @@
// https://docs.microsoft.com/en-us/windows/win32/dataxchg/html-clipboard-format // https://docs.microsoft.com/en-us/windows/win32/dataxchg/html-clipboard-format
// https://www.x.org/releases/X11R7.7/doc/libXext/dbelib.html
// https://www.x.org/releases/X11R7.6/doc/libXext/synclib.html
// on Mac with X11: -L-L/usr/X11/lib // on Mac with X11: -L-L/usr/X11/lib

View File

@ -1628,7 +1628,17 @@ struct Terminal {
Params: Params:
text = text displayed in the terminal text = text displayed in the terminal
identifier = an additional number attached to the text and returned to you in a [LinkEvent]
identifier = an additional number attached to the text and returned to you in a [LinkEvent].
Possible uses of this are to have a small number of "link classes" that are handled based on
the text. For example, maybe identifier == 0 means paste text into the line. identifier == 1
could mean open a browser. identifier == 2 might open details for it. Just be sure to encode
the bulk of the information into the text so the user can copy/paste it out too.
You may also create a mapping of (identifier,text) back to some other activity, but if you do
that, be sure to check [hyperlinkSupported] and fallback in your own code so it still makes
sense to users on other terminals.
autoStyle = set to `false` to suppress the automatic color and underlining of the text. autoStyle = set to `false` to suppress the automatic color and underlining of the text.
Bugs: Bugs:
@ -1662,6 +1672,21 @@ struct Terminal {
} }
} }
/++
Returns true if the terminal advertised compatibility with the [hyperlink] function's
implementation.
History:
Added April 2, 2021
+/
bool hyperlinkSupported() {
if((tcaps & TerminalCapabilities.arsdHyperlinks)) {
return true;
} else {
return false;
}
}
/// Note: the Windows console does not support underlining /// Note: the Windows console does not support underlining
void underline(bool set, ForceOption force = ForceOption.automatic) { void underline(bool set, ForceOption force = ForceOption.automatic) {
if(set == _underlined && force != ForceOption.alwaysSend) if(set == _underlined && force != ForceOption.alwaysSend)
@ -3871,9 +3896,9 @@ struct PasteEvent {
Added March 18, 2020 Added March 18, 2020
+/ +/
struct LinkEvent { struct LinkEvent {
string text; /// string text; /// the text visible to the user that they clicked on
ushort identifier; /// ushort identifier; /// the identifier set when you output the link. This is small because it is packed into extra bits on the text, one bit per character.
ushort command; /// set by the terminal to indicate how it was clicked. values tbd ushort command; /// set by the terminal to indicate how it was clicked. values tbd, currently always 0
} }
/// . /// .
@ -6347,6 +6372,10 @@ class LineGetter {
} }
} }
break; break;
case InputEvent.Type.LinkEvent:
if(handleLinkEvent !is null)
handleLinkEvent(e.linkEvent, this);
break;
case InputEvent.Type.SizeChangedEvent: case InputEvent.Type.SizeChangedEvent:
/* We'll adjust the bounding box. If you don't like this, handle SizeChangedEvent /* We'll adjust the bounding box. If you don't like this, handle SizeChangedEvent
yourself and then don't pass it to this function. */ yourself and then don't pass it to this function. */
@ -6368,6 +6397,29 @@ class LineGetter {
return true; return true;
} }
/++
Gives a convenience hook for subclasses to handle my terminal's hyperlink extension.
You can also handle these by filtering events before you pass them to [workOnLine].
That's still how I recommend handling any overrides or custom events, but making this
a delegate is an easy way to inject handlers into an otherwise linear i/o application.
Does nothing if null.
It passes the event as well as the current line getter to the delegate. You may simply
`lg.addString(ev.text); lg.redraw();` in some cases.
History:
Added April 2, 2021.
See_Also:
[Terminal.hyperlink]
[TerminalCapabilities.arsdHyperlinks]
+/
void delegate(LinkEvent ev, LineGetter lg) handleLinkEvent;
/++ /++
Replaces the line currently being edited with the given line and positions the cursor inside it. Replaces the line currently being edited with the given line and positions the cursor inside it.
@ -7527,6 +7579,18 @@ version(TerminalDirectToEmulator) {
Represents the window that the library pops up for you. Represents the window that the library pops up for you.
+/ +/
final class TerminalEmulatorWindow : MainWindow { final class TerminalEmulatorWindow : MainWindow {
/++
Returns the size of an individual character cell, in pixels.
History:
Added April 2, 2021
+/
Size characterCellSize() {
if(tew && tew.terminalEmulator)
return Size(tew.terminalEmulator.fontWidth, tew.terminalEmulator.fontHeight);
else
return Size(1, 1);
}
/++ /++
Gives access to the underlying terminal emulation object. Gives access to the underlying terminal emulation object.
@ -7943,6 +8007,7 @@ version(TerminalDirectToEmulator) {
widget.smw.setViewableArea(this.width, this.height); widget.smw.setViewableArea(this.width, this.height);
widget.smw.setPageSize(this.width / 2, this.height / 2); widget.smw.setPageSize(this.width / 2, this.height / 2);
} }
notifyScrollbarPosition(0, int.max);
clearScreenRequested = true; clearScreenRequested = true;
if(widget && widget.term) if(widget && widget.term)
widget.term.windowSizeChanged = true; widget.term.windowSizeChanged = true;

View File

@ -45,12 +45,15 @@ interface NonCharacterData {
//const(ubyte)[] serialize(); //const(ubyte)[] serialize();
} }
struct BrokenUpImage { struct BinaryDataTerminalRepresentation {
int width; int width;
int height; int height;
TerminalEmulator.TerminalCell[] representation; TerminalEmulator.TerminalCell[] representation;
} }
// old name, don't use in new programs anymore.
deprecated alias BrokenUpImage = BinaryDataTerminalRepresentation;
struct CustomGlyph { struct CustomGlyph {
TrueColorImage image; TrueColorImage image;
dchar substitute; dchar substitute;
@ -864,8 +867,8 @@ class TerminalEmulator {
} }
/// if a binary extension is triggered, the implementing class is responsible for figuring out how it should be made to fit into the screen buffer /// if a binary extension is triggered, the implementing class is responsible for figuring out how it should be made to fit into the screen buffer
protected /*abstract*/ BrokenUpImage handleBinaryExtensionData(const(ubyte)[]) { protected /*abstract*/ BinaryDataTerminalRepresentation handleBinaryExtensionData(const(ubyte)[]) {
return BrokenUpImage(); return BinaryDataTerminalRepresentation();
} }
/// If you subclass this and return true, you can scroll on command without needing to redraw the entire screen; /// If you subclass this and return true, you can scroll on command without needing to redraw the entire screen;
@ -1504,16 +1507,24 @@ class TerminalEmulator {
int max = cast(int) scrollbackBuffer.length - screenHeight; int max = cast(int) scrollbackBuffer.length - screenHeight;
if(scrollbackReflow && max < 0) { if(scrollbackReflow && max < 0) {
foreach(line; scrollbackBuffer[]) foreach(line; scrollbackBuffer[]) {
max += cast(int) line.length / screenWidth; if(line.length > 2 && (line[0].hasNonCharacterData || line[$-1].hasNonCharacterData))
max += 0;
else
max += cast(int) line.length / screenWidth;
}
} }
if(max < 0) if(max < 0)
max = 0; max = 0;
if(scrollbackReflow && currentScrollback > max) { if(scrollbackReflow && currentScrollback > max) {
foreach(line; scrollbackBuffer[]) foreach(line; scrollbackBuffer[]) {
max += cast(int) line.length / screenWidth; if(line.length > 2 && (line[0].hasNonCharacterData || line[$-1].hasNonCharacterData))
max += 0;
else
max += cast(int) line.length / screenWidth;
}
} }
if(currentScrollback > max) if(currentScrollback > max)
@ -1615,7 +1626,10 @@ class TerminalEmulator {
int max; int max;
if(scrollbackReflow) { if(scrollbackReflow) {
foreach(line; scrollbackBuffer[]) { foreach(line; scrollbackBuffer[]) {
count += cast(int) line.length / screenWidth; if(line.length > 2 && (line[0].hasNonCharacterData || line[$-1].hasNonCharacterData))
{} // intentionally blank, the count is fine since this line isn't reflowed anyway
else
count += cast(int) line.length / screenWidth;
} }
} else { } else {
foreach(line; scrollbackBuffer[]) { foreach(line; scrollbackBuffer[]) {
@ -1684,6 +1698,11 @@ class TerminalEmulator {
int idx = cast(int) scrollbackBuffer.length - 1; int idx = cast(int) scrollbackBuffer.length - 1;
foreach_reverse(line; scrollbackBuffer[]) { foreach_reverse(line; scrollbackBuffer[]) {
auto lineCount = 1 + line.length / screenWidth; auto lineCount = 1 + line.length / screenWidth;
// if the line has an image in it, it cannot be reflowed. this hack to check just the first and last thing is the cheapest way rn
if(line.length > 2 && (line[0].hasNonCharacterData || line[$-1].hasNonCharacterData))
lineCount = 1;
numLines += lineCount; numLines += lineCount;
if(numLines >= (screenHeight + howFar)) { if(numLines >= (screenHeight + howFar)) {
start = cast(int) idx; start = cast(int) idx;
@ -1736,6 +1755,9 @@ class TerminalEmulator {
if(cursorX == screenWidth-1) { if(cursorX == screenWidth-1) {
if(scrollbackReflow) { if(scrollbackReflow) {
// don't attempt to reflow images
if(cell.hasNonCharacterData)
break;
cursorX = 0; cursorX = 0;
if(cursorY + 1 == screenHeight) if(cursorY + 1 == screenHeight)
break outer; break outer;
@ -1765,7 +1787,8 @@ class TerminalEmulator {
if(scrollLock) if(scrollLock)
toggleScrollLock(); toggleScrollLock();
endScrollback(); // FIXME: hack // FIXME: hack
endScrollback();
screenWidth = w; screenWidth = w;
screenHeight = h; screenHeight = h;
@ -2177,7 +2200,11 @@ class TerminalEmulator {
} else { } else {
if(!scrollbackReflow && line.length > scrollbackWidth_) if(!scrollbackReflow && line.length > scrollbackWidth_)
scrollbackWidth_ = cast(int) line.length; scrollbackWidth_ = cast(int) line.length;
scrollbackLength = cast(int) (scrollbackLength + 1 + (scrollbackBuffer[cast(int) scrollbackBuffer.length - 1].length) / screenWidth);
if(line.length > 2 && (line[0].hasNonCharacterData || line[$-1].hasNonCharacterData))
scrollbackLength = scrollbackLength + 1;
else
scrollbackLength = cast(int) (scrollbackLength + 1 + (scrollbackBuffer[cast(int) scrollbackBuffer.length - 1].length) / screenWidth);
notifyScrollbackAdded(); notifyScrollbackAdded();
} }
@ -4180,7 +4207,159 @@ mixin template SdpyImageSupport() {
} }
} }
protected override BrokenUpImage handleBinaryExtensionData(const(ubyte)[] binaryData) { version(TerminalDirectToEmulator)
class NonCharacterData_Widget : NonCharacterData {
this(void* data, size_t idx, int width, int height) {
this.window = cast(SimpleWindow) data;
this.idx = idx;
this.width = width;
this.height = height;
}
void position(int posx, int posy, int width, int height) {
if(posx == this.posx && posy == this.posy && width == this.pixelWidth && height == this.pixelHeight)
return;
this.posx = posx;
this.posy = posy;
this.pixelWidth = width;
this.pixelHeight = height;
window.moveResize(posx, posy, width, height);
import std.stdio; writeln(posx, " ", posy, " ", width, " ", height);
auto painter = this.window.draw;
painter.outlineColor = Color.red;
painter.fillColor = Color.green;
painter.drawRectangle(Point(0, 0), width, height);
}
SimpleWindow window;
size_t idx;
int width;
int height;
int posx;
int posy;
int pixelWidth;
int pixelHeight;
}
private struct CachedImage {
ulong hash;
BinaryDataTerminalRepresentation bui;
int timesSeen;
import core.time;
MonoTime lastUsed;
}
private CachedImage[] imageCache;
private CachedImage* findInCache(ulong hash) {
if(hash == 0)
return null;
/*
import std.stdio;
writeln("***");
foreach(cache; imageCache) {
writeln(cache.hash, " ", cache.timesSeen, " ", cache.lastUsed);
}
*/
foreach(ref i; imageCache)
if(i.hash == hash) {
import core.time;
i.lastUsed = MonoTime.currTime;
i.timesSeen++;
return &i;
}
return null;
}
private BinaryDataTerminalRepresentation addImageCache(ulong hash, BinaryDataTerminalRepresentation bui) {
import core.time;
if(imageCache.length == 0)
imageCache.length = 8;
auto now = MonoTime.currTime;
size_t oldestIndex;
MonoTime oldestTime = now;
size_t leastUsedIndex;
int leastUsedCount = int.max;
foreach(idx, ref cached; imageCache) {
if(cached.hash == 0) {
cached.hash = hash;
cached.bui = bui;
cached.timesSeen = 1;
cached.lastUsed = now;
return bui;
} else {
if(cached.timesSeen < leastUsedCount) {
leastUsedCount = cached.timesSeen;
leastUsedIndex = idx;
}
if(cached.lastUsed < oldestTime) {
oldestTime = cached.lastUsed;
oldestIndex = idx;
}
}
}
// need to overwrite one of the cached items, I'll just use the oldest one here
// but maybe that could be smarter later
imageCache[oldestIndex].hash = hash;
imageCache[oldestIndex].bui = bui;
imageCache[oldestIndex].timesSeen = 1;
imageCache[oldestIndex].lastUsed = now;
return bui;
}
// It has a cache of the 8 most recently used items right now so if there's a loop of 9 you get pwned
// but still the cache does an ok job at helping things while balancing out the big memory consumption it
// could do if just left to grow and grow. i hope.
protected override BinaryDataTerminalRepresentation handleBinaryExtensionData(const(ubyte)[] binaryData) {
version(none) {
//version(TerminalDirectToEmulator)
//if(binaryData.length == size_t.sizeof + 10) {
//if((cast(uint[]) binaryData[0 .. 4])[0] == 0xdeadbeef && (cast(uint[]) binaryData[$-4 .. $])[0] == 0xabcdef32) {
//auto widthInCharacterCells = binaryData[4];
//auto heightInCharacterCells = binaryData[5];
//auto pointer = (cast(void*[]) binaryData[6 .. $-4])[0];
auto widthInCharacterCells = 30;
auto heightInCharacterCells = 20;
SimpleWindow pwin;
foreach(k, v; SimpleWindow.nativeMapping) {
if(v.type == WindowTypes.normal)
pwin = v;
}
auto pointer = cast(void*) (new SimpleWindow(640, 480, null, OpenGlOptions.no, Resizability.automaticallyScaleIfPossible, WindowTypes.nestedChild, WindowFlags.normal, pwin));
BinaryDataTerminalRepresentation bi;
bi.width = widthInCharacterCells;
bi.height = heightInCharacterCells;
bi.representation.length = bi.width * bi.height;
foreach(idx, ref cell; bi.representation) {
cell.nonCharacterData = new NonCharacterData_Widget(pointer, idx, widthInCharacterCells, heightInCharacterCells);
}
return bi;
//}
}
import std.digest.md;
ulong hash = * (cast(ulong*) md5Of(binaryData).ptr);
if(auto cached = findInCache(hash))
return cached.bui;
TrueColorImage mi; TrueColorImage mi;
if(binaryData.length > 8 && binaryData[1] == 'P' && binaryData[2] == 'N' && binaryData[3] == 'G') { if(binaryData.length > 8 && binaryData[1] == 'P' && binaryData[2] == 'N' && binaryData[3] == 'G') {
@ -4196,7 +4375,7 @@ mixin template SdpyImageSupport() {
import arsd.svg; import arsd.svg;
NSVG* image = nsvgParse(cast(const(char)[]) binaryData); NSVG* image = nsvgParse(cast(const(char)[]) binaryData);
if(image is null) if(image is null)
return BrokenUpImage(); return BinaryDataTerminalRepresentation();
int w = cast(int) image.width + 1; int w = cast(int) image.width + 1;
int h = cast(int) image.height + 1; int h = cast(int) image.height + 1;
@ -4205,10 +4384,10 @@ mixin template SdpyImageSupport() {
rasterize(rast, image, 0, 0, 1, mi.imageData.bytes.ptr, w, h, w*4); rasterize(rast, image, 0, 0, 1, mi.imageData.bytes.ptr, w, h, w*4);
image.kill(); image.kill();
} else { } else {
return BrokenUpImage(); return BinaryDataTerminalRepresentation();
} }
BrokenUpImage bi; BinaryDataTerminalRepresentation bi;
bi.width = mi.width / fontWidth + ((mi.width%fontWidth) ? 1 : 0); bi.width = mi.width / fontWidth + ((mi.width%fontWidth) ? 1 : 0);
bi.height = mi.height / fontHeight + ((mi.height%fontHeight) ? 1 : 0); bi.height = mi.height / fontHeight + ((mi.height%fontHeight) ? 1 : 0);
@ -4239,10 +4418,10 @@ mixin template SdpyImageSupport() {
ix = 0; ix = 0;
iy += fontHeight; iy += fontHeight;
} }
} }
return bi; return addImageCache(hash, bi);
//return bi;
} }
} }
@ -4466,8 +4645,7 @@ mixin template SdpyDraw() {
} }
hasBufferedInfo = true; hasBufferedInfo = true;
} catch(Exception e) { } catch(Exception e) {
import std.stdio; // import std.stdio; writeln(cast(uint) cell.ch, " :: ", e.msg);
writeln(cast(uint) cell.ch, " :: ", e.msg);
} }
//} //}
} else if(cell.nonCharacterData !is null) { } else if(cell.nonCharacterData !is null) {
@ -4479,6 +4657,19 @@ mixin template SdpyDraw() {
painter.drawRectangle(Point(posx, posy), fontWidth, fontHeight); painter.drawRectangle(Point(posx, posy), fontWidth, fontHeight);
painter.drawImage(Point(posx, posy), ncdi.data, Point(ncdi.imageOffsetX, ncdi.imageOffsetY), fontWidth, fontHeight); painter.drawImage(Point(posx, posy), ncdi.data, Point(ncdi.imageOffsetX, ncdi.imageOffsetY), fontWidth, fontHeight);
} }
version(TerminalDirectToEmulator)
if(auto wdi = cast(NonCharacterData_Widget) cell.nonCharacterData) {
flushBuffer();
if(wdi.idx == 0) {
wdi.position(posx, posy, fontWidth * wdi.width, fontHeight * wdi.height);
/*
painter.outlineColor = defaultBackground;
painter.fillColor = defaultBackground;
painter.drawRectangle(Point(posx, posy), fontWidth, fontHeight);
*/
}
}
} }
if(!cell.hasNonCharacterData) if(!cell.hasNonCharacterData)