some emscripten basic support

This commit is contained in:
Adam D. Ruppe 2024-10-24 18:04:45 -04:00
parent dd2815bd2b
commit 7901578797
2 changed files with 278 additions and 32 deletions

76
core.d
View File

@ -69,31 +69,33 @@ else
// see: When you only want to track changes on a file or directory, be sure to open it using the O_EVTONLY flag. // see: When you only want to track changes on a file or directory, be sure to open it using the O_EVTONLY flag.
///ArsdUseCustomRuntime is used since other derived work from WebAssembly may be used and thus specified in the CLI ///ArsdUseCustomRuntime is used since other derived work from WebAssembly may be used and thus specified in the CLI
version(WebAssembly) version = ArsdUseCustomRuntime; version(Emscripten) {
version = EmptyEventLoop;
version = EmptyCoreEvent;
version = HasTimer;
} else version(WebAssembly) version = ArsdUseCustomRuntime;
else
// note that kqueue might run an i/o loop on mac, ios, etc. but then NSApp.run on the io thread // note that kqueue might run an i/o loop on mac, ios, etc. but then NSApp.run on the io thread
// but on bsd, you want the kqueue loop in i/o too.... // but on bsd, you want the kqueue loop in i/o too....
version(iOS)
{
version = EmptyEventLoop;
version = EmptyCoreEvent;
}
version(ArsdUseCustomRuntime) version(ArsdUseCustomRuntime)
{ {
version = EmptyEventLoop;
version = UseStdioWriteln; version = UseStdioWriteln;
} }
else else
{ {
version(OSX) version(DigitalMars) { version(D_OpenD) {
version=OSXCocoa; version(OSX)
version=OSXCocoa;
version(iOS)
version=OSXCocoa;
} }
version = HasFile; version = HasFile;
version = HasSocket; version = HasSocket;
version = HasThread; version = HasThread;
version = HasErrno; import core.stdc.errno;
version(Windows) version(Windows)
version = HasTimer; version = HasTimer;
@ -108,26 +110,27 @@ version(HasThread)
import core.thread; import core.thread;
import core.volatile; import core.volatile;
import core.atomic; import core.atomic;
import core.time;
} }
else else
{ {
// polyfill for missing core.time // polyfill for missing core.time
/*
struct Duration { struct Duration {
static Duration max() { return Duration(); } static Duration max() { return Duration(); }
} }
struct MonoTime {}
*/
} }
version(OSX) { import core.time;
version(OSXCocoa) {
version(ArsdNoCocoa) version(ArsdNoCocoa)
enum bool UseCocoa = false; enum bool UseCocoa = false;
else else
enum bool UseCocoa = true; enum bool UseCocoa = true;
} }
version(HasErrno)
import core.stdc.errno;
import core.attribute; import core.attribute;
static if(!__traits(hasMember, core.attribute, "mustuse")) static if(!__traits(hasMember, core.attribute, "mustuse"))
enum mustuse; enum mustuse;
@ -135,7 +138,13 @@ static if(!__traits(hasMember, core.attribute, "mustuse"))
// FIXME: add an arena allocator? can do task local destruction maybe. // FIXME: add an arena allocator? can do task local destruction maybe.
// the three implementations are windows, epoll, and kqueue // the three implementations are windows, epoll, and kqueue
version(Windows) {
version(Emscripten) {
import core.stdc.errno;
import core.atomic;
import core.volatile;
} else version(Windows) {
version=Arsd_core_windows; version=Arsd_core_windows;
// import core.sys.windows.windows; // import core.sys.windows.windows;
@ -182,9 +191,14 @@ version(Windows) {
} else version(OSX) { } else version(OSX) {
version=Arsd_core_kqueue; version=Arsd_core_kqueue;
import core.sys.darwin.sys.event;
} else version(iOS) {
version=Arsd_core_kqueue;
import core.sys.darwin.sys.event; import core.sys.darwin.sys.event;
} }
// FIXME: pragma(linkerDirective, "-framework", "Cocoa") works in ldc
version(OSXCocoa) version(OSXCocoa)
enum CocoaAvailable = true; enum CocoaAvailable = true;
else else
@ -194,9 +208,11 @@ version(Posix) {
import core.sys.posix.signal; import core.sys.posix.signal;
import core.sys.posix.unistd; import core.sys.posix.unistd;
version(Emscripten) {} else {
import core.sys.posix.sys.un; import core.sys.posix.sys.un;
import core.sys.posix.sys.socket; import core.sys.posix.sys.socket;
import core.sys.posix.netinet.in_; import core.sys.posix.netinet.in_;
}
} }
// FIXME: the exceptions should actually give some explanatory text too (at least sometimes) // FIXME: the exceptions should actually give some explanatory text too (at least sometimes)
@ -2934,7 +2950,7 @@ private struct ThreadLocalGcRoots {
+/ +/
// the GC may not be able to see this! remember, it can be hidden inside kernel buffers // the GC may not be able to see this! remember, it can be hidden inside kernel buffers
version(HasThread) package(arsd) class CallbackHelper { package(arsd) class CallbackHelper {
import core.memory; import core.memory;
void call() { void call() {
@ -2946,10 +2962,12 @@ version(HasThread) package(arsd) class CallbackHelper {
void*[3] argsStore; void*[3] argsStore;
void addref() { void addref() {
version(HasThread)
atomicOp!"+="(refcount, 1); atomicOp!"+="(refcount, 1);
} }
void release() { void release() {
version(HasThread)
if(atomicOp!"-="(refcount, 1) <= 0) { if(atomicOp!"-="(refcount, 1) <= 0) {
if(flags & 1) if(flags & 1)
GC.removeRoot(cast(void*) this); GC.removeRoot(cast(void*) this);
@ -2964,6 +2982,7 @@ version(HasThread) package(arsd) class CallbackHelper {
} }
this(void delegate() callback, bool addRoot = true) { this(void delegate() callback, bool addRoot = true) {
version(HasThread)
if(addRoot) { if(addRoot) {
GC.addRoot(cast(void*) this); GC.addRoot(cast(void*) this);
this.flags |= 1; this.flags |= 1;
@ -3428,6 +3447,9 @@ version(HasFile) class AsyncFile : AbstractFile {
} }
} }
else class AsyncFile {
package(arsd) this(NativeFileHandle adoptPreSetup) {}
}
/++ /++
Reads or writes a file in one call. It might internally yield, but is generally blocking if it returns values. The callback ones depend on the implementation. Reads or writes a file in one call. It might internally yield, but is generally blocking if it returns values. The callback ones depend on the implementation.
@ -3811,6 +3833,8 @@ class Timer {
if(handle is null) if(handle is null)
throw new WindowsApiException("CreateWaitableTimer", GetLastError()); throw new WindowsApiException("CreateWaitableTimer", GetLastError());
cbh = new CallbackHelper(&trigger); cbh = new CallbackHelper(&trigger);
} else version(Emscripten) {
assert(0);
} else version(linux) { } else version(linux) {
import core.sys.linux.timerfd; import core.sys.linux.timerfd;
@ -3846,6 +3870,8 @@ class Timer {
initialTime.QuadPart = -intervalInMilliseconds * 10000000L / 1000; // Windows wants hnsecs, we have msecs initialTime.QuadPart = -intervalInMilliseconds * 10000000L / 1000; // Windows wants hnsecs, we have msecs
if(!SetWaitableTimer(handle, &initialTime, repeats ? intervalInMilliseconds : 0, &timerCallback, cast(void*) cbh, false)) if(!SetWaitableTimer(handle, &initialTime, repeats ? intervalInMilliseconds : 0, &timerCallback, cast(void*) cbh, false))
throw new WindowsApiException("SetWaitableTimer", GetLastError()); throw new WindowsApiException("SetWaitableTimer", GetLastError());
} else version(Emscripten) {
assert(0);
} else version(linux) { } else version(linux) {
import core.sys.linux.timerfd; import core.sys.linux.timerfd;
@ -8496,6 +8522,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
NSEventModifierFlagDeviceIndependentFlagsMask = 0xffff0000UL NSEventModifierFlagDeviceIndependentFlagsMask = 0xffff0000UL
} }
version(OSX)
extern class NSEvent : NSObject { extern class NSEvent : NSObject {
NSEventType type() @selector("type"); NSEventType type() @selector("type");
@ -8546,6 +8573,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
alias NSTimeInterval = double; alias NSTimeInterval = double;
version(OSX)
extern class NSResponder : NSObject { extern class NSResponder : NSObject {
NSMenu menu() @selector("menu"); NSMenu menu() @selector("menu");
void menu(NSMenu menu) @selector("setMenu:"); void menu(NSMenu menu) @selector("setMenu:");
@ -8575,6 +8603,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
// touch events should also be here btw among others // touch events should also be here btw among others
} }
version(OSX)
extern class NSApplication : NSResponder { extern class NSApplication : NSResponder {
static NSApplication shared_() @selector("sharedApplication"); static NSApplication shared_() @selector("sharedApplication");
@ -8593,6 +8622,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
void terminate(void*) @selector("terminate:"); void terminate(void*) @selector("terminate:");
} }
version(OSX)
extern interface NSApplicationDelegate { extern interface NSApplicationDelegate {
void applicationWillFinishLaunching(NSNotification notification) @selector("applicationWillFinishLaunching:"); void applicationWillFinishLaunching(NSNotification notification) @selector("applicationWillFinishLaunching:");
void applicationDidFinishLaunching(NSNotification notification) @selector("applicationDidFinishLaunching:"); void applicationDidFinishLaunching(NSNotification notification) @selector("applicationDidFinishLaunching:");
@ -8619,6 +8649,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
NSGraphicsContext graphicsPort() @selector("graphicsPort"); NSGraphicsContext graphicsPort() @selector("graphicsPort");
} }
version(OSX)
extern class NSMenu : NSObject { extern class NSMenu : NSObject {
override static NSMenu alloc() @selector("alloc"); override static NSMenu alloc() @selector("alloc");
@ -8635,6 +8666,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
) @selector("addItemWithTitle:action:keyEquivalent:"); ) @selector("addItemWithTitle:action:keyEquivalent:");
} }
version(OSX)
extern class NSMenuItem : NSObject { extern class NSMenuItem : NSObject {
override static NSMenuItem alloc() @selector("alloc"); override static NSMenuItem alloc() @selector("alloc");
override NSMenuItem init() @selector("init"); override NSMenuItem init() @selector("init");
@ -8682,6 +8714,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
hUDWindow = 1 << 13 // Specifies a heads up display panel hUDWindow = 1 << 13 // Specifies a heads up display panel
} }
version(OSX)
extern class NSWindow : NSObject { extern class NSWindow : NSObject {
override static NSWindow alloc() @selector("alloc"); override static NSWindow alloc() @selector("alloc");
@ -8715,6 +8748,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
void setBackgroundColor(NSColor color) @selector("setBackgroundColor:"); void setBackgroundColor(NSColor color) @selector("setBackgroundColor:");
} }
version(OSX)
extern interface NSWindowDelegate { extern interface NSWindowDelegate {
@optional: @optional:
void windowDidResize(NSNotification notification) @selector("windowDidResize:"); void windowDidResize(NSNotification notification) @selector("windowDidResize:");
@ -8724,8 +8758,9 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
void windowWillClose(NSNotification notification) @selector("windowWillClose:"); void windowWillClose(NSNotification notification) @selector("windowWillClose:");
} }
version(OSX)
extern class NSView : NSResponder { extern class NSView : NSResponder {
override NSView init() @selector("init"); //override NSView init() @selector("init");
NSView initWithFrame(NSRect frameRect) @selector("initWithFrame:"); NSView initWithFrame(NSRect frameRect) @selector("initWithFrame:");
void addSubview(NSView view) @selector("addSubview:"); void addSubview(NSView view) @selector("addSubview:");
@ -8793,6 +8828,7 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
} }
version(OSX)
extern class NSViewController : NSObject { extern class NSViewController : NSObject {
NSView view() @selector("view"); NSView view() @selector("view");
void view(NSView view) @selector("setView:"); void view(NSView view) @selector("setView:");
@ -8910,16 +8946,20 @@ If you are not sure if Cocoa thinks your application is multithreaded or not, yo
extern(C) void NSLog(NSString, ...); extern(C) void NSLog(NSString, ...);
extern(C) SEL sel_registerName(const(char)* str); extern(C) SEL sel_registerName(const(char)* str);
version(OSX)
extern (Objective-C) __gshared NSApplication NSApp_; extern (Objective-C) __gshared NSApplication NSApp_;
version(OSX)
NSApplication NSApp() { NSApplication NSApp() {
if(NSApp_ is null) if(NSApp_ is null)
NSApp_ = NSApplication.shared_; NSApp_ = NSApplication.shared_;
return NSApp_; return NSApp_;
} }
version(DigitalMars) {
// hacks to work around compiler bug // hacks to work around compiler bug
extern(C) __gshared void* _D4arsd4core17NSGraphicsContext7__ClassZ = null; extern(C) __gshared void* _D4arsd4core17NSGraphicsContext7__ClassZ = null;
extern(C) __gshared void* _D4arsd4core6NSView7__ClassZ = null; extern(C) __gshared void* _D4arsd4core6NSView7__ClassZ = null;
extern(C) __gshared void* _D4arsd4core8NSWindow7__ClassZ = null; extern(C) __gshared void* _D4arsd4core8NSWindow7__ClassZ = null;
}
} }

View File

@ -1164,6 +1164,11 @@ unittest {
version(OSX) version(DigitalMars) version=OSXCocoa; version(OSX) version(DigitalMars) version=OSXCocoa;
version(Emscripten) {
version=allow_unimplemented_features;
version=without_opengl;
}
version(OSXCocoa) { version(OSXCocoa) {
version=without_opengl; version=without_opengl;
@ -1186,7 +1191,6 @@ version(without_opengl) {
// enum SdpyIsUsingIVGLBinds = false; // enum SdpyIsUsingIVGLBinds = false;
} }
version(Windows) { version(Windows) {
//import core.sys.windows.windows; //import core.sys.windows.windows;
import core.sys.windows.winnls; import core.sys.windows.winnls;
@ -1309,6 +1313,7 @@ version(Windows) {
pragma(lib, "dwmapi"); pragma(lib, "dwmapi");
} }
} else version(Emscripten) {
} else version (linux) { } else version (linux) {
//k8: this is hack for rdmd. sorry. //k8: this is hack for rdmd. sorry.
static import core.sys.linux.epoll; static import core.sys.linux.epoll;
@ -1388,7 +1393,9 @@ Important Do not use the LOWORD or HIWORD macros to extract the x- and y- coord
*/ */
version(linux) { version(Emscripten) {
} else version(linux) {
version = X11; version = X11;
version(without_libnotify) { version(without_libnotify) {
// we cool // we cool
@ -1497,6 +1504,8 @@ else
/// Does this platform support multiple windows? If not, trying to create another will cause it to throw an exception. /// Does this platform support multiple windows? If not, trying to create another will cause it to throw an exception.
version(Windows) version(Windows)
enum multipleWindowsSupported = true; enum multipleWindowsSupported = true;
else version(Emscripten)
enum multipleWindowsSupported = false;
else version(X11) else version(X11)
enum multipleWindowsSupported = true; enum multipleWindowsSupported = true;
else version(OSXCocoa) else version(OSXCocoa)
@ -2324,6 +2333,8 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
impl.window = nativeWindow; impl.window = nativeWindow;
if(nativeWindow) if(nativeWindow)
display = XDisplayConnection.get(); // get initial display to not segfault display = XDisplayConnection.get(); // get initial display to not segfault
} else version(Emscripten) {
// FIXME
} else version(OSXCocoa) { } else version(OSXCocoa) {
if(nativeWindow !is NullWindow) throw new NotYetImplementedException(); if(nativeWindow !is NullWindow) throw new NotYetImplementedException();
} else featureNotImplemented(); } else featureNotImplemented();
@ -2425,6 +2436,8 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
GetWindowRect(hwnd, &rcClip); GetWindowRect(hwnd, &rcClip);
ClipCursor(&rcClip); ClipCursor(&rcClip);
} }
} else version(Emscripten) {
// nothing necessary
} else version(OSXCocoa) { } else version(OSXCocoa) {
// throw new NotYetImplementedException(); // throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
@ -2545,6 +2558,8 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
ClipCursor(null); ClipCursor(null);
} else version(OSXCocoa) { } else version(OSXCocoa) {
// throw new NotYetImplementedException(); // throw new NotYetImplementedException();
} else version(Emscripten) {
// nothing needed
} else static assert(0); } else static assert(0);
} }
@ -2563,6 +2578,8 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
XSetInputFocus(XDisplayConnection.get, setTo.impl.window, RevertToParent, CurrentTime); XSetInputFocus(XDisplayConnection.get, setTo.impl.window, RevertToParent, CurrentTime);
} else version(Windows) { } else version(Windows) {
SetFocus(this.impl.hwnd); SetFocus(this.impl.hwnd);
} else version(Emscripten) {
throw new NotYetImplementedException();
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
@ -2607,6 +2624,8 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
} else version(X11) { } else version(X11) {
demandingAttention = true; demandingAttention = true;
demandAttention(this, true); demandAttention(this, true);
} else version(Emscripten) {
throw new NotYetImplementedException();
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
@ -2861,6 +2880,7 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
XMapWindow(impl.display, impl.window); XMapWindow(impl.display, impl.window);
} else version(OSXCocoa) { } else version(OSXCocoa) {
// throw new NotYetImplementedException(); // throw new NotYetImplementedException();
} else version(Emscripten) {
} else static assert(0); } else static assert(0);
} }
@ -3380,6 +3400,8 @@ class SimpleWindow : CapableOfHandlingNativeEvent, CapableOfBeingDrawnUpon {
cast(int) buffer.length); cast(int) buffer.length);
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else version(Emscripten) {
throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
} }
@ -4359,6 +4381,11 @@ struct EventLoopImpl {
bool notExited = true; bool notExited = true;
version(Emscripten) {
void delegate(int) signalHandler;
static import unix = core.sys.posix.unistd;
static import err = core.stdc.errno;
} else
version(linux) { version(linux) {
static import ep = core.sys.linux.epoll; static import ep = core.sys.linux.epoll;
static import unix = core.sys.posix.unistd; static import unix = core.sys.posix.unistd;
@ -4370,6 +4397,7 @@ struct EventLoopImpl {
version(X11) { version(X11) {
int pulseFd = -1; int pulseFd = -1;
version(Emscripten) {} else
version(linux) ep.epoll_event[16] events = void; version(linux) ep.epoll_event[16] events = void;
} else version(Windows) { } else version(Windows) {
Timer pulser; Timer pulser;
@ -4417,7 +4445,9 @@ struct EventLoopImpl {
SimpleWindow.processAllCustomEvents(); // process events added before event object creation SimpleWindow.processAllCustomEvents(); // process events added before event object creation
} }
version(linux) { version(Emscripten) {
} else version(linux) {
prepareEventLoop(); prepareEventLoop();
{ {
auto display = XDisplayConnection.get; auto display = XDisplayConnection.get;
@ -4509,7 +4539,8 @@ struct EventLoopImpl {
version(linux) { version(linux) {
this.mtLock(); this.mtLock();
scope(exit) this.mtUnlock(); scope(exit) this.mtUnlock();
XPending(display); // no, really version(X11)
XPending(display); // no, really
} }
disposed = false; disposed = false;
@ -4531,6 +4562,7 @@ struct EventLoopImpl {
pulseFd = -1; pulseFd = -1;
} }
version(Emscripten) {} else
version(linux) version(linux)
if(displayFd != -1) { if(displayFd != -1) {
// clean up xlib fd when we exit, in case we come back later e.g. X disconnect and reconnect with new FD, don't want to still keep the old one around // clean up xlib fd when we exit, in case we come back later e.g. X disconnect and reconnect with new FD, don't want to still keep the old one around
@ -4997,6 +5029,7 @@ struct EventLoopImpl {
If this is an issue, let me know, it'd take about an hour to get it back in there, but I suggest If this is an issue, let me know, it'd take about an hour to get it back in there, but I suggest
you use arsd 10.x when targeting Windows XP. you use arsd 10.x when targeting Windows XP.
+/ +/
version(Emscripten) {} else
version(OSXCocoa) {} else // NotYetImplementedException version(OSXCocoa) {} else // NotYetImplementedException
class NotificationAreaIcon : CapableOfHandlingNativeEvent { class NotificationAreaIcon : CapableOfHandlingNativeEvent {
@ -5960,6 +5993,7 @@ class Timer {
mapping[handle] = this; mapping[handle] = this;
} else version(Emscripten) {
} else version(linux) { } else version(linux) {
static import ep = core.sys.linux.epoll; static import ep = core.sys.linux.epoll;
@ -6015,6 +6049,10 @@ class Timer {
CloseHandle(handle); CloseHandle(handle);
} }
} }
else version(Emscripten)
static void staticDestroy(int fd) @system {
assert(0);
}
else version(linux) else version(linux)
static void staticDestroy(int fd) @system { static void staticDestroy(int fd) @system {
if(fd != -1) { if(fd != -1) {
@ -6295,7 +6333,9 @@ class PosixFdReader {
} }
void ready(uint flags) { void ready(uint flags) {
version(linux) { version(Emscripten) {
assert(0);
} else version(linux) {
static import ep = core.sys.linux.epoll; static import ep = core.sys.linux.epoll;
onReady(fd, (flags & ep.EPOLLIN) ? true : false, (flags & ep.EPOLLOUT) ? true : false); onReady(fd, (flags & ep.EPOLLIN) ? true : false, (flags & ep.EPOLLOUT) ? true : false);
} else { } else {
@ -6369,6 +6409,8 @@ void getClipboardText(SimpleWindow clipboardOwner, void delegate(in char[]) rece
getX11Selection!"CLIPBOARD"(clipboardOwner, receiver); getX11Selection!"CLIPBOARD"(clipboardOwner, receiver);
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else version(Emscripten) {
throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
} }
@ -6404,6 +6446,8 @@ void getClipboardImage()(SimpleWindow clipboardOwner, void delegate(MemoryImage)
getX11Selection!"CLIPBOARD"(clipboardOwner, receiver); getX11Selection!"CLIPBOARD"(clipboardOwner, receiver);
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else version(Emscripten) {
throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
} }
@ -6433,6 +6477,8 @@ void setClipboardText(SimpleWindow clipboardOwner, string text) {
setX11Selection!"CLIPBOARD"(clipboardOwner, text); setX11Selection!"CLIPBOARD"(clipboardOwner, text);
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else version(Emscripten) {
throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
} }
@ -6512,6 +6558,8 @@ void setClipboardImage()(SimpleWindow clipboardOwner, MemoryImage img) {
setX11Selection!"CLIPBOARD"(clipboardOwner, new X11SetSelectionHandler_Image(img)); setX11Selection!"CLIPBOARD"(clipboardOwner, new X11SetSelectionHandler_Image(img));
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else version(Emscripten) {
throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
} }
@ -8015,6 +8063,8 @@ struct MouseEvent {
return p; return p;
} else version(OSXCocoa) { } else version(OSXCocoa) {
throw new NotYetImplementedException(); throw new NotYetImplementedException();
} else version(Emscripten) {
throw new NotYetImplementedException();
} else static assert(0); } else static assert(0);
} }
@ -8428,6 +8478,8 @@ final class Image {
} }
} else version(OSXCocoa) { } else version(OSXCocoa) {
return 0 ; //throw new NotYetImplementedException(); return 0 ; //throw new NotYetImplementedException();
} else version(Emscripten) {
return 0;
} else static assert(0, "fill in this info for other OSes"); } else static assert(0, "fill in this info for other OSes");
} }
@ -8449,7 +8501,9 @@ final class Image {
return offset; return offset;
} }
} else version(OSXCocoa) { } else version(OSXCocoa) {
return 0 ; //throw new NotYetImplementedException(); return (y * width + x) * 4 ; //throw new NotYetImplementedException();
} else version(Emscripten) {
return (y * width + x) * 4 ; //throw new NotYetImplementedException();
} else static assert(0, "fill in this info for other OSes"); } else static assert(0, "fill in this info for other OSes");
} }
@ -8464,7 +8518,9 @@ final class Image {
else else
return -((cast(int) width * 3 + 3) / 4) * 4; return -((cast(int) width * 3 + 3) / 4) * 4;
} else version(OSXCocoa) { } else version(OSXCocoa) {
return 0 ; //throw new NotYetImplementedException(); return width * 4 ; //throw new NotYetImplementedException();
} else version(Emscripten) {
return width * 4 ; //throw new NotYetImplementedException();
} else static assert(0, "fill in this info for other OSes"); } else static assert(0, "fill in this info for other OSes");
} }
@ -8475,7 +8531,9 @@ final class Image {
} else version(Windows) { } else version(Windows) {
return 2; return 2;
} else version(OSXCocoa) { } else version(OSXCocoa) {
return 0 ; //throw new NotYetImplementedException(); return 2 ; //throw new NotYetImplementedException();
} else version(Emscripten) {
return 2 ; //throw new NotYetImplementedException();
} else static assert(0, "fill in this info for other OSes"); } else static assert(0, "fill in this info for other OSes");
} }
@ -8486,7 +8544,9 @@ final class Image {
} else version(Windows) { } else version(Windows) {
return 1; return 1;
} else version(OSXCocoa) { } else version(OSXCocoa) {
return 0 ; //throw new NotYetImplementedException(); return 1 ; //throw new NotYetImplementedException();
} else version(Emscripten) {
return 1 ; //throw new NotYetImplementedException();
} else static assert(0, "fill in this info for other OSes"); } else static assert(0, "fill in this info for other OSes");
} }
@ -8498,6 +8558,8 @@ final class Image {
return 0; return 0;
} else version(OSXCocoa) { } else version(OSXCocoa) {
return 0 ; //throw new NotYetImplementedException(); return 0 ; //throw new NotYetImplementedException();
} else version(Emscripten) {
return 0 ; //throw new NotYetImplementedException();
} else static assert(0, "fill in this info for other OSes"); } else static assert(0, "fill in this info for other OSes");
} }
@ -8509,6 +8571,8 @@ final class Image {
return 3; return 3;
} else version(OSXCocoa) { } else version(OSXCocoa) {
return 3; //throw new NotYetImplementedException(); return 3; //throw new NotYetImplementedException();
} else version(Emscripten) {
return 3 ; //throw new NotYetImplementedException();
} else static assert(0, "fill in this info for other OSes"); } else static assert(0, "fill in this info for other OSes");
} }
} }
@ -8727,7 +8791,9 @@ class OperatingSystemFont : MeasurableFont {
// FIXME: when the X Connection is lost, these need to be invalidated! // FIXME: when the X Connection is lost, these need to be invalidated!
// that means I need to store the original stuff again to reconstruct it too. // that means I need to store the original stuff again to reconstruct it too.
version(X11) { version(Emscripten) {
void* font;
} else version(X11) {
XFontStruct* font; XFontStruct* font;
XFontSet fontset; XFontSet fontset;
@ -10323,6 +10389,8 @@ class Sprite : CapableOfBeingDrawnUpon {
HBITMAP handle; HBITMAP handle;
else version(OSXCocoa) else version(OSXCocoa)
CGContextRef handle; CGContextRef handle;
else version(Emscripten)
void* handle;
else static assert(0); else static assert(0);
} }
@ -11039,6 +11107,19 @@ enum ModifierState : uint {
middleButtonDown = 512, /// ditto middleButtonDown = 512, /// ditto
rightButtonDown = 1024, /// ditto rightButtonDown = 1024, /// ditto
} }
else version(Emscripten)
enum ModifierState : uint {
shift = 1, ///
capsLock = 2, ///
ctrl = 4, ///
alt = 8, /// Not always available on Windows
windows = 64, /// ditto
numLock = 16, ///
leftButtonDown = 256, /// these aren't available on Windows for key events, so don't use them for that unless your app is X only.
middleButtonDown = 512, /// ditto
rightButtonDown = 1024, /// ditto
}
else version(Windows) else version(Windows)
/// ditto /// ditto
enum ModifierState : uint { enum ModifierState : uint {
@ -11097,7 +11178,118 @@ enum MouseButtonLinear : ubyte {
forwardButton, /// often found on the thumb and used for forward in browsers forwardButton, /// often found on the thumb and used for forward in browsers
} }
version(X11) { version(WebAssembly) {
/// Do not trust the numeric values as they are platform-specific. Always use the symbolic name.
enum Key {
Escape = 0xff1b, ///
F1 = 0xffbe, ///
F2 = 0xffbf, ///
F3 = 0xffc0, ///
F4 = 0xffc1, ///
F5 = 0xffc2, ///
F6 = 0xffc3, ///
F7 = 0xffc4, ///
F8 = 0xffc5, ///
F9 = 0xffc6, ///
F10 = 0xffc7, ///
F11 = 0xffc8, ///
F12 = 0xffc9, ///
PrintScreen = 0xff61, ///
ScrollLock = 0xff14, ///
Pause = 0xff13, ///
Grave = 0x60, /// The $(BACKTICK) ~ key
// number keys across the top of the keyboard
N1 = 0x31, /// Number key atop the keyboard
N2 = 0x32, ///
N3 = 0x33, ///
N4 = 0x34, ///
N5 = 0x35, ///
N6 = 0x36, ///
N7 = 0x37, ///
N8 = 0x38, ///
N9 = 0x39, ///
N0 = 0x30, ///
Dash = 0x2d, ///
Equals = 0x3d, ///
Backslash = 0x5c, /// The \ | key
Backspace = 0xff08, ///
Insert = 0xff63, ///
Home = 0xff50, ///
PageUp = 0xff55, ///
Delete = 0xffff, ///
End = 0xff57, ///
PageDown = 0xff56, ///
Up = 0xff52, ///
Down = 0xff54, ///
Left = 0xff51, ///
Right = 0xff53, ///
Tab = 0xff09, ///
Q = 0x71, ///
W = 0x77, ///
E = 0x65, ///
R = 0x72, ///
T = 0x74, ///
Y = 0x79, ///
U = 0x75, ///
I = 0x69, ///
O = 0x6f, ///
P = 0x70, ///
LeftBracket = 0x5b, /// the [ { key
RightBracket = 0x5d, /// the ] } key
CapsLock = 0xffe5, ///
A = 0x61, ///
S = 0x73, ///
D = 0x64, ///
F = 0x66, ///
G = 0x67, ///
H = 0x68, ///
J = 0x6a, ///
K = 0x6b, ///
L = 0x6c, ///
Semicolon = 0x3b, ///
Apostrophe = 0x27, ///
Enter = 0xff0d, ///
Shift = 0xffe1, ///
Z = 0x7a, ///
X = 0x78, ///
C = 0x63, ///
V = 0x76, ///
B = 0x62, ///
N = 0x6e, ///
M = 0x6d, ///
Comma = 0x2c, ///
Period = 0x2e, ///
Slash = 0x2f, /// the / ? key
Shift_r = 0xffe2, /// Note: this isn't sent on all computers, sometimes it just sends Shift, so don't rely on it. If it is supported though, it is the right Shift key, as opposed to the left Shift key
Ctrl = 0xffe3, ///
Windows = 0xffeb, ///
Alt = 0xffe9, ///
Space = 0x20, ///
Alt_r = 0xffea, /// ditto of shift_r
Windows_r = 0xffec, ///
Menu = 0xff67, ///
Ctrl_r = 0xffe4, ///
NumLock = 0xff7f, ///
Divide = 0xffaf, /// The / key on the number pad
Multiply = 0xffaa, /// The * key on the number pad
Minus = 0xffad, /// The - key on the number pad
Plus = 0xffab, /// The + key on the number pad
PadEnter = 0xff8d, /// Numberpad enter key
Pad1 = 0xff9c, /// Numberpad keys
Pad2 = 0xff99, ///
Pad3 = 0xff9b, ///
Pad4 = 0xff96, ///
Pad5 = 0xff9d, ///
Pad6 = 0xff98, ///
Pad7 = 0xff95, ///
Pad8 = 0xff97, ///
Pad9 = 0xff9a, ///
Pad0 = 0xff9e, ///
PadDot = 0xff9f, ///
}
} version(X11) {
// FIXME: match ASCII whenever we can. Most of it is already there, // FIXME: match ASCII whenever we can. Most of it is already there,
// but there's a few exceptions and mismatches with Windows // but there's a few exceptions and mismatches with Windows
@ -13118,6 +13310,16 @@ version(Windows) {
enum KEY_ESCAPE = 27; enum KEY_ESCAPE = 27;
} }
version(Emscripten) {
alias int delegate(void*) NativeEventHandler;
alias void* NativeWindowHandle;
mixin template NativeSimpleWindowImplementation() { }
mixin template NativeScreenPainterImplementation() { }
mixin template NativeImageImplementation() { }
}
version(X11) { version(X11) {
/// This is the default font used. You might change this before doing anything else with /// This is the default font used. You might change this before doing anything else with
/// the library if you want to try something else. Surround that in `static if(UsingSimpledisplayX11)` /// the library if you want to try something else. Surround that in `static if(UsingSimpledisplayX11)`
@ -14802,8 +15004,9 @@ mixin DynamicLoad!(XRandr, "Xrandr", 2, XRandrLibrarySuccessfullyLoaded) XRandrL
else else
isLocal_ = true; isLocal_ = true;
XSetErrorHandler(&adrlogger);
debug(sdpy_x_errors) { debug(sdpy_x_errors) {
XSetErrorHandler(&adrlogger);
XSynchronize(display, true); XSynchronize(display, true);
extern(C) int wtf() { extern(C) int wtf() {
@ -16548,6 +16751,7 @@ version(X11) {
// Necessary C library bindings follow // Necessary C library bindings follow
version(Windows) {} else version(Windows) {} else
version(Emscripten) {} else
version(X11) { version(X11) {
extern(C) int eventfd (uint initval, int flags) nothrow @trusted @nogc; extern(C) int eventfd (uint initval, int flags) nothrow @trusted @nogc;
@ -20255,7 +20459,9 @@ private string typeToGl(T)() {
} }
version(linux) { version(Emscripten) {
} else version(linux) {
version(with_eventloop) {} else { version(with_eventloop) {} else {
private int epollFd = -1; private int epollFd = -1;
void prepareEventLoop() { void prepareEventLoop() {
@ -22705,7 +22911,7 @@ private int doDragDropX11(SimpleWindow window, X11SetSelectionHandler handler, D
timeval tv; timeval tv;
gettimeofday(&tv, null); gettimeofday(&tv, null);
Time dataTimestamp = tv.tv_sec * 1000 + tv.tv_usec / 1000; Time dataTimestamp = cast(Time) ( tv.tv_sec * 1000 + tv.tv_usec / 1000 );
Time lastMouseTimestamp; Time lastMouseTimestamp;