shithub: choc

Download patch

ref: 3ac144025a055a3b85c1f4b2ef05b6f2892788f5
parent: adbfa3216cf3ef6f71cc5b6b22c25a04ae737212
author: Simon Howard <fraggle@gmail.com>
date: Sun Mar 31 14:46:25 EDT 2013

Add file selector widget to textscreen library.

Subversion-branch: /trunk/chocolate-doom
Subversion-revision: 2570

--- a/textscreen/Makefile.am
+++ b/textscreen/Makefile.am
@@ -16,6 +16,7 @@
 	txt_checkbox.c           txt_checkbox.h           \
 	txt_desktop.c            txt_desktop.h            \
 	txt_dropdown.c           txt_dropdown.h           \
+	txt_fileselect.c         txt_fileselect.h         \
 	txt_gui.c                txt_gui.h                \
 	txt_inputbox.c           txt_inputbox.h           \
 	txt_io.c                 txt_io.h                 \
--- a/textscreen/examples/guitest.c
+++ b/textscreen/examples/guitest.c
@@ -39,10 +39,13 @@
     RADIO_VALUE_MUSHROOM,
     RADIO_VALUE_SNAKE,
 };
+char *extensions[] = { "wad", "lmp", "txt", NULL };
 char *radio_values[] = { "Badger", "Mushroom", "Snake" };
 char *textbox_value = NULL;
 int numbox_value = 0;
 int radiobutton_value;
+char *file_path = NULL;
+char *dir_path = NULL;
 txt_label_t *value_label;
 txt_window_t *firstwin;
 int cheesy;
@@ -187,12 +190,20 @@
     TXT_AddWidget(window, TXT_NewSeparator("Input boxes"));
     table = TXT_NewTable(2);
     TXT_AddWidget(window, table);
-    TXT_AddWidget(table, TXT_NewLabel("String: "));
-    TXT_AddWidget(table, TXT_NewInputBox(&textbox_value, 20));
-    TXT_AddWidget(table, TXT_NewLabel("Int: "));
-    TXT_AddWidget(table, TXT_NewIntInputBox(&numbox_value, 10));
-    TXT_AddWidget(table, TXT_NewLabel("Spin control:"));
-    TXT_AddWidget(table, TXT_NewSpinControl(&numbox_value, 0, 15));
+    TXT_AddWidgets(table,
+                   TXT_NewLabel("String: "),
+                   TXT_NewInputBox(&textbox_value, 20),
+                   TXT_NewLabel("Int: "),
+                   TXT_NewIntInputBox(&numbox_value, 10),
+                   TXT_NewLabel("Spin control:"),
+                   TXT_NewSpinControl(&numbox_value, 0, 15),
+                   TXT_NewLabel("File:"),
+                   TXT_NewFileSelector(&file_path, 28, "Select file:",
+                                       extensions),
+                   TXT_NewLabel("Directory:"),
+                   TXT_NewFileSelector(&dir_path, 28, "Select directory:",
+                                       TXT_DIRECTORY),
+                   NULL);
 
     TXT_AddWidget(window, TXT_NewSeparator("Scroll pane test"));
     scrollpane = TXT_NewScrollPane(40, 5, TXT_NewLabel(
--- a/textscreen/textscreen.h
+++ b/textscreen/textscreen.h
@@ -29,6 +29,7 @@
 #include "txt_checkbox.h"
 #include "txt_desktop.h"
 #include "txt_dropdown.h"
+#include "txt_fileselect.h"
 #include "txt_inputbox.h"
 #include "txt_label.h"
 #include "txt_radiobutton.h"
--- /dev/null
+++ b/textscreen/txt_fileselect.c
@@ -1,0 +1,678 @@
+// Emacs style mode select   -*- C++ -*-
+//-----------------------------------------------------------------------------
+//
+// Copyright(C) 2013 Simon Howard
+//
+// This program 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 2
+// of the License, or (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program; if not, write to the Free Software
+// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
+// 02111-1307, USA.
+//
+//-----------------------------------------------------------------------------
+//
+// Routines for selecting files.
+//
+//-----------------------------------------------------------------------------
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "doomkeys.h"
+
+#include "txt_fileselect.h"
+#include "txt_inputbox.h"
+#include "txt_main.h"
+#include "txt_widget.h"
+
+struct txt_fileselect_s {
+    txt_widget_t widget;
+    txt_inputbox_t *inputbox;
+    int size;
+    char *prompt;
+    char **extensions;
+};
+
+// Dummy value to select a directory.
+
+char *TXT_DIRECTORY[] = { "__directory__", NULL };
+
+#ifndef _WIN32
+
+#include <fcntl.h>
+#include <unistd.h>
+#include <errno.h>
+#include <sys/wait.h>
+
+static char *ExecReadOutput(char **argv)
+{
+    char *result;
+    int completed;
+    int pid, status, result_len;
+    int pipefd[2];
+
+    if (pipe(pipefd) != 0)
+    {
+        return NULL;
+    }
+
+    pid = fork();
+
+    if (pid == 0)
+    {
+        dup2(pipefd[1], fileno(stdout));
+        execv(argv[0], argv);
+        exit(-1);
+    }
+
+    fcntl(pipefd[0], F_SETFL, O_NONBLOCK);
+
+    // Read program output into 'result' string.
+    // Wait until the program has completed and (if it was successful)
+    // a full line has been read.
+
+    result = NULL;
+    result_len = 0;
+    completed = 0;
+
+    while (!completed
+        || (status == 0 && (result == NULL || strchr(result, '\n') == NULL)))
+    {
+        char buf[64];
+        int bytes;
+
+        if (!completed && waitpid(pid, &status, WNOHANG) != 0)
+        {
+            completed = 1;
+        }
+
+        bytes = read(pipefd[0], buf, sizeof(buf));
+
+        if (bytes < 0)
+        {
+            if (errno != EAGAIN && errno != EWOULDBLOCK)
+            {
+                status = -1;
+                break;
+            }
+        }
+        else
+        {
+            result = realloc(result, result_len + bytes + 1);
+            memcpy(result + result_len, buf, bytes);
+            result_len += bytes;
+            result[result_len] = '\0';
+        }
+
+        TXT_Sleep(25);
+        TXT_UpdateScreen();
+    }
+
+    close(pipefd[0]);
+    close(pipefd[1]);
+
+    // Must have a success exit code.
+
+    if (WEXITSTATUS(status) != 0)
+    {
+        free(result);
+        result = NULL;
+    }
+
+    // Strip off newline from the end.
+
+    if (result != NULL && result[result_len - 1] == '\n')
+    {
+        result[result_len - 1] = '\0';
+    }
+
+    return result;
+}
+
+#endif
+
+#if defined(_WIN32)
+
+// Windows code. Use comdlg32 to pop up a dialog box.
+
+#include <windows.h>
+#include <shlobj.h>
+
+static BOOL WINAPI (*MyGetOpenFileName)(LPOPENFILENAME) = NULL;
+static PIDLIST_ABSOLUTE (*MySHBrowseForFolder)(LPBROWSEINFO) = NULL;
+static BOOL (*MySHGetPathFromIDList)(PCIDLIST_ABSOLUTE, LPTSTR) = NULL;
+
+// Load library functions from DLL files.
+
+static int LoadDLLs(void)
+{
+    HMODULE comdlg32 = LoadLibraryW(L"comdlg32.dll");
+    HMODULE shell32 = LoadLibraryW(L"shell32.dll");
+
+    if (comdlg32 == NULL || shell32 == NULL)
+    {
+        return 0;
+    }
+
+    MyGetOpenFileName =
+        (void *) GetProcAddress(comdlg32, "GetOpenFileNameA");
+    MySHBrowseForFolder =
+        (void *) GetProcAddress(shell32, "SHBrowseForFolder");
+    MySHGetPathFromIDList =
+        (void *) GetProcAddress(shell32, "SHGetPathFromIDList");
+
+    return MyGetOpenFileName != NULL
+        && MySHBrowseForFolder != NULL
+        && MySHGetPathFromIDList != NULL;
+}
+
+static InitLibraries(void)
+{
+    static int initted = 0, success = 0;
+
+    if (!initted)
+    {
+        success = LoadDLLs();
+        initted = 1;
+    }
+
+    return success;
+}
+
+// Generate the "filter" string from the list of extensions.
+
+static char *GenerateFilterString(char **extensions)
+{
+    unsigned int result_len = 1;
+    unsigned int i;
+    char *result, *out;
+
+    if (extensions == NULL)
+    {
+        return NULL;
+    }
+
+    for (i = 0; extensions[i] != NULL; ++i)
+    {
+        result_len += 16 + strlen(extensions[i]) * 3;
+    }
+
+    result = malloc(result_len);
+    out = result;
+
+    for (i = 0; extensions[i] != NULL; ++i)
+    {
+        // .wad files (*.wad)\0
+        out += 1 + sprintf(out, "%s files (*.%s)",
+                           extensions[i], extensions[i]);
+        // *.wad\0
+        out += 1 + sprintf(out, "*.%s", extensions[i]);
+    }
+
+    *out = '\0';
+
+    return result;
+}
+
+int TXT_CanSelectFiles(void)
+{
+    return InitLibraries();
+}
+
+static char *SelectDirectory(char *window_title)
+{
+    LPITEMIDLIST pidl;
+    BROWSEINFO bi;
+    LPMALLOC allocator;
+    char selected[MAX_PATH] = "";
+    char *result;
+
+    ZeroMemory(&bi, sizeof(bi));
+    bi.hwndOwner = NULL;
+    bi.lpszTitle = window_title;
+    bi.pszDisplayName = selected;
+    bi.ulFlags = BIF_RETURNONLYFSDIRS | BIF_NEWDIALOGSTYLE;
+
+    pidl = MySHBrowseForFolder(&bi);
+
+    result = NULL;
+
+    if (pidl != NULL)
+    {
+        if (MySHGetPathFromIDList(pidl, selected))
+        {
+            result = strdup(selected);
+        }
+
+        // TODO: Free pidl
+    }
+
+    return result;
+}
+
+char *TXT_SelectFile(char *window_title, char **extensions)
+{
+    OPENFILENAME fm;
+    char selected[MAX_PATH] = "";
+    char *filter_string, *result;
+
+    if (!InitLibraries())
+    {
+        return NULL;
+    }
+
+    if (extensions == TXT_DIRECTORY)
+    {
+        return SelectDirectory(window_title);
+    }
+
+    filter_string = GenerateFilterString(extensions);
+
+    ZeroMemory(&fm, sizeof(fm));
+    fm.lStructSize = sizeof(OPENFILENAME);
+    fm.hwndOwner = NULL;
+    fm.lpstrTitle = window_title;
+    fm.lpstrFilter = filter_string;
+    fm.lpstrFile = selected;
+    fm.nMaxFile = MAX_PATH;
+    fm.Flags = OFN_EXPLORER | OFN_FILEMUSTEXIST | OFN_HIDEREADONLY;
+    fm.lpstrDefExt = "";
+
+    if (!MyGetOpenFileName(&fm))
+    {
+        result = NULL;
+    }
+    else
+    {
+        result = strdup(selected);
+    }
+
+    free(filter_string);
+
+    return result;
+}
+
+#elif defined(__MACOSX__)
+
+// Mac OS X code. Popping up a dialog requires Objective C/Cocoa
+// but we can get away with using AppleScript which avoids adding
+// an Objective C dependency. This is rather silly.
+
+// Printf format string for the "wrapper" portion of the AppleScript:
+
+#define APPLESCRIPT_WRAPPER \
+    "tell application (path to frontmost application as text)\n" \
+    "    set theFile to (%s)\n" \
+    "    copy POSIX path of theFile to stdout\n" \
+    "end tell\n"
+
+//    "tell app appname\n"
+
+static char *EscapedString(char *s)
+{
+    char *result;
+    char *in, *out;
+
+    result = malloc(strlen(s) + 3);
+    out = result;
+    *out++ = '\"';
+    for (in = s; *in != '\0'; ++in)
+    {
+        if (*in == '\"' || *in == '\\')
+        {
+            *out++ = '\\';
+        }
+        *out++ = *in;
+    }
+    *out++ = '\"';
+    *out = '\0';
+
+    return result;
+}
+
+// Build list of extensions, like: {"wad","lmp","txt"}
+
+static char *ExtensionsList(char **extensions)
+{
+    char *result, *escaped;
+    unsigned int result_len;
+    unsigned int i;
+
+    if (extensions == NULL)
+    {
+        return NULL;
+    }
+
+    result_len = 3;
+    for (i = 0; extensions[i] != NULL; ++i)
+    {
+        result_len += 5 + strlen(extensions[i]) * 2;
+    }
+
+    result = malloc(result_len);
+    strcpy(result, "{");
+
+    for (i = 0; extensions[i] != NULL; ++i)
+    {
+        escaped = EscapedString(extensions[i]);
+        strcat(result, escaped);
+        free(escaped);
+
+        if (extensions[i + 1] != NULL)
+            strcat(result, ",");
+    }
+
+    strcat(result, "}");
+
+    return result;
+}
+
+static char *GenerateSelector(char *window_title, char **extensions)
+{
+    char *chooser, *ext_list, *result;
+    unsigned int result_len;
+
+    result_len = 64;
+
+    if (extensions == TXT_DIRECTORY)
+    {
+        chooser = "choose folder";
+        ext_list = NULL;
+    }
+    else
+    {
+        chooser = "choose file";
+        ext_list = ExtensionsList(extensions);
+    }
+
+    // Calculate size.
+
+    if (window_title != NULL)
+    {
+        window_title = EscapedString(window_title);
+        result_len += strlen(window_title);
+    }
+    if (ext_list != NULL)
+    {
+        result_len += strlen(ext_list);
+    }
+
+    result = malloc(result_len);
+
+    strcpy(result, chooser);
+
+    if (window_title != NULL)
+    {
+        strcat(result, " with prompt ");
+        strcat(result, window_title);
+        free(window_title);
+    }
+
+    if (ext_list != NULL)
+    {
+        strcat(result, "of type ");
+        strcat(result, ext_list);
+        free(ext_list);
+    }
+
+    return result;
+}
+
+static char *GenerateAppleScript(char *window_title, char **extensions)
+{
+    char *selector, *result;
+
+    selector = GenerateSelector(window_title, extensions);
+
+    result = malloc(strlen(APPLESCRIPT_WRAPPER) + strlen(selector));
+    sprintf(result, APPLESCRIPT_WRAPPER, selector);
+    free(selector);
+
+    return result;
+}
+
+int TXT_CanSelectFiles(void)
+{
+    return 1;
+}
+
+char *TXT_SelectFile(char *window_title, char **extensions)
+{
+    char *argv[4];
+    char *result, *applescript;
+
+    applescript = GenerateAppleScript(window_title, extensions);
+
+    argv[0] = "/usr/bin/osascript";
+    argv[1] = "-e";
+    argv[2] = applescript;
+    argv[3] = NULL;
+
+    result = ExecReadOutput(argv);
+
+    free(applescript);
+
+    return result;
+}
+
+#else
+
+// Linux version: invoke the Zenity command line program to pop up a
+// dialog box. This avoids adding Gtk+ as a compile dependency.
+
+#define ZENITY_BINARY "/usr/bin/zenity"
+
+static unsigned int NumExtensions(char **extensions)
+{
+    unsigned int result = 0;
+
+    if (extensions != NULL)
+    {
+        for (result = 0; extensions[result] != NULL; ++result);
+    }
+
+    return result;
+}
+
+static int ZenityAvailable(void)
+{
+    return system(ZENITY_BINARY " --help >/dev/null 2>&1") == 0;
+}
+
+int TXT_CanSelectFiles(void)
+{
+    return ZenityAvailable();
+}
+
+char *TXT_SelectFile(char *window_title, char **extensions)
+{
+    unsigned int i;
+    char *result;
+    char **argv;
+    int argc;
+
+    if (!ZenityAvailable())
+    {
+        return NULL;
+    }
+
+    argv = calloc(4 + NumExtensions(extensions), sizeof(char *));
+    argv[0] = ZENITY_BINARY;
+    argv[1] = "--file-selection";
+    argc = 2;
+
+    if (window_title != NULL)
+    {
+        argv[argc] = malloc(10 + strlen(window_title));
+        sprintf(argv[argc], "--title=%s", window_title);
+        ++argc;
+    }
+
+    if (extensions == TXT_DIRECTORY)
+    {
+        argv[argc] = strdup("--directory");
+        ++argc;
+    }
+    else if (extensions != NULL)
+    {
+        for (i = 0; extensions[i] != NULL; ++i)
+        {
+            argv[argc] = malloc(30 + strlen(extensions[i]) * 2);
+            sprintf(argv[argc], "--file-filter=.%s | *.%s",
+                    extensions[i], extensions[i]);
+            ++argc;
+        }
+    }
+
+    argv[argc] = NULL;
+
+    result = ExecReadOutput(argv);
+
+    for (i = 2; i < argc; ++i)
+    {
+        free(argv[i]);
+    }
+
+    free(argv);
+
+    return result;
+}
+
+#endif
+
+static void TXT_FileSelectSizeCalc(TXT_UNCAST_ARG(fileselect))
+{
+    TXT_CAST_ARG(txt_fileselect_t, fileselect);
+
+    // Calculate widget size, but override the width to always
+    // be the configured size.
+
+    TXT_CalcWidgetSize(fileselect->inputbox);
+    fileselect->widget.w = fileselect->size;
+    fileselect->widget.h = fileselect->inputbox->widget.h;
+}
+
+static void TXT_FileSelectDrawer(TXT_UNCAST_ARG(fileselect))
+{
+    TXT_CAST_ARG(txt_fileselect_t, fileselect);
+
+    // Input box widget inherits all the properties of the
+    // file selector.
+
+    fileselect->inputbox->widget.x = fileselect->widget.x;
+    fileselect->inputbox->widget.y = fileselect->widget.y;
+    fileselect->inputbox->widget.w = fileselect->widget.w;
+    fileselect->inputbox->widget.h = fileselect->widget.h;
+
+    TXT_DrawWidget(fileselect->inputbox);
+}
+
+static void TXT_FileSelectDestructor(TXT_UNCAST_ARG(fileselect))
+{
+    TXT_CAST_ARG(txt_fileselect_t, fileselect);
+
+    TXT_DestroyWidget(fileselect->inputbox);
+}
+
+static int DoSelectFile(txt_fileselect_t *fileselect)
+{
+    char *path;
+    char **var;
+
+    if (TXT_CanSelectFiles())
+    {
+        path = TXT_SelectFile(fileselect->prompt,
+                              fileselect->extensions);
+        var = fileselect->inputbox->value;
+        free(*var);
+        *var = path;
+        return 1;
+    }
+
+    return 0;
+}
+
+static int TXT_FileSelectKeyPress(TXT_UNCAST_ARG(fileselect), int key)
+{
+    TXT_CAST_ARG(txt_fileselect_t, fileselect);
+
+    // When the enter key is pressed, pop up a file selection dialog,
+    // if file selectors work. Allow holding down 'alt' to override
+    // use of the native file selector, so the user can just type a path.
+
+    if (!fileselect->inputbox->editing
+     && !TXT_GetModifierState(TXT_MOD_ALT)
+     && key == KEY_ENTER)
+    {
+        if (DoSelectFile(fileselect))
+        {
+            return 1;
+        }
+    }
+
+    return TXT_WidgetKeyPress(fileselect->inputbox, key);
+}
+
+static void TXT_FileSelectMousePress(TXT_UNCAST_ARG(fileselect),
+                                     int x, int y, int b)
+{
+    TXT_CAST_ARG(txt_fileselect_t, fileselect);
+
+    if (!fileselect->inputbox->editing
+     && !TXT_GetModifierState(TXT_MOD_ALT)
+     && b == TXT_MOUSE_LEFT)
+    {
+        if (DoSelectFile(fileselect))
+        {
+            return;
+        }
+    }
+
+    return TXT_WidgetMousePress(fileselect->inputbox, x, y, b);
+}
+
+static void TXT_FileSelectFocused(TXT_UNCAST_ARG(fileselect), int focused)
+{
+    TXT_CAST_ARG(txt_fileselect_t, fileselect);
+
+    TXT_SetWidgetFocus(fileselect->inputbox, focused);
+}
+
+txt_widget_class_t txt_fileselect_class =
+{
+    TXT_AlwaysSelectable,
+    TXT_FileSelectSizeCalc,
+    TXT_FileSelectDrawer,
+    TXT_FileSelectKeyPress,
+    TXT_FileSelectDestructor,
+    TXT_FileSelectMousePress,
+    NULL,
+    TXT_FileSelectFocused,
+};
+
+txt_fileselect_t *TXT_NewFileSelector(char **variable, int size,
+                                      char *prompt, char **extensions)
+{
+    txt_fileselect_t *fileselect;
+
+    fileselect = malloc(sizeof(txt_fileselect_t));
+    TXT_InitWidget(fileselect, &txt_fileselect_class);
+    fileselect->inputbox = TXT_NewInputBox(variable, 1024);
+    fileselect->inputbox->widget.parent = &fileselect->widget;
+    fileselect->size = size;
+    fileselect->prompt = prompt;
+    fileselect->extensions = extensions;
+
+    return fileselect;
+}
+
--- /dev/null
+++ b/textscreen/txt_fileselect.h
@@ -1,0 +1,75 @@
+// Emacs style mode select   -*- C++ -*-
+//-----------------------------------------------------------------------------
+//
+// Copyright(C) 2013 Simon Howard
+//
+// This program 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 2
+// of the License, or (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program; if not, write to the Free Software
+// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
+// 02111-1307, USA.
+//
+//-----------------------------------------------------------------------------
+//
+// Routines for selecting files, and the txt_fileselect_t widget.
+//
+//-----------------------------------------------------------------------------
+
+#ifndef TXT_FILESELECT_H
+#define TXT_FILESELECT_H
+
+typedef struct txt_fileselect_s txt_fileselect_t;
+
+/**
+ * Returns non-zero if a native file selector is available on this
+ * platform.
+ */
+
+int TXT_CanSelectFiles(void);
+
+/**
+ * Open a native file selector to select a file.
+ *
+ * @param title       Pointer to a string containing a prompt to display
+ *                    in the window.
+ * @param extensions  NULL-terminated list of filename extensions for
+ *                    files that can be selected, or @ref TXT_DIRECTORY
+ *                    to select directories.
+ */
+
+char *TXT_SelectFile(char *window_title, char **extensions);
+
+/**
+ * Create a new txt_fileselect_t widget.
+ *
+ * @param variable    Pointer to a char * variable in which the selected
+ *                    file should be stored.
+ * @param size        Width of the file selector widget in characters.
+ * @param title       Pointer to a string containing a prompt to display
+ *                    in the file selection window.
+ * @param extensions  NULL-terminated list of filename extensions that
+ *                    can be used for this widget, or @ref TXT_DIRECTORY
+ *                    to select directories.
+ */
+
+txt_fileselect_t *TXT_NewFileSelector(char **variable, int size,
+                                      char *prompt, char **extensions);
+
+/**
+ * Special value to use for 'extensions' that selects a directory
+ * instead of a file.
+ */
+
+extern char *TXT_DIRECTORY[];
+
+#endif /* #ifndef TXT_FILESELECT_H */
+
--- a/textscreen/txt_inputbox.c
+++ b/textscreen/txt_inputbox.c
@@ -139,9 +139,21 @@
         SetBufferFromValue(inputbox);
     }
 
-    TXT_DrawUTF8String(inputbox->buffer);
+    // If string size exceeds the widget's width, show only the end.
 
-    chars = TXT_UTF8_Strlen(inputbox->buffer);
+    if (TXT_UTF8_Strlen(inputbox->buffer) > w - 1)
+    {
+        TXT_DrawString("\xae");
+        TXT_DrawUTF8String(
+            TXT_UTF8_SkipChars(inputbox->buffer,
+                               TXT_UTF8_Strlen(inputbox->buffer) - w + 2));
+        chars = w - 1;
+    }
+    else
+    {
+        TXT_DrawUTF8String(inputbox->buffer);
+        chars = TXT_UTF8_Strlen(inputbox->buffer);
+    }
 
     if (chars < w && inputbox->editing && focused)
     {