2019-12-28 12:16:32 +01:00
|
|
|
#include <stdio.h>
|
2021-06-03 21:48:15 +02:00
|
|
|
#include <stdlib.h>
|
2020-06-29 15:24:08 +02:00
|
|
|
#include <SDL.h>
|
2019-12-28 12:16:32 +01:00
|
|
|
#include "api/api.h"
|
2021-05-05 09:32:24 +02:00
|
|
|
#include "rencache.h"
|
2019-12-28 12:16:32 +01:00
|
|
|
#include "renderer.h"
|
2019-12-31 11:49:07 +01:00
|
|
|
|
2019-12-28 12:16:32 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <windows.h>
|
2019-12-31 11:49:07 +01:00
|
|
|
#elif __linux__
|
|
|
|
#include <unistd.h>
|
2021-04-02 16:43:21 +02:00
|
|
|
#include <SDL_syswm.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xatom.h>
|
|
|
|
#include <X11/Xresource.h>
|
2021-06-02 17:27:40 +02:00
|
|
|
#include <signal.h>
|
2020-06-11 16:05:39 +02:00
|
|
|
#elif __APPLE__
|
|
|
|
#include <mach-o/dyld.h>
|
2019-12-28 12:16:32 +01:00
|
|
|
#endif
|
|
|
|
|
2021-07-12 18:21:27 +02:00
|
|
|
#define DMON_IMPL
|
|
|
|
#include "dmon.h"
|
|
|
|
|
2019-12-28 12:16:32 +01:00
|
|
|
|
|
|
|
SDL_Window *window;
|
2021-07-12 18:21:27 +02:00
|
|
|
Uint32 dmon_event_no;
|
2019-12-28 12:16:32 +01:00
|
|
|
|
2019-12-31 11:49:07 +01:00
|
|
|
static double get_scale(void) {
|
2021-04-02 16:43:21 +02:00
|
|
|
#ifdef _WIN32
|
2019-12-31 11:49:07 +01:00
|
|
|
float dpi;
|
|
|
|
SDL_GetDisplayDPI(0, NULL, &dpi, NULL);
|
|
|
|
return dpi / 96.0;
|
2021-04-02 16:43:21 +02:00
|
|
|
#elif __linux__
|
|
|
|
SDL_SysWMinfo info;
|
|
|
|
XrmDatabase db;
|
|
|
|
XrmValue value;
|
|
|
|
char *type = NULL;
|
|
|
|
|
|
|
|
SDL_VERSION(&info.version);
|
|
|
|
if (!SDL_GetWindowWMInfo(window, &info)
|
|
|
|
|| info.subsystem != SDL_SYSWM_X11)
|
|
|
|
return 1.0;
|
|
|
|
|
|
|
|
char *resource = XResourceManagerString(info.info.x11.display);
|
|
|
|
if (resource == NULL)
|
|
|
|
return 1.0;
|
2021-04-21 09:52:16 +02:00
|
|
|
|
2021-04-02 16:43:21 +02:00
|
|
|
XrmInitialize();
|
|
|
|
db = XrmGetStringDatabase(resource);
|
|
|
|
if (XrmGetResource(db, "Xft.dpi", "String", &type, &value) == False
|
|
|
|
|| value.addr == NULL)
|
|
|
|
return 1.0;
|
|
|
|
|
|
|
|
return atof(value.addr) / 96.0;
|
2019-12-31 11:49:07 +01:00
|
|
|
#else
|
|
|
|
return 1.0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-17 10:36:46 +02:00
|
|
|
static void get_exe_filename(char *buf, int sz) {
|
2019-12-31 11:49:07 +01:00
|
|
|
#if _WIN32
|
|
|
|
int len = GetModuleFileName(NULL, buf, sz - 1);
|
|
|
|
buf[len] = '\0';
|
|
|
|
#elif __linux__
|
|
|
|
char path[512];
|
|
|
|
sprintf(path, "/proc/%d/exe", getpid());
|
|
|
|
int len = readlink(path, buf, sz - 1);
|
|
|
|
buf[len] = '\0';
|
|
|
|
#elif __APPLE__
|
2021-06-03 21:48:15 +02:00
|
|
|
/* use realpath to resolve a symlink if the process was launched from one.
|
|
|
|
** This happens when Homebrew installs a cack and creates a symlink in
|
|
|
|
** /usr/loca/bin for launching the executable from the command line. */
|
2019-12-31 11:49:07 +01:00
|
|
|
unsigned size = sz;
|
2021-06-03 21:48:15 +02:00
|
|
|
char exepath[size];
|
|
|
|
_NSGetExecutablePath(exepath, &size);
|
|
|
|
realpath(exepath, buf);
|
2019-12-31 11:49:07 +01:00
|
|
|
#else
|
2020-05-17 10:36:46 +02:00
|
|
|
strcpy(buf, "./lite");
|
2019-12-31 11:49:07 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-06 20:49:34 +02:00
|
|
|
static void init_window_icon(void) {
|
|
|
|
#ifndef _WIN32
|
2021-06-20 17:37:08 +02:00
|
|
|
#include "../resources/icons/icon.inl"
|
2020-05-06 20:49:34 +02:00
|
|
|
(void) icon_rgba_len; /* unused */
|
|
|
|
SDL_Surface *surf = SDL_CreateRGBSurfaceFrom(
|
|
|
|
icon_rgba, 64, 64,
|
|
|
|
32, 64 * 4,
|
|
|
|
0x000000ff,
|
|
|
|
0x0000ff00,
|
|
|
|
0x00ff0000,
|
|
|
|
0xff000000);
|
|
|
|
SDL_SetWindowIcon(window, surf);
|
|
|
|
SDL_FreeSurface(surf);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-03-05 09:42:48 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
#define LITE_OS_HOME "USERPROFILE"
|
|
|
|
#define LITE_PATHSEP_PATTERN "\\\\"
|
2021-03-05 23:46:41 +01:00
|
|
|
#define LITE_NONPATHSEP_PATTERN "[^\\\\]+"
|
2021-03-05 09:42:48 +01:00
|
|
|
#else
|
|
|
|
#define LITE_OS_HOME "HOME"
|
|
|
|
#define LITE_PATHSEP_PATTERN "/"
|
2021-03-05 23:46:41 +01:00
|
|
|
#define LITE_NONPATHSEP_PATTERN "[^/]+"
|
2021-03-05 09:42:48 +01:00
|
|
|
#endif
|
2019-12-31 11:49:07 +01:00
|
|
|
|
2021-04-19 09:52:00 +02:00
|
|
|
#ifdef __APPLE__
|
|
|
|
void set_macos_bundle_resources(lua_State *L);
|
2021-04-21 09:52:16 +02:00
|
|
|
void enable_momentum_scroll();
|
2021-04-19 09:52:00 +02:00
|
|
|
#endif
|
|
|
|
|
2019-12-28 12:16:32 +01:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
#ifdef _WIN32
|
|
|
|
HINSTANCE lib = LoadLibrary("user32.dll");
|
|
|
|
int (*SetProcessDPIAware)() = (void*) GetProcAddress(lib, "SetProcessDPIAware");
|
|
|
|
SetProcessDPIAware();
|
2021-06-02 17:27:40 +02:00
|
|
|
#else
|
|
|
|
signal(SIGPIPE, SIG_IGN);
|
2019-12-28 12:16:32 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
SDL_Init(SDL_INIT_VIDEO | SDL_INIT_EVENTS);
|
|
|
|
SDL_EnableScreenSaver();
|
|
|
|
SDL_EventState(SDL_DROPFILE, SDL_ENABLE);
|
|
|
|
atexit(SDL_Quit);
|
2020-06-03 15:05:55 +02:00
|
|
|
|
|
|
|
#ifdef SDL_HINT_VIDEO_X11_NET_WM_BYPASS_COMPOSITOR /* Available since 2.0.8 */
|
|
|
|
SDL_SetHint(SDL_HINT_VIDEO_X11_NET_WM_BYPASS_COMPOSITOR, "0");
|
|
|
|
#endif
|
2019-12-28 12:16:32 +01:00
|
|
|
#if SDL_VERSION_ATLEAST(2, 0, 5)
|
|
|
|
SDL_SetHint(SDL_HINT_MOUSE_FOCUS_CLICKTHROUGH, "1");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
SDL_DisplayMode dm;
|
|
|
|
SDL_GetCurrentDisplayMode(0, &dm);
|
|
|
|
|
2021-07-12 18:21:27 +02:00
|
|
|
dmon_init();
|
|
|
|
dmon_event_no = SDL_RegisterEvents(1);
|
|
|
|
if (dmon_event_no == (Uint32)-1) {
|
|
|
|
fprintf(stderr, "internal error registering SDL dmon event.\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2019-12-28 12:16:32 +01:00
|
|
|
window = SDL_CreateWindow(
|
2020-05-14 23:28:22 +02:00
|
|
|
"", SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, dm.w * 0.8, dm.h * 0.8,
|
|
|
|
SDL_WINDOW_RESIZABLE | SDL_WINDOW_ALLOW_HIGHDPI | SDL_WINDOW_HIDDEN);
|
2020-05-06 20:49:34 +02:00
|
|
|
init_window_icon();
|
2019-12-28 12:16:32 +01:00
|
|
|
ren_init(window);
|
|
|
|
|
2020-12-10 12:44:01 +01:00
|
|
|
lua_State *L;
|
|
|
|
init_lua:
|
|
|
|
L = luaL_newstate();
|
2019-12-28 12:16:32 +01:00
|
|
|
luaL_openlibs(L);
|
|
|
|
api_load_libs(L);
|
|
|
|
|
|
|
|
|
|
|
|
lua_newtable(L);
|
|
|
|
for (int i = 0; i < argc; i++) {
|
|
|
|
lua_pushstring(L, argv[i]);
|
|
|
|
lua_rawseti(L, -2, i + 1);
|
|
|
|
}
|
2020-04-25 10:57:35 +02:00
|
|
|
lua_setglobal(L, "ARGS");
|
|
|
|
|
2020-04-23 21:03:14 +02:00
|
|
|
lua_pushstring(L, SDL_GetPlatform());
|
2020-04-25 10:57:35 +02:00
|
|
|
lua_setglobal(L, "PLATFORM");
|
2019-12-28 12:16:32 +01:00
|
|
|
|
2019-12-31 11:49:07 +01:00
|
|
|
lua_pushnumber(L, get_scale());
|
2020-04-25 10:57:35 +02:00
|
|
|
lua_setglobal(L, "SCALE");
|
2019-12-28 12:16:32 +01:00
|
|
|
|
2020-05-17 10:36:46 +02:00
|
|
|
char exename[2048];
|
|
|
|
get_exe_filename(exename, sizeof(exename));
|
|
|
|
lua_pushstring(L, exename);
|
|
|
|
lua_setglobal(L, "EXEFILE");
|
2019-12-31 11:49:07 +01:00
|
|
|
|
2021-04-18 17:51:31 +02:00
|
|
|
#ifdef __APPLE__
|
|
|
|
set_macos_bundle_resources(L);
|
2021-04-21 09:52:16 +02:00
|
|
|
enable_momentum_scroll();
|
2021-04-18 17:51:31 +02:00
|
|
|
#endif
|
|
|
|
|
2020-12-10 12:44:01 +01:00
|
|
|
const char *init_lite_code = \
|
2019-12-28 12:16:32 +01:00
|
|
|
"local core\n"
|
|
|
|
"xpcall(function()\n"
|
2021-03-05 09:42:48 +01:00
|
|
|
" HOME = os.getenv('" LITE_OS_HOME "')\n"
|
2021-03-05 23:46:41 +01:00
|
|
|
" local exedir = EXEFILE:match(\"^(.*)" LITE_PATHSEP_PATTERN LITE_NONPATHSEP_PATTERN "$\")\n"
|
|
|
|
" local prefix = exedir:match(\"^(.*)" LITE_PATHSEP_PATTERN "bin$\")\n"
|
2021-04-18 17:51:31 +02:00
|
|
|
" dofile((MACOS_RESOURCES or (prefix and prefix .. '/share/lite-xl' or exedir .. '/data')) .. '/core/start.lua')\n"
|
2019-12-28 12:16:32 +01:00
|
|
|
" core = require('core')\n"
|
|
|
|
" core.init()\n"
|
|
|
|
" core.run()\n"
|
|
|
|
"end, function(err)\n"
|
2021-03-05 23:46:41 +01:00
|
|
|
" local error_dir\n"
|
2021-03-07 09:48:44 +01:00
|
|
|
" io.stdout:write('Error: '..tostring(err)..'\\n')\n"
|
|
|
|
" io.stdout:write(debug.traceback(nil, 4)..'\\n')\n"
|
2019-12-28 12:16:32 +01:00
|
|
|
" if core and core.on_error then\n"
|
2021-03-05 23:46:41 +01:00
|
|
|
" error_dir=USERDIR\n"
|
2019-12-28 12:16:32 +01:00
|
|
|
" pcall(core.on_error, err)\n"
|
2021-03-05 23:46:41 +01:00
|
|
|
" else\n"
|
|
|
|
" error_dir=system.absolute_path('.')\n"
|
|
|
|
" local fp = io.open('error.txt', 'wb')\n"
|
|
|
|
" fp:write('Error: ' .. tostring(err) .. '\\n')\n"
|
2021-03-07 09:48:44 +01:00
|
|
|
" fp:write(debug.traceback(nil, 4)..'\\n')\n"
|
2021-03-05 23:46:41 +01:00
|
|
|
" fp:close()\n"
|
2019-12-28 12:16:32 +01:00
|
|
|
" end\n"
|
2021-03-05 23:46:41 +01:00
|
|
|
" system.show_fatal_error('Lite XL internal error',\n"
|
|
|
|
" 'An internal error occurred in a critical part of the application.\\n\\n'..\n"
|
|
|
|
" 'Please verify the file \\\"error.txt\\\" in the directory '..error_dir)\n"
|
2019-12-28 12:16:32 +01:00
|
|
|
" os.exit(1)\n"
|
2020-12-10 12:44:01 +01:00
|
|
|
"end)\n"
|
|
|
|
"return core and core.restart_request\n";
|
2019-12-28 12:16:32 +01:00
|
|
|
|
2020-12-10 12:44:01 +01:00
|
|
|
if (luaL_loadstring(L, init_lite_code)) {
|
|
|
|
fprintf(stderr, "internal error when starting the application\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
lua_pcall(L, 0, 1, 0);
|
|
|
|
if (lua_toboolean(L, -1)) {
|
|
|
|
lua_close(L);
|
|
|
|
goto init_lua;
|
|
|
|
}
|
2019-12-28 12:16:32 +01:00
|
|
|
|
|
|
|
lua_close(L);
|
2021-04-23 14:54:25 +02:00
|
|
|
ren_free_window_resources();
|
2021-07-12 18:21:27 +02:00
|
|
|
dmon_deinit();
|
2019-12-28 12:16:32 +01:00
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|