├── .gitmodules
├── LICENSE
├── README.md
├── examples
├── calc.lua
├── htdocs
│ ├── FileChooser.css
│ ├── FileChooser.js
│ ├── FileChooser.lua
│ ├── assets
│ │ ├── fetch.umd.js
│ │ ├── promise.min.js
│ │ ├── toastui
│ │ │ ├── LICENSE
│ │ │ ├── codemirror.min.css
│ │ │ ├── toastui-editor-all.min.js
│ │ │ ├── toastui-editor-viewer.min.css
│ │ │ ├── toastui-editor-viewer.min.js
│ │ │ └── toastui-editor.min.css
│ │ └── vue.min.js
│ ├── calc.css
│ ├── calc.html
│ ├── calc.js
│ ├── file.html
│ ├── md.html
│ ├── simple.html
│ ├── todo.css
│ ├── todo.html
│ ├── todo.js
│ ├── winexp.css
│ ├── winexp.html
│ ├── winexp.js
│ └── winexp.lua
├── launch.lua
├── open.lua
└── simple.lua
├── rock.mk
├── test.lua
├── webview-init.js
├── webview-launcher.lua
└── webview.c
/.gitmodules:
--------------------------------------------------------------------------------
1 | [submodule "webview-c"]
2 | path = webview-c
3 | url = https://github.com/javalikescript/webview-c.git
4 | [submodule "MemoryModule"]
5 | path = MemoryModule
6 | url = https://github.com/fancycode/MemoryModule.git
7 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | "lua-webview" contained in the files "webview.c" and "webview-launcher.lua" is licensed under the MIT License as follows:
2 |
3 | ====
4 |
5 | Copyright (c) 2019-2024 SPYL, javalikescript@free.fr
6 |
7 | Permission is hereby granted, free of charge, to any person obtaining a copy
8 | of this software and associated documentation files (the "Software"), to deal
9 | in the Software without restriction, including without limitation the rights
10 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11 | copies of the Software, and to permit persons to whom the Software is
12 | furnished to do so, subject to the following conditions:
13 |
14 | The above copyright notice and this permission notice shall be included in all
15 | copies or substantial portions of the Software.
16 |
17 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
23 | SOFTWARE.
24 |
25 | ====
26 |
27 | The examples maintained within the "lua-webview" project,
28 | excluding the folder "examples/htdocs/assets" and the files "examples/htdocs/todo.*",
29 | are licensed under the MIT License unless otherwise noticed.
30 |
31 |
32 |
33 | This repository also contains external software/libraries, the related license information is provided below for information only.
34 |
35 | The dependent libraries are licensed under the MIT License:
36 | - "webview-c" contained in the folder "webview-c" is licensed under the MIT License see https://github.com/javalikescript/webview-c
37 | "webview-c" is based on prior work by Serge Zaitsev, see https://github.com/zserge/webview
38 | "webview-c" embeds part of the Microsoft Edge WebView2 SDK, see https://aka.ms/webviewnuget
39 | - "MemoryModule" contained in the folder "MemoryModule" is licensed under the Mozilla Public License 2.0 see https://github.com/fancycode/MemoryModule
40 | Copyright (c) 2004-2015 by Joachim Bauch
41 |
42 | The example dependent libraries are mainly licensed under the MIT License:
43 | - "fetch" is licensed under the MIT License see https://github.com/github/fetch/releases
44 | - "vuejs" is licensed under the MIT License see https://vuejs.org/
45 | - "promise" is licensed under the MIT License see https://github.com/taylorhakes/promise-polyfill
46 | - "todo" is licensed under the MIT License see https://github.com/tastejs/todomvc/tree/gh-pages/examples/vue
47 | - "toastui" is licensed under the MIT License see https://github.com/nhn/tui.editor
48 | - "Babylon.js" is licensed under the Apache-2.0 License see https://github.com/BabylonJS/Babylon.js
49 | - "shapes.html" is in the public domain see https://github.com/end3r/MDN-Games-3D/blob/gh-pages/Babylon.js/shapes.html
50 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## Overview
2 |
3 | The Lua webview module provides functions to open a web page in a dedicated window from [Lua](http://www.lua.org/).
4 |
5 | ```lua
6 | require('webview').open('http://www.lua.org/')
7 | ```
8 |
9 | It uses *gtk-webkit2* on Linux and *MSHTML* (IE10/11) or *Edge* (Chromium) on Windows.
10 |
11 | Lua can evaluate JavaScript code and JavaScript can call a registered Lua function, see the `simple.lua` file in the examples.
12 |
13 | This module is a binding of the tiny cross-platform [webview-c](https://github.com/javalikescript/webview-c) C library.
14 |
15 | This module is part of the [luaclibs](https://github.com/javalikescript/luaclibs) project,
16 | the binaries can be found on the [luajls](https://github.com/javalikescript/luajls/releases/latest) releases.
17 | You could also install it using [LuaRocks](#luarocks).
18 |
19 | Lua webview is covered by the MIT license.
20 |
21 | ## Build
22 |
23 | The Lua webview module could be build using the rock makefile.
24 |
25 | ```lua
26 | make -f rock.mk PLAT=windows MAKE=make CC=gcc LD=gcc LUA_LIBDIR=.../lib LUA_INCDIR=.../include LUA=.../bin/lua54.exe
27 | ```
28 |
29 | ## Usage
30 |
31 | [Fast Cut](https://github.com/javalikescript/fcut) is an advanced example of webview usage.
32 |
33 | It allows to visually cut and join videos then export them losslessly thanks to FFmpeg.
34 |
35 | ## Launcher
36 |
37 | An optional Lua launcher module `webview-launcher.lua` is available.
38 | The HTML scripts using the type `text/lua` will be loaded automatically.
39 | The Lua scripts could expose named functions with callbacks to JavaScript.
40 |
41 | ```html
42 | Say Hello
43 |
50 | ```
51 | or using a Lua file
52 | ```html
53 |
54 | ```
55 |
56 | Additionally a JavaScript file `webview-init.js` is available to deal with the launcher initialization including in case of reloading.
57 |
58 | The launcher supports events in Lua when used with [luajls](https://github.com/javalikescript/luajls).
59 |
60 | ## Examples
61 |
62 | Using an HTTP server
63 | ```sh
64 | lua examples/calc.lua
65 | ```
66 |
67 |
68 |
69 |
70 | Using the file system
71 | ```sh
72 | lua examples/open.lua %CD%\examples\htdocs\todo.html
73 | ```
74 |
75 |
76 |
77 |
78 | Pure Lua
79 | ```sh
80 | wlua54 examples/simple.lua
81 | ```
82 |
83 |
84 |
85 |
86 | Generic launcher, with helper function to pass JSON objects and more
87 | ```sh
88 | lua examples/launch.lua examples/htdocs/simple.html --wv-event=thread
89 | ```
90 |
91 | ## LuaRocks
92 |
93 | Lua webview can be intalled using LuaRocks
94 |
95 | ### LuaRocks on Linux
96 |
97 | Prerequisites:
98 | ```sh
99 | sudo apt install luarocks lua5.3 lua5.3-dev
100 | sudo apt install libgtk-3-dev libwebkit2gtk-4.0-dev
101 | ```
102 |
103 | ```sh
104 | luarocks install lua-webview --local
105 | ```
106 |
107 | ### LuaRocks on Windows
108 |
109 | Prerequisites:
110 | Download the Lua 64 bits dynamic libraries built with MinGW-w64 from [Lua Binaries](https://sourceforge.net/projects/luabinaries/).
111 | Add [MSYS2](https://www.msys2.org/), MinGW-w64 and [git](https://git-scm.com/) in the path.
112 |
113 |
114 | ```Batchfile
115 | luarocks --lua-dir C:/bin/lua-5.3.5 MAKE=make CC=gcc LD=gcc install lua-webview
116 | ```
117 | Take care to use forward slashes for the Lua path.
118 |
--------------------------------------------------------------------------------
/examples/calc.lua:
--------------------------------------------------------------------------------
1 | local event = require('jls.lang.event')
2 | local File = require('jls.io.File')
3 | local WebView = require('jls.util.WebView')
4 | local FileHttpHandler = require('jls.net.http.handler.FileHttpHandler')
5 | local RestHttpHandler = require('jls.net.http.handler.RestHttpHandler')
6 |
7 | local scriptFile = File:new(arg[0]):getAbsoluteFile()
8 | local scriptDir = scriptFile:getParentFile()
9 | local htdocsDir = File:new(scriptDir, 'htdocs')
10 |
11 | -- localhost ::1 127.0.0.1
12 | WebView.open('http://localhost:0/calc.html', 'Calc', 320, 480, true):next(function(webview)
13 | local httpServer = webview:getHttpServer()
14 | print('WebView opened with HTTP Server bound on address', httpServer:getAddress())
15 | httpServer:createContext('/(.*)', FileHttpHandler:new(htdocsDir))
16 | httpServer:createContext('/rest/(.*)', RestHttpHandler:new({
17 | ['calculate(requestJson)?method=POST&content-type=application/json'] = function(exchange, requestJson)
18 | local f, err = load('return '..tostring(requestJson.line))
19 | return {line = f and f() or err or ''}
20 | end
21 | }))
22 | return webview:getThread():ended()
23 | end):next(function()
24 | print('WebView closed')
25 | end):catch(function(err)
26 | print('Cannot open webview due to '..tostring(err))
27 | end)
28 |
29 | --print('Looping')
30 | event:loop()
31 | event:close()
32 |
--------------------------------------------------------------------------------
/examples/htdocs/FileChooser.css:
--------------------------------------------------------------------------------
1 | .file-chooser-dialog {
2 | display: flex;
3 | flex-direction: column;
4 | }
5 | .file-chooser-content {
6 | flex-grow: 1;
7 | }
8 | .file-chooser-flex-row {
9 | min-width: 100%;
10 | display: flex;
11 | flex-direction: row;
12 | }
13 | .file-chooser-flex-row > * {
14 | margin-top: auto;
15 | margin-bottom: auto;
16 | }
17 | .file-chooser-flex-row-content {
18 | flex-grow: 1;
19 | }
20 | .file-chooser-footer.file-chooser-flex-row {
21 | justify-content: flex-end;
22 | }
23 | .file-chooser-dialog ul {
24 | list-style-type: none;
25 | }
26 | .file-chooser-directory {
27 | text-decoration: underline;
28 | }
29 |
--------------------------------------------------------------------------------
/examples/htdocs/FileChooser.js:
--------------------------------------------------------------------------------
1 | (function() {
2 |
3 | function directoryFirst(fa, fb) {
4 | var da = fa.isDirectory;
5 | var db = fb.isDirectory;
6 | return da === db ? 0 : (da ? -1 : 1);
7 | }
8 |
9 | function filterFiles(files, showAll, contains) {
10 | var containsLowerCase = contains && contains.toLowerCase();
11 | return files.filter(function(file) {
12 | if (!showAll && file.name.charAt(0) === '.' && file.name !== '..') {
13 | return false;
14 | }
15 | return file.isDirectory || !containsLowerCase || file.name.toLowerCase().indexOf(containsLowerCase) >= 0;
16 | })
17 | }
18 |
19 | function fileList(path, useFetch, callback) {
20 | if (useFetch) {
21 | if (webview && webview.fileList) {
22 | webview.fileList(path, callback);
23 | } else {
24 | callback('webview.fileList is not available');
25 | }
26 | } else {
27 | fetch('rest/listFiles', {
28 | method: 'POST',
29 | headers: {
30 | "Content-Type": "text/plain"
31 | },
32 | body: path
33 | }).then(function(response) {
34 | return response.json();
35 | }).then(function(list) {
36 | callback(null, list);
37 | }, function(reason) {
38 | callback(reason || 'Unknown error');
39 | });
40 | }
41 | }
42 |
43 | var FileChooserDialog = Vue.component('file-chooser-dialog', {
44 | template: '
',
69 | data: function() { return {
70 | extention: '',
71 | inputPath: '',
72 | name: '',
73 | path: '',
74 | files: [],
75 | label: 'Open',
76 | multiple: false,
77 | save: false,
78 | directory: false,
79 | showAll: false,
80 | showCancel: true,
81 | showSettings: false,
82 | fetch: false
83 | }; },
84 | methods: {
85 | onFilePressed: function(file) {
86 | if (file.isDirectory) {
87 | this.setPath(this.path + '/' + file.name);
88 | return;
89 | }
90 | if (!this.multiple) {
91 | this.files.forEach(function(f) {
92 | if (f !== file) {
93 | f.selected = false;
94 | }
95 | });
96 | }
97 | if (!this.directory) {
98 | if (this.save) {
99 | this.name = file.name;
100 | }
101 | file.selected = !file.selected;
102 | }
103 | },
104 | setPath: function(path) {
105 | if (this.path !== path) {
106 | this.list(path);
107 | }
108 | },
109 | refresh: function() {
110 | if (this.save) {
111 | this.multiple = false;
112 | }
113 | this.list(this.path !== '' ? this.path : '.')
114 | },
115 | list: function(path) {
116 | var fc = this;
117 | fileList(path, this.fetch, function(reason, files) {
118 | if (files) {
119 | var path = files.shift();
120 | fc.show(path, files);
121 | } else {
122 | fc.error(reason);
123 | }
124 | });
125 | },
126 | error: function(message) {
127 | console.error('file-chooser error', message);
128 | this.$emit('selected', []);
129 | },
130 | cancel: function() {
131 | this.$emit('selected', []);
132 | },
133 | done: function() {
134 | var files;
135 | if (this.directory) {
136 | files = [];
137 | } else if (this.save) {
138 | files = [this.name];
139 | } else {
140 | files = this.files.filter(function(file) {
141 | return file.selected;
142 | }).map(function(file) {
143 | return file.name;
144 | });
145 | }
146 | files.unshift(this.path)
147 | this.$emit('selected', files)
148 | },
149 | show: function(path, files) {
150 | this.inputPath = path;
151 | this.path = path;
152 | if (!Array.isArray(files)) {
153 | files = [];
154 | }
155 | files.forEach(function(file) {
156 | file.selected = false;
157 | file.selectable = true;
158 | });
159 | this.files = files;
160 | }
161 | },
162 | computed: {
163 | filteredList: function () {
164 | var files = filterFiles(this.files, this.showAll, this.extention);
165 | files.sort(directoryFirst);
166 | return files;
167 | }
168 | }
169 | });
170 |
171 | FileChooserDialog.show = function(vm, options) {
172 | var fileChooser = new FileChooserDialog();
173 | fileChooser.$mount();
174 | if (typeof options === 'object') {
175 | for (var k in options) {
176 | fileChooser[k] = options[k];
177 | }
178 | }
179 | vm.$el.appendChild(fileChooser.$el);
180 | fileChooser.$on('selected', function(files) {
181 | vm.$el.removeChild(fileChooser.$el);
182 | fileChooser.$destroy();
183 | fileChooser = null;
184 | });
185 | return fileChooser;
186 | }
187 |
188 | Vue.component('file-chooser-input', {
189 | template: '' +
190 | ' ' +
192 | '' +
193 | ' ' +
194 | ' ',
195 | data: function() { return {
196 | name: '',
197 | path: '',
198 | files: [],
199 | size: 40,
200 | fetch: false
201 | }; },
202 | methods: {
203 | nameChanged: function() {
204 | console.info('nameChanged() "' + this.name + '"');
205 | var value = this.name;
206 | if (this.path === '') {
207 | this.list('.');
208 | return;
209 | }
210 | if (value === '') {
211 | return;
212 | }
213 | for (var i = 0; i < this.files.length; i++) {
214 | var file = this.files[i];
215 | if (file.isDirectory && file.name === value) {
216 | this.list(this.path + '/' + file.name);
217 | break;
218 | }
219 | }
220 | },
221 | clean: function() {
222 | if (this.path === '') {
223 | this.list('.');
224 | } else if (this.name !== '') {
225 | console.info('clean() "' + this.path + '"');
226 | this.name = '';
227 | }
228 | },
229 | refresh: function() {
230 | this.list(this.path !== '' ? this.path : '.')
231 | },
232 | list: function(path) {
233 | console.info('list("' + path + '")');
234 | var fc = this;
235 | fileList(path, this.fetch, function(reason, files) {
236 | if (files) {
237 | var path = files.shift();
238 | fc.show(path, files);
239 | } else {
240 | this.error(reason);
241 | }
242 | });
243 | },
244 | error: function(message) {
245 | console.error('file-chooser error', message);
246 | this.$emit('selected', []);
247 | },
248 | show: function(path, files) {
249 | console.info('show("' + path + '")');
250 | this.path = path;
251 | if (!Array.isArray(files)) {
252 | files = [];
253 | }
254 | this.files = files;
255 | this.name = '';
256 | }
257 | },
258 | computed: {
259 | filteredList: function () {
260 | var files = filterFiles(this.files);
261 | files.sort(directoryFirst);
262 | return files;
263 | },
264 | placeholder: function() {
265 | if (this.path) {
266 | return this.path.length <= this.size ? this.path : '...' + this.path.slice(3-this.size);
267 | }
268 | return 'Click to browse';
269 | }
270 | }
271 | });
272 |
273 | })();
--------------------------------------------------------------------------------
/examples/htdocs/FileChooser.lua:
--------------------------------------------------------------------------------
1 | local Path = require('jls.io.Path')
2 | local File = require('jls.io.File')
3 |
4 | local function listFiles(value, callback)
5 | if type(callback) ~= 'function' then
6 | return
7 | end
8 | local dir = File:new(value or '.'):getAbsoluteFile()
9 | local files = dir:listFiles()
10 | local parent = dir:getParent()
11 | if files then
12 | local list = {}
13 | if parent then
14 | table.insert(list, {
15 | name = '..',
16 | isDirectory = true,
17 | length = 0,
18 | lastModified = dir:lastModified(),
19 | })
20 | end
21 | for _, file in ipairs(files) do
22 | table.insert(list, {
23 | name = file:getName(),
24 | isDirectory = file:isDirectory(),
25 | length = file:length(),
26 | lastModified = file:lastModified(),
27 | })
28 | end
29 | local path = Path.normalizePath(dir:getPath())
30 | --print('listFiles('..tostring(value)..') "'..tostring(path)..'" found '..tostring(#list)..' entries, parent: "'..tostring(parent)..'"')
31 | table.insert(list, 1, path)
32 | callback(nil, list)
33 | end
34 | end
35 |
36 | if expose ~= nil then
37 | -- loaded as html src
38 | expose('fileList', listFiles)
39 | else
40 | -- loaded as module
41 | return {
42 | listFiles = listFiles
43 | }
44 | end
45 |
--------------------------------------------------------------------------------
/examples/htdocs/assets/fetch.umd.js:
--------------------------------------------------------------------------------
1 | (function (global, factory) {
2 | typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
3 | typeof define === 'function' && define.amd ? define(['exports'], factory) :
4 | (factory((global.WHATWGFetch = {})));
5 | }(this, (function (exports) { 'use strict';
6 |
7 | var support = {
8 | searchParams: 'URLSearchParams' in self,
9 | iterable: 'Symbol' in self && 'iterator' in Symbol,
10 | blob:
11 | 'FileReader' in self &&
12 | 'Blob' in self &&
13 | (function() {
14 | try {
15 | new Blob();
16 | return true
17 | } catch (e) {
18 | return false
19 | }
20 | })(),
21 | formData: 'FormData' in self,
22 | arrayBuffer: 'ArrayBuffer' in self
23 | };
24 |
25 | function isDataView(obj) {
26 | return obj && DataView.prototype.isPrototypeOf(obj)
27 | }
28 |
29 | if (support.arrayBuffer) {
30 | var viewClasses = [
31 | '[object Int8Array]',
32 | '[object Uint8Array]',
33 | '[object Uint8ClampedArray]',
34 | '[object Int16Array]',
35 | '[object Uint16Array]',
36 | '[object Int32Array]',
37 | '[object Uint32Array]',
38 | '[object Float32Array]',
39 | '[object Float64Array]'
40 | ];
41 |
42 | var isArrayBufferView =
43 | ArrayBuffer.isView ||
44 | function(obj) {
45 | return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1
46 | };
47 | }
48 |
49 | function normalizeName(name) {
50 | if (typeof name !== 'string') {
51 | name = String(name);
52 | }
53 | if (/[^a-z0-9\-#$%&'*+.^_`|~]/i.test(name)) {
54 | throw new TypeError('Invalid character in header field name')
55 | }
56 | return name.toLowerCase()
57 | }
58 |
59 | function normalizeValue(value) {
60 | if (typeof value !== 'string') {
61 | value = String(value);
62 | }
63 | return value
64 | }
65 |
66 | // Build a destructive iterator for the value list
67 | function iteratorFor(items) {
68 | var iterator = {
69 | next: function() {
70 | var value = items.shift();
71 | return {done: value === undefined, value: value}
72 | }
73 | };
74 |
75 | if (support.iterable) {
76 | iterator[Symbol.iterator] = function() {
77 | return iterator
78 | };
79 | }
80 |
81 | return iterator
82 | }
83 |
84 | function Headers(headers) {
85 | this.map = {};
86 |
87 | if (headers instanceof Headers) {
88 | headers.forEach(function(value, name) {
89 | this.append(name, value);
90 | }, this);
91 | } else if (Array.isArray(headers)) {
92 | headers.forEach(function(header) {
93 | this.append(header[0], header[1]);
94 | }, this);
95 | } else if (headers) {
96 | Object.getOwnPropertyNames(headers).forEach(function(name) {
97 | this.append(name, headers[name]);
98 | }, this);
99 | }
100 | }
101 |
102 | Headers.prototype.append = function(name, value) {
103 | name = normalizeName(name);
104 | value = normalizeValue(value);
105 | var oldValue = this.map[name];
106 | this.map[name] = oldValue ? oldValue + ', ' + value : value;
107 | };
108 |
109 | Headers.prototype['delete'] = function(name) {
110 | delete this.map[normalizeName(name)];
111 | };
112 |
113 | Headers.prototype.get = function(name) {
114 | name = normalizeName(name);
115 | return this.has(name) ? this.map[name] : null
116 | };
117 |
118 | Headers.prototype.has = function(name) {
119 | return this.map.hasOwnProperty(normalizeName(name))
120 | };
121 |
122 | Headers.prototype.set = function(name, value) {
123 | this.map[normalizeName(name)] = normalizeValue(value);
124 | };
125 |
126 | Headers.prototype.forEach = function(callback, thisArg) {
127 | for (var name in this.map) {
128 | if (this.map.hasOwnProperty(name)) {
129 | callback.call(thisArg, this.map[name], name, this);
130 | }
131 | }
132 | };
133 |
134 | Headers.prototype.keys = function() {
135 | var items = [];
136 | this.forEach(function(value, name) {
137 | items.push(name);
138 | });
139 | return iteratorFor(items)
140 | };
141 |
142 | Headers.prototype.values = function() {
143 | var items = [];
144 | this.forEach(function(value) {
145 | items.push(value);
146 | });
147 | return iteratorFor(items)
148 | };
149 |
150 | Headers.prototype.entries = function() {
151 | var items = [];
152 | this.forEach(function(value, name) {
153 | items.push([name, value]);
154 | });
155 | return iteratorFor(items)
156 | };
157 |
158 | if (support.iterable) {
159 | Headers.prototype[Symbol.iterator] = Headers.prototype.entries;
160 | }
161 |
162 | function consumed(body) {
163 | if (body.bodyUsed) {
164 | return Promise.reject(new TypeError('Already read'))
165 | }
166 | body.bodyUsed = true;
167 | }
168 |
169 | function fileReaderReady(reader) {
170 | return new Promise(function(resolve, reject) {
171 | reader.onload = function() {
172 | resolve(reader.result);
173 | };
174 | reader.onerror = function() {
175 | reject(reader.error);
176 | };
177 | })
178 | }
179 |
180 | function readBlobAsArrayBuffer(blob) {
181 | var reader = new FileReader();
182 | var promise = fileReaderReady(reader);
183 | reader.readAsArrayBuffer(blob);
184 | return promise
185 | }
186 |
187 | function readBlobAsText(blob) {
188 | var reader = new FileReader();
189 | var promise = fileReaderReady(reader);
190 | reader.readAsText(blob);
191 | return promise
192 | }
193 |
194 | function readArrayBufferAsText(buf) {
195 | var view = new Uint8Array(buf);
196 | var chars = new Array(view.length);
197 |
198 | for (var i = 0; i < view.length; i++) {
199 | chars[i] = String.fromCharCode(view[i]);
200 | }
201 | return chars.join('')
202 | }
203 |
204 | function bufferClone(buf) {
205 | if (buf.slice) {
206 | return buf.slice(0)
207 | } else {
208 | var view = new Uint8Array(buf.byteLength);
209 | view.set(new Uint8Array(buf));
210 | return view.buffer
211 | }
212 | }
213 |
214 | function Body() {
215 | this.bodyUsed = false;
216 |
217 | this._initBody = function(body) {
218 | this._bodyInit = body;
219 | if (!body) {
220 | this._bodyText = '';
221 | } else if (typeof body === 'string') {
222 | this._bodyText = body;
223 | } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
224 | this._bodyBlob = body;
225 | } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
226 | this._bodyFormData = body;
227 | } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
228 | this._bodyText = body.toString();
229 | } else if (support.arrayBuffer && support.blob && isDataView(body)) {
230 | this._bodyArrayBuffer = bufferClone(body.buffer);
231 | // IE 10-11 can't handle a DataView body.
232 | this._bodyInit = new Blob([this._bodyArrayBuffer]);
233 | } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {
234 | this._bodyArrayBuffer = bufferClone(body);
235 | } else {
236 | this._bodyText = body = Object.prototype.toString.call(body);
237 | }
238 |
239 | if (!this.headers.get('content-type')) {
240 | if (typeof body === 'string') {
241 | this.headers.set('content-type', 'text/plain;charset=UTF-8');
242 | } else if (this._bodyBlob && this._bodyBlob.type) {
243 | this.headers.set('content-type', this._bodyBlob.type);
244 | } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
245 | this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
246 | }
247 | }
248 | };
249 |
250 | if (support.blob) {
251 | this.blob = function() {
252 | var rejected = consumed(this);
253 | if (rejected) {
254 | return rejected
255 | }
256 |
257 | if (this._bodyBlob) {
258 | return Promise.resolve(this._bodyBlob)
259 | } else if (this._bodyArrayBuffer) {
260 | return Promise.resolve(new Blob([this._bodyArrayBuffer]))
261 | } else if (this._bodyFormData) {
262 | throw new Error('could not read FormData body as blob')
263 | } else {
264 | return Promise.resolve(new Blob([this._bodyText]))
265 | }
266 | };
267 |
268 | this.arrayBuffer = function() {
269 | if (this._bodyArrayBuffer) {
270 | return consumed(this) || Promise.resolve(this._bodyArrayBuffer)
271 | } else {
272 | return this.blob().then(readBlobAsArrayBuffer)
273 | }
274 | };
275 | }
276 |
277 | this.text = function() {
278 | var rejected = consumed(this);
279 | if (rejected) {
280 | return rejected
281 | }
282 |
283 | if (this._bodyBlob) {
284 | return readBlobAsText(this._bodyBlob)
285 | } else if (this._bodyArrayBuffer) {
286 | return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))
287 | } else if (this._bodyFormData) {
288 | throw new Error('could not read FormData body as text')
289 | } else {
290 | return Promise.resolve(this._bodyText)
291 | }
292 | };
293 |
294 | if (support.formData) {
295 | this.formData = function() {
296 | return this.text().then(decode)
297 | };
298 | }
299 |
300 | this.json = function() {
301 | return this.text().then(JSON.parse)
302 | };
303 |
304 | return this
305 | }
306 |
307 | // HTTP methods whose capitalization should be normalized
308 | var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];
309 |
310 | function normalizeMethod(method) {
311 | var upcased = method.toUpperCase();
312 | return methods.indexOf(upcased) > -1 ? upcased : method
313 | }
314 |
315 | function Request(input, options) {
316 | options = options || {};
317 | var body = options.body;
318 |
319 | if (input instanceof Request) {
320 | if (input.bodyUsed) {
321 | throw new TypeError('Already read')
322 | }
323 | this.url = input.url;
324 | this.credentials = input.credentials;
325 | if (!options.headers) {
326 | this.headers = new Headers(input.headers);
327 | }
328 | this.method = input.method;
329 | this.mode = input.mode;
330 | this.signal = input.signal;
331 | if (!body && input._bodyInit != null) {
332 | body = input._bodyInit;
333 | input.bodyUsed = true;
334 | }
335 | } else {
336 | this.url = String(input);
337 | }
338 |
339 | this.credentials = options.credentials || this.credentials || 'same-origin';
340 | if (options.headers || !this.headers) {
341 | this.headers = new Headers(options.headers);
342 | }
343 | this.method = normalizeMethod(options.method || this.method || 'GET');
344 | this.mode = options.mode || this.mode || null;
345 | this.signal = options.signal || this.signal;
346 | this.referrer = null;
347 |
348 | if ((this.method === 'GET' || this.method === 'HEAD') && body) {
349 | throw new TypeError('Body not allowed for GET or HEAD requests')
350 | }
351 | this._initBody(body);
352 | }
353 |
354 | Request.prototype.clone = function() {
355 | return new Request(this, {body: this._bodyInit})
356 | };
357 |
358 | function decode(body) {
359 | var form = new FormData();
360 | body
361 | .trim()
362 | .split('&')
363 | .forEach(function(bytes) {
364 | if (bytes) {
365 | var split = bytes.split('=');
366 | var name = split.shift().replace(/\+/g, ' ');
367 | var value = split.join('=').replace(/\+/g, ' ');
368 | form.append(decodeURIComponent(name), decodeURIComponent(value));
369 | }
370 | });
371 | return form
372 | }
373 |
374 | function parseHeaders(rawHeaders) {
375 | var headers = new Headers();
376 | // Replace instances of \r\n and \n followed by at least one space or horizontal tab with a space
377 | // https://tools.ietf.org/html/rfc7230#section-3.2
378 | var preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, ' ');
379 | preProcessedHeaders.split(/\r?\n/).forEach(function(line) {
380 | var parts = line.split(':');
381 | var key = parts.shift().trim();
382 | if (key) {
383 | var value = parts.join(':').trim();
384 | headers.append(key, value);
385 | }
386 | });
387 | return headers
388 | }
389 |
390 | Body.call(Request.prototype);
391 |
392 | function Response(bodyInit, options) {
393 | if (!options) {
394 | options = {};
395 | }
396 |
397 | this.type = 'default';
398 | this.status = options.status === undefined ? 200 : options.status;
399 | this.ok = this.status >= 200 && this.status < 300;
400 | this.statusText = 'statusText' in options ? options.statusText : 'OK';
401 | this.headers = new Headers(options.headers);
402 | this.url = options.url || '';
403 | this._initBody(bodyInit);
404 | }
405 |
406 | Body.call(Response.prototype);
407 |
408 | Response.prototype.clone = function() {
409 | return new Response(this._bodyInit, {
410 | status: this.status,
411 | statusText: this.statusText,
412 | headers: new Headers(this.headers),
413 | url: this.url
414 | })
415 | };
416 |
417 | Response.error = function() {
418 | var response = new Response(null, {status: 0, statusText: ''});
419 | response.type = 'error';
420 | return response
421 | };
422 |
423 | var redirectStatuses = [301, 302, 303, 307, 308];
424 |
425 | Response.redirect = function(url, status) {
426 | if (redirectStatuses.indexOf(status) === -1) {
427 | throw new RangeError('Invalid status code')
428 | }
429 |
430 | return new Response(null, {status: status, headers: {location: url}})
431 | };
432 |
433 | exports.DOMException = self.DOMException;
434 | try {
435 | new exports.DOMException();
436 | } catch (err) {
437 | exports.DOMException = function(message, name) {
438 | this.message = message;
439 | this.name = name;
440 | var error = Error(message);
441 | this.stack = error.stack;
442 | };
443 | exports.DOMException.prototype = Object.create(Error.prototype);
444 | exports.DOMException.prototype.constructor = exports.DOMException;
445 | }
446 |
447 | function fetch(input, init) {
448 | return new Promise(function(resolve, reject) {
449 | var request = new Request(input, init);
450 |
451 | if (request.signal && request.signal.aborted) {
452 | return reject(new exports.DOMException('Aborted', 'AbortError'))
453 | }
454 |
455 | var xhr = new XMLHttpRequest();
456 |
457 | function abortXhr() {
458 | xhr.abort();
459 | }
460 |
461 | xhr.onload = function() {
462 | var options = {
463 | status: xhr.status,
464 | statusText: xhr.statusText,
465 | headers: parseHeaders(xhr.getAllResponseHeaders() || '')
466 | };
467 | options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');
468 | var body = 'response' in xhr ? xhr.response : xhr.responseText;
469 | resolve(new Response(body, options));
470 | };
471 |
472 | xhr.onerror = function() {
473 | reject(new TypeError('Network request failed'));
474 | };
475 |
476 | xhr.ontimeout = function() {
477 | reject(new TypeError('Network request failed'));
478 | };
479 |
480 | xhr.onabort = function() {
481 | reject(new exports.DOMException('Aborted', 'AbortError'));
482 | };
483 |
484 | xhr.open(request.method, request.url, true);
485 |
486 | if (request.credentials === 'include') {
487 | xhr.withCredentials = true;
488 | } else if (request.credentials === 'omit') {
489 | xhr.withCredentials = false;
490 | }
491 |
492 | if ('responseType' in xhr && support.blob) {
493 | xhr.responseType = 'blob';
494 | }
495 |
496 | request.headers.forEach(function(value, name) {
497 | xhr.setRequestHeader(name, value);
498 | });
499 |
500 | if (request.signal) {
501 | request.signal.addEventListener('abort', abortXhr);
502 |
503 | xhr.onreadystatechange = function() {
504 | // DONE (success or failure)
505 | if (xhr.readyState === 4) {
506 | request.signal.removeEventListener('abort', abortXhr);
507 | }
508 | };
509 | }
510 |
511 | xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);
512 | })
513 | }
514 |
515 | fetch.polyfill = true;
516 |
517 | if (!self.fetch) {
518 | self.fetch = fetch;
519 | self.Headers = Headers;
520 | self.Request = Request;
521 | self.Response = Response;
522 | }
523 |
524 | exports.Headers = Headers;
525 | exports.Request = Request;
526 | exports.Response = Response;
527 | exports.fetch = fetch;
528 |
529 | Object.defineProperty(exports, '__esModule', { value: true });
530 |
531 | })));
532 |
--------------------------------------------------------------------------------
/examples/htdocs/assets/promise.min.js:
--------------------------------------------------------------------------------
1 | !function(e,n){"object"==typeof exports&&"undefined"!=typeof module?n():"function"==typeof define&&define.amd?define(n):n()}(0,function(){"use strict";function e(e){var n=this.constructor;return this.then(function(t){return n.resolve(e()).then(function(){return t})},function(t){return n.resolve(e()).then(function(){return n.reject(t)})})}function n(e){return!(!e||"undefined"==typeof e.length)}function t(){}function o(e){if(!(this instanceof o))throw new TypeError("Promises must be constructed via new");if("function"!=typeof e)throw new TypeError("not a function");this._state=0,this._handled=!1,this._value=undefined,this._deferreds=[],c(e,this)}function r(e,n){for(;3===e._state;)e=e._value;0!==e._state?(e._handled=!0,o._immediateFn(function(){var t=1===e._state?n.onFulfilled:n.onRejected;if(null!==t){var o;try{o=t(e._value)}catch(r){return void f(n.promise,r)}i(n.promise,o)}else(1===e._state?i:f)(n.promise,e._value)})):e._deferreds.push(n)}function i(e,n){try{if(n===e)throw new TypeError("A promise cannot be resolved with itself.");if(n&&("object"==typeof n||"function"==typeof n)){var t=n.then;if(n instanceof o)return e._state=3,e._value=n,void u(e);if("function"==typeof t)return void c(function(e,n){return function(){e.apply(n,arguments)}}(t,n),e)}e._state=1,e._value=n,u(e)}catch(r){f(e,r)}}function f(e,n){e._state=2,e._value=n,u(e)}function u(e){2===e._state&&0===e._deferreds.length&&o._immediateFn(function(){e._handled||o._unhandledRejectionFn(e._value)});for(var n=0,t=e._deferreds.length;t>n;n++)r(e,e._deferreds[n]);e._deferreds=null}function c(e,n){var t=!1;try{e(function(e){t||(t=!0,i(n,e))},function(e){t||(t=!0,f(n,e))})}catch(o){if(t)return;t=!0,f(n,o)}}var a=setTimeout;o.prototype["catch"]=function(e){return this.then(null,e)},o.prototype.then=function(e,n){var o=new this.constructor(t);return r(this,new function(e,n,t){this.onFulfilled="function"==typeof e?e:null,this.onRejected="function"==typeof n?n:null,this.promise=t}(e,n,o)),o},o.prototype["finally"]=e,o.all=function(e){return new o(function(t,o){function r(e,n){try{if(n&&("object"==typeof n||"function"==typeof n)){var u=n.then;if("function"==typeof u)return void u.call(n,function(n){r(e,n)},o)}i[e]=n,0==--f&&t(i)}catch(c){o(c)}}if(!n(e))return o(new TypeError("Promise.all accepts an array"));var i=Array.prototype.slice.call(e);if(0===i.length)return t([]);for(var f=i.length,u=0;i.length>u;u++)r(u,i[u])})},o.resolve=function(e){return e&&"object"==typeof e&&e.constructor===o?e:new o(function(n){n(e)})},o.reject=function(e){return new o(function(n,t){t(e)})},o.race=function(e){return new o(function(t,r){if(!n(e))return r(new TypeError("Promise.race accepts an array"));for(var i=0,f=e.length;f>i;i++)o.resolve(e[i]).then(t,r)})},o._immediateFn="function"==typeof setImmediate&&function(e){setImmediate(e)}||function(e){a(e,0)},o._unhandledRejectionFn=function(e){void 0!==console&&console&&console.warn("Possible Unhandled Promise Rejection:",e)};var l=function(){if("undefined"!=typeof self)return self;if("undefined"!=typeof window)return window;if("undefined"!=typeof global)return global;throw Error("unable to locate global object")}();"Promise"in l?l.Promise.prototype["finally"]||(l.Promise.prototype["finally"]=e):l.Promise=o});
2 |
--------------------------------------------------------------------------------
/examples/htdocs/assets/toastui/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2020 NHN Corp.
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in
13 | all copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21 | THE SOFTWARE.
--------------------------------------------------------------------------------
/examples/htdocs/assets/toastui/codemirror.min.css:
--------------------------------------------------------------------------------
1 | .CodeMirror{font-family:monospace;height:300px;color:#000;direction:ltr}.CodeMirror-lines{padding:4px 0}.CodeMirror pre.CodeMirror-line,.CodeMirror pre.CodeMirror-line-like{padding:0 4px}.CodeMirror-gutter-filler,.CodeMirror-scrollbar-filler{background-color:#fff}.CodeMirror-gutters{border-right:1px solid #ddd;background-color:#f7f7f7;white-space:nowrap}.CodeMirror-linenumber{padding:0 3px 0 5px;min-width:20px;text-align:right;color:#999;white-space:nowrap}.CodeMirror-guttermarker{color:#000}.CodeMirror-guttermarker-subtle{color:#999}.CodeMirror-cursor{border-left:1px solid #000;border-right:none;width:0}.CodeMirror div.CodeMirror-secondarycursor{border-left:1px solid silver}.cm-fat-cursor .CodeMirror-cursor{width:auto;border:0!important;background:#7e7}.cm-fat-cursor div.CodeMirror-cursors{z-index:1}.cm-fat-cursor-mark{background-color:rgba(20,255,20,.5);-webkit-animation:blink 1.06s steps(1) infinite;-moz-animation:blink 1.06s steps(1) infinite;animation:blink 1.06s steps(1) infinite}.cm-animate-fat-cursor{width:auto;border:0;-webkit-animation:blink 1.06s steps(1) infinite;-moz-animation:blink 1.06s steps(1) infinite;animation:blink 1.06s steps(1) infinite;background-color:#7e7}@-moz-keyframes blink{50%{background-color:transparent}}@-webkit-keyframes blink{50%{background-color:transparent}}@keyframes blink{50%{background-color:transparent}}.cm-tab{display:inline-block;text-decoration:inherit}.CodeMirror-rulers{position:absolute;left:0;right:0;top:-50px;bottom:0;overflow:hidden}.CodeMirror-ruler{border-left:1px solid #ccc;top:0;bottom:0;position:absolute}.cm-s-default .cm-header{color:#00f}.cm-s-default .cm-quote{color:#090}.cm-negative{color:#d44}.cm-positive{color:#292}.cm-header,.cm-strong{font-weight:700}.cm-em{font-style:italic}.cm-link{text-decoration:underline}.cm-strikethrough{text-decoration:line-through}.cm-s-default .cm-keyword{color:#708}.cm-s-default .cm-atom{color:#219}.cm-s-default .cm-number{color:#164}.cm-s-default .cm-def{color:#00f}.cm-s-default .cm-variable-2{color:#05a}.cm-s-default .cm-type,.cm-s-default .cm-variable-3{color:#085}.cm-s-default .cm-comment{color:#a50}.cm-s-default .cm-string{color:#a11}.cm-s-default .cm-string-2{color:#f50}.cm-s-default .cm-meta{color:#555}.cm-s-default .cm-qualifier{color:#555}.cm-s-default .cm-builtin{color:#30a}.cm-s-default .cm-bracket{color:#997}.cm-s-default .cm-tag{color:#170}.cm-s-default .cm-attribute{color:#00c}.cm-s-default .cm-hr{color:#999}.cm-s-default .cm-link{color:#00c}.cm-s-default .cm-error{color:red}.cm-invalidchar{color:red}.CodeMirror-composing{border-bottom:2px solid}div.CodeMirror span.CodeMirror-matchingbracket{color:#0b0}div.CodeMirror span.CodeMirror-nonmatchingbracket{color:#a22}.CodeMirror-matchingtag{background:rgba(255,150,0,.3)}.CodeMirror-activeline-background{background:#e8f2ff}.CodeMirror{position:relative;overflow:hidden;background:#fff}.CodeMirror-scroll{overflow:scroll!important;margin-bottom:-30px;margin-right:-30px;padding-bottom:30px;height:100%;outline:0;position:relative}.CodeMirror-sizer{position:relative;border-right:30px solid transparent}.CodeMirror-gutter-filler,.CodeMirror-hscrollbar,.CodeMirror-scrollbar-filler,.CodeMirror-vscrollbar{position:absolute;z-index:6;display:none}.CodeMirror-vscrollbar{right:0;top:0;overflow-x:hidden;overflow-y:scroll}.CodeMirror-hscrollbar{bottom:0;left:0;overflow-y:hidden;overflow-x:scroll}.CodeMirror-scrollbar-filler{right:0;bottom:0}.CodeMirror-gutter-filler{left:0;bottom:0}.CodeMirror-gutters{position:absolute;left:0;top:0;min-height:100%;z-index:3}.CodeMirror-gutter{white-space:normal;height:100%;display:inline-block;vertical-align:top;margin-bottom:-30px}.CodeMirror-gutter-wrapper{position:absolute;z-index:4;background:0 0!important;border:none!important}.CodeMirror-gutter-background{position:absolute;top:0;bottom:0;z-index:4}.CodeMirror-gutter-elt{position:absolute;cursor:default;z-index:4}.CodeMirror-gutter-wrapper ::selection{background-color:transparent}.CodeMirror-gutter-wrapper ::-moz-selection{background-color:transparent}.CodeMirror-lines{cursor:text;min-height:1px}.CodeMirror pre.CodeMirror-line,.CodeMirror pre.CodeMirror-line-like{-moz-border-radius:0;-webkit-border-radius:0;border-radius:0;border-width:0;background:0 0;font-family:inherit;font-size:inherit;margin:0;white-space:pre;word-wrap:normal;line-height:inherit;color:inherit;z-index:2;position:relative;overflow:visible;-webkit-tap-highlight-color:transparent;-webkit-font-variant-ligatures:contextual;font-variant-ligatures:contextual}.CodeMirror-wrap pre.CodeMirror-line,.CodeMirror-wrap pre.CodeMirror-line-like{word-wrap:break-word;white-space:pre-wrap;word-break:normal}.CodeMirror-linebackground{position:absolute;left:0;right:0;top:0;bottom:0;z-index:0}.CodeMirror-linewidget{position:relative;z-index:2;padding:.1px}.CodeMirror-rtl pre{direction:rtl}.CodeMirror-code{outline:0}.CodeMirror-gutter,.CodeMirror-gutters,.CodeMirror-linenumber,.CodeMirror-scroll,.CodeMirror-sizer{-moz-box-sizing:content-box;box-sizing:content-box}.CodeMirror-measure{position:absolute;width:100%;height:0;overflow:hidden;visibility:hidden}.CodeMirror-cursor{position:absolute;pointer-events:none}.CodeMirror-measure pre{position:static}div.CodeMirror-cursors{visibility:hidden;position:relative;z-index:3}div.CodeMirror-dragcursors{visibility:visible}.CodeMirror-focused div.CodeMirror-cursors{visibility:visible}.CodeMirror-selected{background:#d9d9d9}.CodeMirror-focused .CodeMirror-selected{background:#d7d4f0}.CodeMirror-crosshair{cursor:crosshair}.CodeMirror-line::selection,.CodeMirror-line>span::selection,.CodeMirror-line>span>span::selection{background:#d7d4f0}.CodeMirror-line::-moz-selection,.CodeMirror-line>span::-moz-selection,.CodeMirror-line>span>span::-moz-selection{background:#d7d4f0}.cm-searching{background-color:#ffa;background-color:rgba(255,255,0,.4)}.cm-force-border{padding-right:.1px}@media print{.CodeMirror div.CodeMirror-cursors{visibility:hidden}}.cm-tab-wrap-hack:after{content:''}span.CodeMirror-selectedtext{background:0 0}
2 | /*# sourceMappingURL=codemirror.min.css.map */
--------------------------------------------------------------------------------
/examples/htdocs/assets/toastui/toastui-editor-viewer.min.css:
--------------------------------------------------------------------------------
1 | @charset "utf-8";
2 | /*!
3 | * @toast-ui/editor
4 | * @version 2.5.1 | Tue Nov 24 2020
5 | * @author NHN FE Development Lab
6 | * @license MIT
7 | */.tui-editor-contents{margin:0;padding:0;font-size:13px;font-family:Open Sans,Helvetica Neue,Helvetica,Arial,나눔바른고딕,Nanum Barun Gothic,맑은고딕,Malgun Gothic,sans-serif}.tui-editor-contents :not(table){line-height:160%;box-sizing:content-box}.tui-editor-contents address,.tui-editor-contents cite,.tui-editor-contents dfn,.tui-editor-contents em,.tui-editor-contents i,.tui-editor-contents var{font-style:italic}.tui-editor-contents strong{font-weight:700}.tui-editor-contents p{margin:10px 0;color:#222}.tui-editor-contents>div>div:first-of-type h1,.tui-editor-contents>h1:first-of-type{margin-top:14px}.tui-editor-contents h1,.tui-editor-contents h2,.tui-editor-contents h3,.tui-editor-contents h4,.tui-editor-contents h5,.tui-editor-contents h6{font-weight:700;color:#222}.tui-editor-contents h1{font-size:24px;line-height:28px;border-bottom:3px double #999;margin:52px 0 15px;padding-bottom:7px}.tui-editor-contents h2{font-size:22px;line-height:23px;border-bottom:1px solid #dbdbdb;margin:20px 0 13px;padding-bottom:7px}.tui-editor-contents h3{font-size:20px;margin:18px 0 2px}.tui-editor-contents h4{font-size:18px;margin:10px 0 2px}.tui-editor-contents h3,.tui-editor-contents h4{line-height:18px}.tui-editor-contents h5{font-size:16px}.tui-editor-contents h6{font-size:14px}.tui-editor-contents h5,.tui-editor-contents h6{line-height:17px;margin:9px 0 -4px}.tui-editor-contents del{color:#999}.tui-editor-contents blockquote{margin:14px 0;border-left:4px solid #e5e5e5;padding:0 16px;color:#999}.tui-editor-contents blockquote ol,.tui-editor-contents blockquote p,.tui-editor-contents blockquote ul{color:#999}.tui-editor-contents blockquote>:first-child{margin-top:0}.tui-editor-contents blockquote>:last-child{margin-bottom:0}.tui-editor-contents code,.tui-editor-contents pre{font-family:Consolas,Courier,Apple SD 산돌고딕 Neo,-apple-system,Lucida Grande,Apple SD Gothic Neo,맑은 고딕,Malgun Gothic,Segoe UI,돋움,dotum,sans-serif;border:0;border-radius:0}.tui-editor-contents pre{margin:2px 0 8px;padding:18px;background-color:#f5f7f8}.tui-editor-contents code{color:#c1798b;background-color:#f9f2f4;padding:2px 3px;letter-spacing:-.3px;border-radius:2px}.tui-editor-contents pre code{padding:0;color:inherit;white-space:pre-wrap;background-color:transparent}.tui-editor-contents pre.addon{border:1px solid #e8ebed;background-color:#fff}.tui-editor-contents img{margin:4px 0 10px;box-sizing:border-box;vertical-align:top;max-width:100%}.tui-editor-contents table{border:1px solid rgba(0,0,0,.1);margin:12px 0 14px;color:#222;width:auto;border-collapse:collapse;box-sizing:border-box}.tui-editor-contents table td,.tui-editor-contents table th{border:1px solid rgba(0,0,0,.1);padding:5px 14px 5px 12px;height:32px}.tui-editor-contents table th{background-color:#555;font-weight:300;color:#fff;padding-top:6px}.tui-editor-contents dir,.tui-editor-contents menu,.tui-editor-contents ol,.tui-editor-contents ul{display:block;list-style-type:none;padding-left:24px;margin:6px 0 10px;color:#222}.tui-editor-contents ol{list-style-type:none;counter-reset:li}.tui-editor-contents ol>li{counter-increment:li}.tui-editor-contents ol>li:before,.tui-editor-contents ul>li:before{display:inline-block;position:absolute}.tui-editor-contents ul>li:before{content:"";margin-top:6px;margin-left:-17px;width:5px;height:5px;border-radius:50%;background-color:#ccc}.tui-editor-contents ol>li:before{content:"." counter(li);margin-left:-28px;width:24px;text-align:right;direction:rtl;color:#aaa}.tui-editor-contents ol ol,.tui-editor-contents ol ul,.tui-editor-contents ul ol,.tui-editor-contents ul ul{margin-top:0!important;margin-bottom:0!important}.tui-editor-contents ol li,.tui-editor-contents ul li{position:relative}.tui-editor-contents ol p,.tui-editor-contents ul p{margin:0}.tui-editor-contents ol li.task-list-item:before,.tui-editor-contents pre ul li:before,.tui-editor-contents ul li.task-list-item:before{content:""}.tui-editor-contents th ol,.tui-editor-contents th ul{color:#fff}.tui-editor-contents hr{border-top:1px solid #eee;margin:16px 0}.tui-editor-contents a{text-decoration:underline;color:#4b96e6}.tui-editor-contents a:hover{color:#1f70de}.tui-editor-contents a.image-link{position:relative}.tui-editor-contents a.image-link:before{content:"";position:absolute;margin:0;width:20px;height:20px;top:2px;right:2px;background-repeat:no-repeat;background-image:url("");cursor:pointer}.tui-editor-contents .task-list-item{border:0;list-style:none;padding-left:24px;margin-left:-24px}.tui-editor-contents .task-list-item:before{background-repeat:no-repeat;background-size:18px 18px;background-position:50%;content:"";margin-left:0;margin-top:0;border-radius:0;height:18px;width:18px;position:absolute;left:0;top:1px;cursor:pointer;background-image:url("")}.tui-editor-contents .task-list-item.checked:before{background-image:url("")}.tui-editor-contents .task-list-item .task-list-item-checkbox,.tui-editor-contents .task-list-item input[type=checkbox]{margin-left:-17px;margin-right:3.8px;margin-top:3px}.tui-editor-contents-placeholder:before{content:attr(data-placeholder);color:grey;line-height:160%;position:absolute}.te-preview .tui-editor-contents h1{min-height:28px}.te-preview .tui-editor-contents h2{min-height:23px}.te-preview .tui-editor-contents blockquote{min-height:20px}.te-preview .tui-editor-contents li{min-height:22px}@media (-ms-high-contrast:active),(-ms-high-contrast:none){.te-ww-container .tui-editor-contents li{vertical-align:middle}.te-ww-container .tui-editor-contents .task-list-item:before,.te-ww-container .tui-editor-contents ol>li:before,.te-ww-container .tui-editor-contents ul>li:before{position:static;vertical-align:middle}.te-ww-container .tui-editor-contents ul>li:before{margin-top:-3px;margin-right:12px}.te-ww-container .tui-editor-contents ol>li:before{margin-right:6px}.te-ww-container .tui-editor-contents .task-list-item{padding-left:2px}}
--------------------------------------------------------------------------------
/examples/htdocs/assets/toastui/toastui-editor.min.css:
--------------------------------------------------------------------------------
1 | @charset "utf-8";
2 | /*!
3 | * @toast-ui/editor
4 | * @version 2.5.1 | Tue Nov 24 2020
5 | * @author NHN FE Development Lab
6 | * @license MIT
7 | */.auto-height,.auto-height .tui-editor-defaultUI{height:auto}.auto-height .tui-editor{position:relative}:not(.auto-height)>.tui-editor-defaultUI,:not(.auto-height)>.tui-editor-defaultUI>.te-editor-section{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column}:not(.auto-height)>.tui-editor-defaultUI>.te-editor-section{-ms-flex:1;flex:1}.tui-editor-defaultUI-toolbar:after,.tui-editor:after{content:"";display:block;height:0;clear:both}.tui-editor{position:absolute;line-height:1;color:#222;width:100%;height:inherit}.te-editor-section{min-height:0;position:relative;height:inherit}.te-md-container{display:none;overflow:hidden;height:100%}.te-md-container .te-editor{line-height:1.5}.te-md-container .te-editor,.te-md-container .te-preview{box-sizing:border-box;padding:0;height:inherit}.te-md-container .CodeMirror{font-size:13px;height:inherit}.te-md-container .te-preview{overflow:auto;padding:0 25px;height:100%}.te-md-container .te-preview>p:first-child{margin-top:0!important}.te-md-container .te-preview .tui-editor-contents{padding-top:8px}.tui-editor .te-preview-style-tab>.te-editor,.tui-editor .te-preview-style-tab>.te-preview{float:left;width:100%;display:none}.tui-editor .te-preview-style-tab>.te-tab-active{display:block}.tui-editor .te-preview-style-vertical>.te-tab-section{display:none}.tui-editor .te-preview-style-tab>.te-tab-section{display:block}.tui-editor .te-preview-style-vertical .te-editor,.tui-editor .te-preview-style-vertical .te-preview{float:left;width:50%}.tui-editor .te-md-splitter{display:none;position:absolute;left:50%;top:0;height:100%;width:1px;border-left:1px solid #e5e5e5}.tui-editor .te-preview-style-vertical .te-md-splitter{display:block}.te-ww-container{display:none;overflow:hidden;z-index:10;height:inherit;background-color:#fff}.te-ww-container>.te-editor{overflow:auto;height:inherit}.te-ww-container .tui-editor-contents:focus{outline:none}.te-ww-container .tui-editor-contents{padding:0 25px}.te-ww-container .tui-editor-contents:first-child{box-sizing:border-box;margin:0;padding:16px 25px 0;height:inherit}.te-ww-container .tui-editor-contents:last-child{margin-bottom:16px}.te-md-mode .te-md-container,.te-ww-mode .te-ww-container{display:block;z-index:100}.tui-editor-defaultUI.te-hide,.tui-editor.te-hide{display:none}.tui-editor-defaultUI .CodeMirror-lines{padding-top:18px;padding-bottom:18px}.tui-editor-defaultUI pre.CodeMirror-line{padding-left:25px;padding-right:25px}.tui-editor-defaultUI .CodeMirror pre.CodeMirror-placeholder{margin:0;padding-left:25px;color:grey}.tui-editor-defaultUI .CodeMirror-scroll{cursor:text}.tui-editor-contents td.te-cell-selected{background-color:#d8dfec}.tui-editor-contents td.te-cell-selected::selection{background-color:#d8dfec}.tui-editor-contents th.te-cell-selected{background-color:#908f8f}.tui-editor-contents th.te-cell-selected::selection{background-color:#908f8f}.tui-editor-defaultUI{position:relative;border:1px solid #e5e5e5;height:100%;font-family:Open Sans,Helvetica Neue,Helvetica,Arial,나눔바른고딕,Nanum Barun Gothic,맑은고딕,Malgun Gothic,sans-serif}.tui-editor-defaultUI button{color:#fff;padding:0 14px 0 15px;height:28px;font-size:12px;border:none;cursor:pointer;outline:none}.tui-editor-defaultUI button.te-ok-button{background-color:#4b96e6}.tui-editor-defaultUI button.te-close-button{background-color:#777}.tui-editor-defaultUI-toolbar{padding:0 25px;height:31px;background-color:#fff;border:0;overflow:hidden}.tui-toolbar-divider{float:left;display:inline-block;width:1px;height:14px;background-color:#ddd;margin:9px 6px}.tui-toolbar-button-group{height:28px;border-right:1px solid #d9d9d9;float:left}.te-toolbar-section{height:32px;box-sizing:border-box;border-bottom:1px solid #e5e5e5}.tui-editor-defaultUI-toolbar button{float:left;box-sizing:border-box;outline:none;cursor:pointer;background-color:#fff;width:22px;height:22px;padding:3px;border-radius:0;margin:5px 3px;border:1px solid #fff}.tui-editor-defaultUI-toolbar button.active,.tui-editor-defaultUI-toolbar button:active,.tui-editor-defaultUI-toolbar button:hover{border:1px solid #aaa;background-color:#fff}.tui-editor-defaultUI-toolbar button:first-child{margin-left:0}.tui-editor-defaultUI-toolbar button:last-child{margin-right:0}.tui-editor-defaultUI-toolbar button.tui-scrollsync{width:auto;color:#777;border:0}.tui-editor-defaultUI button.tui-scrollsync:after{content:"Scroll off"}.tui-editor-defaultUI button.tui-scrollsync.active{color:#4b96e6;font-weight:700}.tui-editor-defaultUI button.tui-scrollsync.active:after{content:"Scroll on"}.tui-editor-defaultUI .te-mode-switch-section{background-color:#f9f9f9;border-top:1px solid #e5e5e5;height:20px;font-size:12px}.tui-editor-defaultUI .te-mode-switch{float:right;height:100%}.tui-editor-defaultUI .te-switch-button{width:92px;height:inherit;background:#e5e5e5;outline:0;color:#a0aabf;cursor:pointer;border:0;border-left:1px solid #ddd;border-right:1px solid #ddd}.tui-editor-defaultUI .te-switch-button.active{background-color:#fff;color:#000}.tui-editor-defaultUI .te-markdown-tab-section{float:left;height:31px;background:#fff}.te-markdown-tab-section .te-tab{margin:0 -7px 0 24px;background:#fff}.tui-editor-defaultUI .te-tab button{box-sizing:border-box;line-height:100%;position:relative;cursor:pointer;z-index:1;font-size:13px;background-color:#f9f9f9;border:1px solid #e5e5e5;border-top:0;padding:0 9px;color:#777;border-radius:0;outline:0}.te-markdown-tab-section .te-tab button:last-child{margin-left:-1px}.te-markdown-tab-section .te-tab button.te-tab-active,.te-markdown-tab-section .te-tab button:hover.te-tab-active{background-color:#fff;color:#333;border-bottom:1px solid #fff;z-index:2}.te-markdown-tab-section .te-tab button:hover{background-color:#fff;color:#333}.tui-popup-modal-background{background-color:hsla(0,0%,79.2%,.6);position:fixed;margin:0;left:0;top:0;width:100%;height:100%;z-index:9999}.tui-popup-modal-background.fit-window .tui-popup-wrapper,.tui-popup-wrapper.fit-window{width:100%;height:100%}.tui-popup-wrapper{width:500px;margin-right:auto;border:1px solid #cacaca;background:#fff;z-index:9999}.tui-popup-modal-background .tui-popup-wrapper{position:absolute;margin:auto;top:0;right:0;bottom:0;left:0}.tui-popup-header{padding:10px;height:auto;line-height:normal;position:relative;border-bottom:1px solid #cacaca}.tui-popup-header .tui-popup-header-buttons{float:right}.tui-popup-header .tui-popup-header-buttons button{padding:0;background-color:transparent;background-size:cover;float:left}.tui-popup-header .tui-popup-close-button{margin:3px;width:13px;height:13px;background-image:url()}.tui-popup-header .tui-popup-title{font-size:13px;font-weight:700;color:#333;vertical-align:bottom}.tui-popup-body{padding:15px;font-size:12px}.tui-editor-popup{position:absolute;top:30px;left:50%;margin-left:-250px}.tui-editor-popup.tui-popup-modal-background{position:fixed;top:0;left:0;margin:0}.tui-editor-popup .tui-popup-body label{font-weight:700;color:#666;display:block;margin:10px 0 5px}.tui-editor-popup .tui-popup-body .te-button-section{margin-top:15px}.tui-editor-popup .tui-popup-body input[type=file],.tui-editor-popup .tui-popup-body input[type=text]{padding:4px 10px;border:1px solid #bfbfbf;box-sizing:border-box;width:100%}.tui-editor-popup .tui-popup-body input[type=text].disabled{border-color:#e5e5e5;background-color:#eee;color:#e5e5e5}.tui-editor-popup .tui-popup-body input.wrong{border-color:red}.te-popup-add-link .tui-popup-wrapper{height:219px}.te-popup-add-image .tui-popup-wrapper{height:243px}.te-popup-add-image .te-tab{display:block;background:none;border-bottom:1px solid #ebebeb;margin-bottom:8px}.te-popup-add-image .te-file-type,.te-popup-add-image .te-url-type{display:none}.te-popup-add-image div.te-tab-active,.te-popup-add-image form.te-tab-active{display:block}.te-popup-add-image .te-tab button{border:1px solid #ccc;background:#eee;min-width:100px;margin-left:-1px;border-bottom:0;border-radius:3px 3px 0 0}.te-popup-add-image .te-tab button.te-tab-active{background:#fff}.te-popup-add-table .te-table-selection{position:relative}.te-popup-add-table .te-table-body{background-image:url("")}.te-popup-add-table .te-table-header{background-image:url("")}.te-popup-add-table .te-selection-area{position:absolute;top:0;left:0;background:#80d2ff;opacity:.3;z-index:999}.te-popup-add-table .te-description{margin:10px 0 0;text-align:center}.te-popup-table-utils{width:auto;min-width:120px}.te-popup-table-utils .tui-popup-body{padding:0}.te-popup-table-utils button{display:block;width:100%;background-color:#fff;border:none;outline:0;padding:0 10px;font-size:12px;line-height:28px;text-align:left;color:#777}.te-popup-table-utils button:hover{background-color:#f4f4f4}.te-popup-table-utils hr{margin:0;background-color:#cacaca;border-style:none;height:1px}.te-popup-table-utils .te-context-menu-disabled{color:#ccc}.te-popup-table-utils .te-context-menu-disabled:hover{background-color:#fff}.te-heading-add{width:auto}.te-heading-add .tui-popup-body{padding:0}.te-heading-add h1,.te-heading-add h2,.te-heading-add h3,.te-heading-add h4,.te-heading-add h5,.te-heading-add h6,.te-heading-add p,.te-heading-add ul{padding:0;margin:0}.te-heading-add ul{list-style:none}.te-heading-add ul li{padding:2px 10px;cursor:pointer}.te-heading-add ul li:hover{background-color:#eee}.te-heading-add h1{font-size:24px}.te-heading-add h2{font-size:22px}.te-heading-add h3{font-size:20px}.te-heading-add h4{font-size:18px}.te-heading-add h5{font-size:16px}.te-heading-add h6{font-size:14px}.te-dropdown-toolbar{position:absolute;width:auto}.te-dropdown-toolbar .tui-popup-body,.tui-popup-color{padding:0}.tui-popup-color .tui-colorpicker-container,.tui-popup-color .tui-colorpicker-palette-container{width:144px}.tui-popup-color .tui-colorpicker-container ul{width:144px;margin-bottom:8px}.tui-popup-color .tui-colorpicker-container li{padding:0 1px 1px 0}.tui-popup-color .tui-colorpicker-container li .tui-colorpicker-palette-button{border:0;width:17px;height:17px}.tui-popup-color .tui-popup-body{padding:10px}.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-toggle-slider{display:none}.tui-popup-color .te-apply-button,.tui-popup-color .tui-colorpicker-palette-hex{float:right}.tui-popup-color .te-apply-button{height:21px;width:35px;background:#fff;border:1px solid #efefef;position:absolute;bottom:135px;right:10px;color:#000}.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-hex{border:1px solid #e1e1e1;padding:3px 14px;margin-left:-1px}.tui-popup-color .tui-colorpicker-container div.tui-colorpicker-clearfix{display:inline-block}.tui-popup-color .tui-colorpicker-container .tui-colorpicker-palette-preview{width:19px;height:19px}.tui-popup-color .tui-colorpicker-slider-container .tui-colorpicker-slider-right{width:22px}.tui-popup-color .tui-colorpicker-slider-container .tui-colorpicker-huebar-handle{display:none}.tui-tooltip{z-index:999;opacity:.8;color:#fff;padding:2px 5px;font-size:10px}.tui-tooltip,.tui-tooltip .arrow{position:absolute;background-color:#222}.tui-tooltip .arrow{content:"";display:inline-block;width:10px;height:10px;-webkit-transform:rotate(45deg);-moz-transform:rotate(45deg);-ms-transform:rotate(45deg);-o-transform:rotate(45deg);transform:rotate(45deg);top:-3px;left:6px;z-index:-1}.tui-toolbar-icons{background:url();background-size:218px 188px;display:inline-block}@media only screen and (-o-min-device-pixel-ratio:2/1),only screen and (-webkit-min-device-pixel-ratio:2),only screen and (min--moz-device-pixel-ratio:2),only screen and (min-device-pixel-ratio:2),only screen and (min-resolution:2dppx),only screen and (min-resolution:192dpi){.tui-toolbar-icons{background:url();background-size:218px 188px;display:inline-block}}.tui-toolbar-icons.tui-heading{background-position:-172px -48px}.tui-toolbar-icons.tui-heading:disabled{background-position:-193px -48px}.tui-toolbar-icons.tui-bold{background-position:-4px -4px}.tui-toolbar-icons.tui-bold:disabled{background-position:-25px -4px}.tui-toolbar-icons.tui-italic{background-position:-4px -48px}.tui-toolbar-icons.tui-italic:disabled{background-position:-25px -48px}.tui-toolbar-icons.tui-color{background-position:-172px -70px}.tui-toolbar-icons.tui-color:disabled{background-position:-193px -70px}.tui-toolbar-icons.tui-strike{background-position:-4px -26px}.tui-toolbar-icons.tui-strike:disabled{background-position:-25px -26px}.tui-toolbar-icons.tui-hrline{background-position:-46px -92px}.tui-toolbar-icons.tui-hrline:disabled{background-position:-67px -92px}.tui-toolbar-icons.tui-quote{background-position:-4px -114px}.tui-toolbar-icons.tui-quote:disabled{background-position:-25px -114px}.tui-toolbar-icons.tui-ul{background-position:-46px -4px}.tui-toolbar-icons.tui-ul:disabled{background-position:-67px -4px}.tui-toolbar-icons.tui-ol{background-position:-46px -26px}.tui-toolbar-icons.tui-ol:disabled{background-position:-67px -26px}.tui-toolbar-icons.tui-task{background-position:-130px -48px}.tui-toolbar-icons.tui-task:disabled{background-position:-151px -48px}.tui-toolbar-icons.tui-indent{background-position:-46px -48px}.tui-toolbar-icons.tui-indent:disabled{background-position:-67px -48px}.tui-toolbar-icons.tui-outdent{background-position:-46px -70px}.tui-toolbar-icons.tui-outdent:disabled{background-position:-67px -70px}.tui-toolbar-icons.tui-table{background-position:-88px -92px}.tui-toolbar-icons.tui-table:disabled{background-position:-109px -92px}.tui-toolbar-icons.tui-image{background-position:-130px -4px}.tui-toolbar-icons.tui-image:disabled{background-position:-151px -4px}.tui-toolbar-icons.tui-link{background-position:-130px -26px}.tui-toolbar-icons.tui-link:disabled{background-position:-151px -26px}.tui-toolbar-icons.tui-code{background-position:-130px -92px}.tui-toolbar-icons.tui-code:disabled{background-position:-151px -92px}.tui-toolbar-icons.tui-codeblock{background-position:-130px -70px}.tui-toolbar-icons.tui-codeblock:disabled{background-position:-151px -70px}.tui-toolbar-icons.tui-more{background-position:-172px -92px}.tui-toolbar-icons.tui-more:disabled{background-position:-193px -92px}.tui-colorpicker-svg-huebar,.tui-colorpicker-svg-slider,.tui-colorpicker-vml-slider{border:1px solid #ebebeb}.tui-editor-pseudo-clipboard{position:fixed;left:-1000px;top:-1000px;width:100px;height:100px}.te-ww-block-overlay.code-block-header{text-align:right;font-family:Open Sans,Helvetica Neue,Helvetica,Arial,sans-serif}.te-ww-block-overlay.code-block-header span{font-size:10px;font-weight:600;padding:0 10px;color:#333;cursor:default}.te-ww-block-overlay.code-block-header button{margin:8px;font-size:10px;color:#333;background-color:#f9f9f9;border:1px solid #ddd;padding:4px;height:auto}.te-popup-code-block-languages{position:fixed;box-sizing:border-box;width:130px}.te-popup-code-block-languages .tui-popup-body{max-height:169px;overflow:auto;padding:0}.te-popup-code-block-languages button{width:100%;background-color:#fff;border:none;outline:0;padding:0 10px;font-size:12px;line-height:24px;text-align:left;color:#777}.te-popup-code-block-languages button.active{background-color:#f4f4f4}.tui-popup-code-block-editor .tui-popup-wrapper{width:70%;height:70%;margin:auto;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column}.te-input-language{position:relative;margin-left:15px;cursor:pointer}.te-input-language input{font-family:Open Sans,Helvetica Neue,Helvetica,Arial,sans-serif;font-size:10px;padding:3px 5px;border:1px solid #ddd;background-color:#f9f9f9;box-sizing:border-box;width:130px;outline:none}.te-input-language input::-ms-clear{display:none}.te-input-language:after{content:url();position:absolute;top:1px;right:3px}.te-input-language.active:after{content:url()}.tui-popup-code-block-editor button{margin:-1px 3px}.tui-popup-code-block-editor .tui-popup-header-buttons{height:20px}.tui-popup-code-block-editor .popup-editor-toggle-preview:after{content:"Preview off";color:#777;margin-right:22px}.tui-popup-code-block-editor .popup-editor-toggle-preview.active:after{content:"Preview on";color:#4b96e6}.tui-popup-code-block-editor .popup-editor-toggle-scroll:after{content:"Scroll off";color:#777;margin-right:16px}.tui-popup-code-block-editor .popup-editor-toggle-scroll.active:after{content:"Scroll on";color:#4b96e6}.tui-popup-code-block-editor .popup-editor-toggle-fit{width:18px;height:18px;margin-top:4px;margin-right:14px;background-image:url()}.tui-popup-code-block-editor .popup-editor-toggle-fit.active{background-image:url()}.tui-popup-code-block-editor .tui-popup-close-button{margin-top:6px}.tui-popup-code-block-editor .tui-popup-body{z-index:-1;padding:0;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex:1;flex:1}.tui-popup-code-block-editor .popup-editor-body{position:relative;-ms-flex:1;flex:1;border-bottom:1px solid #cacaca}.tui-popup-code-block-editor .te-button-section{padding:15px}.tui-popup-code-block-editor .te-button-section button{float:left}.tui-popup-code-block-editor .tui-editor-contents pre{margin:0;background-color:transparent}.tui-popup-code-block-editor .CodeMirror{height:auto}.tui-popup-code-block-editor .CodeMirror-line{font-family:Consolas,Courier,Lucida Grande,나눔바른고딕,Nanum Barun Gothic,맑은고딕,Malgun Gothic,sans-serif;font-size:13px;line-height:160%;letter-spacing:-.3px}.tui-popup-code-block-editor .popup-editor-editor-wrapper{min-height:100%}.tui-split-scroll-wrapper{position:relative}.tui-split-scroll{position:absolute}.tui-split-scroll,.tui-split-scroll-wrapper{width:100%;height:100%}.tui-split-scroll .tui-split-content-left,.tui-split-scroll .tui-split-content-right{position:absolute;top:0;width:50%;box-sizing:border-box}.tui-split-scroll .tui-split-content-left{left:0}.tui-split-scroll .tui-split-content-right{left:50%}.tui-split-scroll .tui-splitter{position:absolute;left:50%;top:0;height:100%;width:1px;border-left:1px solid #cacaca}.tui-split-scroll .tui-split-scroll-content{width:100%;height:100%;overflow:hidden;position:relative}.tui-split-scroll .tui-split-content-left,.tui-split-scroll .tui-split-content-right{height:100%;overflow-x:hidden;overflow-y:auto}.tui-split-scroll button.tui-scrollsync{top:10px;opacity:.2}.tui-split-scroll button.tui-scrollsync:after{content:"scroll off"}.tui-split-scroll.scroll-sync button.tui-scrollsync{opacity:.5}.tui-split-scroll.scroll-sync .tui-split-content-left,.tui-split-scroll.scroll-sync .tui-split-content-right{height:auto;overflow:initial}.tui-split-scroll.scroll-sync button.tui-scrollsync:after{content:"scroll on"}.tui-split-scroll.scroll-sync .tui-split-scroll-content{overflow-y:auto}.tui-split-scroll.single-content .tui-splitter{display:none}.tui-split-scroll.single-content .tui-split-content-left{width:100%}.tui-split-scroll.single-content .tui-split-content-right,.tui-split-scroll.single-content button.tui-scrollsync{display:none}@media (-ms-high-contrast:active),(-ms-high-contrast:none){.tui-split-scroll-wrapper .tui-splitter{left:calc(50% - 9px)}}@supports (-ms-accelerator:true){.tui-split-scroll-wrapper .tui-splitter{left:calc(50% - 9px)}}@media screen and (max-width:480px){.tui-popup-wrapper{max-width:300px}.tui-editor-popup{margin-left:-150px}.te-dropdown-toolbar{max-width:none}}.tui-editor-contents{margin:0;padding:0;font-size:13px;font-family:Open Sans,Helvetica Neue,Helvetica,Arial,나눔바른고딕,Nanum Barun Gothic,맑은고딕,Malgun Gothic,sans-serif}.tui-editor-contents :not(table){line-height:160%;box-sizing:content-box}.tui-editor-contents address,.tui-editor-contents cite,.tui-editor-contents dfn,.tui-editor-contents em,.tui-editor-contents i,.tui-editor-contents var{font-style:italic}.tui-editor-contents strong{font-weight:700}.tui-editor-contents p{margin:10px 0;color:#222}.tui-editor-contents>div>div:first-of-type h1,.tui-editor-contents>h1:first-of-type{margin-top:14px}.tui-editor-contents h1,.tui-editor-contents h2,.tui-editor-contents h3,.tui-editor-contents h4,.tui-editor-contents h5,.tui-editor-contents h6{font-weight:700;color:#222}.tui-editor-contents h1{font-size:24px;line-height:28px;border-bottom:3px double #999;margin:52px 0 15px;padding-bottom:7px}.tui-editor-contents h2{font-size:22px;line-height:23px;border-bottom:1px solid #dbdbdb;margin:20px 0 13px;padding-bottom:7px}.tui-editor-contents h3{font-size:20px;margin:18px 0 2px}.tui-editor-contents h4{font-size:18px;margin:10px 0 2px}.tui-editor-contents h3,.tui-editor-contents h4{line-height:18px}.tui-editor-contents h5{font-size:16px}.tui-editor-contents h6{font-size:14px}.tui-editor-contents h5,.tui-editor-contents h6{line-height:17px;margin:9px 0 -4px}.tui-editor-contents del{color:#999}.tui-editor-contents blockquote{margin:14px 0;border-left:4px solid #e5e5e5;padding:0 16px;color:#999}.tui-editor-contents blockquote ol,.tui-editor-contents blockquote p,.tui-editor-contents blockquote ul{color:#999}.tui-editor-contents blockquote>:first-child{margin-top:0}.tui-editor-contents blockquote>:last-child{margin-bottom:0}.tui-editor-contents code,.tui-editor-contents pre{font-family:Consolas,Courier,Apple SD 산돌고딕 Neo,-apple-system,Lucida Grande,Apple SD Gothic Neo,맑은 고딕,Malgun Gothic,Segoe UI,돋움,dotum,sans-serif;border:0;border-radius:0}.tui-editor-contents pre{margin:2px 0 8px;padding:18px;background-color:#f5f7f8}.tui-editor-contents code{color:#c1798b;background-color:#f9f2f4;padding:2px 3px;letter-spacing:-.3px;border-radius:2px}.tui-editor-contents pre code{padding:0;color:inherit;white-space:pre-wrap;background-color:transparent}.tui-editor-contents pre.addon{border:1px solid #e8ebed;background-color:#fff}.tui-editor-contents img{margin:4px 0 10px;box-sizing:border-box;vertical-align:top;max-width:100%}.tui-editor-contents table{border:1px solid rgba(0,0,0,.1);margin:12px 0 14px;color:#222;width:auto;border-collapse:collapse;box-sizing:border-box}.tui-editor-contents table td,.tui-editor-contents table th{border:1px solid rgba(0,0,0,.1);padding:5px 14px 5px 12px;height:32px}.tui-editor-contents table th{background-color:#555;font-weight:300;color:#fff;padding-top:6px}.tui-editor-contents dir,.tui-editor-contents menu,.tui-editor-contents ol,.tui-editor-contents ul{display:block;list-style-type:none;padding-left:24px;margin:6px 0 10px;color:#222}.tui-editor-contents ol{list-style-type:none;counter-reset:li}.tui-editor-contents ol>li{counter-increment:li}.tui-editor-contents ol>li:before,.tui-editor-contents ul>li:before{display:inline-block;position:absolute}.tui-editor-contents ul>li:before{content:"";margin-top:6px;margin-left:-17px;width:5px;height:5px;border-radius:50%;background-color:#ccc}.tui-editor-contents ol>li:before{content:"." counter(li);margin-left:-28px;width:24px;text-align:right;direction:rtl;color:#aaa}.tui-editor-contents ol ol,.tui-editor-contents ol ul,.tui-editor-contents ul ol,.tui-editor-contents ul ul{margin-top:0!important;margin-bottom:0!important}.tui-editor-contents ol li,.tui-editor-contents ul li{position:relative}.tui-editor-contents ol p,.tui-editor-contents ul p{margin:0}.tui-editor-contents ol li.task-list-item:before,.tui-editor-contents pre ul li:before,.tui-editor-contents ul li.task-list-item:before{content:""}.tui-editor-contents th ol,.tui-editor-contents th ul{color:#fff}.tui-editor-contents hr{border-top:1px solid #eee;margin:16px 0}.tui-editor-contents a{text-decoration:underline;color:#4b96e6}.tui-editor-contents a:hover{color:#1f70de}.tui-editor-contents a.image-link{position:relative}.tui-editor-contents a.image-link:before{content:"";position:absolute;margin:0;width:20px;height:20px;top:2px;right:2px;background-repeat:no-repeat;background-image:url("");cursor:pointer}.tui-editor-contents .task-list-item{border:0;list-style:none;padding-left:24px;margin-left:-24px}.tui-editor-contents .task-list-item:before{background-repeat:no-repeat;background-size:18px 18px;background-position:50%;content:"";margin-left:0;margin-top:0;border-radius:0;height:18px;width:18px;position:absolute;left:0;top:1px;cursor:pointer;background-image:url("")}.tui-editor-contents .task-list-item.checked:before{background-image:url("")}.tui-editor-contents .task-list-item .task-list-item-checkbox,.tui-editor-contents .task-list-item input[type=checkbox]{margin-left:-17px;margin-right:3.8px;margin-top:3px}.tui-editor-contents-placeholder:before{content:attr(data-placeholder);color:grey;line-height:160%;position:absolute}.te-preview .tui-editor-contents h1{min-height:28px}.te-preview .tui-editor-contents h2{min-height:23px}.te-preview .tui-editor-contents blockquote{min-height:20px}.te-preview .tui-editor-contents li{min-height:22px}@media (-ms-high-contrast:active),(-ms-high-contrast:none){.te-ww-container .tui-editor-contents li{vertical-align:middle}.te-ww-container .tui-editor-contents .task-list-item:before,.te-ww-container .tui-editor-contents ol>li:before,.te-ww-container .tui-editor-contents ul>li:before{position:static;vertical-align:middle}.te-ww-container .tui-editor-contents ul>li:before{margin-top:-3px;margin-right:12px}.te-ww-container .tui-editor-contents ol>li:before{margin-right:6px}.te-ww-container .tui-editor-contents .task-list-item{padding-left:2px}}.tui-editor-contents .te-preview-highlight{position:relative;z-index:0}.tui-editor-contents .te-preview-highlight:after{content:"";background-color:rgba(255,245,131,.5);border-radius:4px;z-index:-1;position:absolute;top:-4px;right:-4px;left:-4px;bottom:-4px}.tui-editor-contents h1.te-preview-highlight:after,.tui-editor-contents h2.te-preview-highlight:after{bottom:0}.tui-editor-contents td.te-preview-highlight:after,.tui-editor-contents th.te-preview-highlight:after{display:none}.tui-editor-contents td.te-preview-highlight,.tui-editor-contents th.te-preview-highlight{background-color:rgba(255,245,131,.5)}.tui-editor-contents th.te-preview-highlight{color:#222}.te-md-container .CodeMirror{font-family:Open Sans,Helvetica Neue,Helvetica,Arial,나눔바른고딕,Nanum Barun Gothic,맑은고딕,Malgun Gothic,sans-serif;color:#222}.tui-md-heading1{font-size:24px}.tui-md-heading2{font-size:22px}.tui-md-heading3{font-size:20px}.tui-md-heading4{font-size:18px}.tui-md-heading5{font-size:16px}.tui-md-heading6{font-size:14px}.tui-md-heading.tui-md-delimiter.setext{line-height:15px}.tui-md-heading,.tui-md-list-item.tui-md-list-item-bullet,.tui-md-list-item.tui-md-meta,.tui-md-strong{font-weight:700}.tui-md-emph{font-style:italic}.tui-md-strike{text-decoration:line-through}.tui-md-strike.tui-md-delimiter{text-decoration:none}.tui-md-block-quote,.tui-md-delimiter,.tui-md-link,.tui-md-table,.tui-md-thematic-break{color:#ccc}.tui-md-code-block.tui-md-meta,.tui-md-code.tui-md-delimiter{color:#aaa}.tui-md-html,.tui-md-link.tui-md-link-url.tui-md-marked-text,.tui-md-meta{color:#999}.tui-md-block-quote.tui-md-marked-text,.tui-md-list-item.tui-md-meta{color:#555}.tui-md-table.tui-md-marked-text{color:#222}.tui-md-link.tui-md-link-desc.tui-md-marked-text,.tui-md-list-item-odd.tui-md-list-item-bullet{color:#4b96e6}.tui-md-list-item-even.tui-md-list-item-bullet{color:#cb4848}.tui-md-code.tui-md-marked-text{color:#c1798b}.tui-md-code{background-color:rgba(243,229,233,.5);padding:2px 0;letter-spacing:-.3px}.tui-md-code.tui-md-delimiter.start{padding-left:2px;border-top-left-radius:2px;border-bottom-left-radius:2px}.tui-md-code.tui-md-delimiter.end{padding-right:2px;border-top-right-radius:2px;border-bottom-right-radius:2px}.tui-md-code-block.CodeMirror-linebackground{left:20px;right:20px;background-color:#f5f7f8}.tui-md-code-block.CodeMirror-linebackground.start{top:2px}.tui-md-code-block.CodeMirror-linebackground.end{bottom:2px}.tui-md-code,.tui-md-code-block{font-family:Consolas,Courier,Lucida Grande,나눔바른고딕,Nanum Barun Gothic,맑은고딕,Malgun Gothic,sans-serif}
--------------------------------------------------------------------------------
/examples/htdocs/calc.css:
--------------------------------------------------------------------------------
1 | body {
2 | font-family: sans-serif;
3 | background: #ededed;
4 | }
5 | div.toolbar {
6 | text-align: right;
7 | }
8 | .line {
9 | width: 90%;
10 | height: 4rem;
11 | font-size: 3rem;
12 | text-align: right;
13 | text-overflow: ellipsis;
14 | }
15 | table {
16 | width: 100%;
17 | }
18 | button {
19 | background: none;
20 | border: none;
21 | overflow: hidden;
22 | margin: 1px;
23 | height: 3rem;
24 | font-size: 1.5rem;
25 | }
26 | table button {
27 | background: #fbfbfb;
28 | width: 100%;
29 | }
30 | table button.input {
31 | background: #f3f3f3;
32 | }
33 | table button:hover, button:hover {
34 | background: #c0c0c0;
35 | }
36 | button:active {
37 | transform: scale(0.9);
38 | }
39 |
--------------------------------------------------------------------------------
/examples/htdocs/calc.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Calc
6 |
7 |
8 |
9 |
10 |
11 |
12 | ↻
13 |
14 | ⚙
15 |
16 |
17 | {{ value }}
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 | C
30 | ←
31 | /
32 |
33 |
34 | 7
35 | 8
36 | 9
37 | *
38 |
39 |
40 | 4
41 | 5
42 | 6
43 | -
44 |
45 |
46 | 1
47 | 2
48 | 3
49 | +
50 |
51 |
52 |
53 | 0
54 | .
55 | =
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
--------------------------------------------------------------------------------
/examples/htdocs/calc.js:
--------------------------------------------------------------------------------
1 | var calc = new Vue({
2 | el: '#calc',
3 | data: {
4 | seen: false,
5 | value: '0'
6 | },
7 | methods: {
8 | clearLine: function() {
9 | this.value = '0';
10 | },
11 | append: function(value) {
12 | if (this.value == '0') {
13 | this.value = value;
14 | } else {
15 | this.value += value;
16 | }
17 | },
18 | backspace: function() {
19 | if (this.value.length > 1) {
20 | this.value = this.value.substring(0, this.value.length - 1);
21 | } else {
22 | this.value = '0';
23 | }
24 | },
25 | calculate: function() {
26 | var calc = this;
27 | fetch('rest/calculate', {
28 | method: 'POST',
29 | headers: {
30 | "Content-Type": "application/json"
31 | },
32 | body: JSON.stringify({
33 | line: calc.value
34 | })
35 | }).then(function(response) {
36 | return response.json();
37 | }).then(function(response) {
38 | calc.value = '' + response.line;
39 | });
40 | }
41 | }
42 | });
43 |
--------------------------------------------------------------------------------
/examples/htdocs/file.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | File
6 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
This example shows how to choose a file to operate using Lua.
37 |
38 |
Using a dedicated panel
39 |
Open
40 |
Multiple
41 |
Save
42 |
Directory
43 |
{{ filename }}
44 |
45 |
46 |
Using a single input field
47 |
48 |
49 |
50 |
71 |
--------------------------------------------------------------------------------
/examples/htdocs/md.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Markdown Editor
6 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
60 |
84 |
--------------------------------------------------------------------------------
/examples/htdocs/simple.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Simple WebView Example
5 |
6 |
7 |
8 | Initializing...
9 | Clear
10 | User Agent
11 | Show Hello
12 | Change Title
13 | Say Hello
14 | Show Lua Date
15 | Throw Error
16 | Timer
17 |
18 | Print Lua:
19 | Hello
20 | Args
21 | Locale
22 | Date
23 | Seconds
24 | Fullscreen
25 | Table
26 |
27 |
28 | ⤢
29 | ↻
30 | ✖
31 |
32 |
33 |
47 |
85 |
86 |
--------------------------------------------------------------------------------
/examples/htdocs/todo.css:
--------------------------------------------------------------------------------
1 | /* See https://github.com/tastejs/todomvc/tree/gh-pages/examples/vue */
2 |
3 | html,
4 | body {
5 | margin: 0;
6 | padding: 0;
7 | }
8 |
9 | button {
10 | margin: 0;
11 | padding: 0;
12 | border: 0;
13 | background: none;
14 | font-size: 100%;
15 | vertical-align: baseline;
16 | font-family: inherit;
17 | font-weight: inherit;
18 | color: inherit;
19 | -webkit-appearance: none;
20 | appearance: none;
21 | -webkit-font-smoothing: antialiased;
22 | -moz-osx-font-smoothing: grayscale;
23 | }
24 |
25 | body {
26 | font: 14px 'Helvetica Neue', Helvetica, Arial, sans-serif;
27 | line-height: 1.4em;
28 | background: #f5f5f5;
29 | color: #4d4d4d;
30 | min-width: 230px;
31 | max-width: 550px;
32 | margin: 0 auto;
33 | -webkit-font-smoothing: antialiased;
34 | -moz-osx-font-smoothing: grayscale;
35 | font-weight: 300;
36 | }
37 |
38 | :focus {
39 | outline: 0;
40 | }
41 |
42 | .hidden {
43 | display: none;
44 | }
45 |
46 | .todoapp {
47 | background: #fff;
48 | margin: 130px 0 40px 0;
49 | position: relative;
50 | box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.2),
51 | 0 25px 50px 0 rgba(0, 0, 0, 0.1);
52 | }
53 |
54 | .todoapp input::-webkit-input-placeholder {
55 | font-style: italic;
56 | font-weight: 300;
57 | color: #e6e6e6;
58 | }
59 |
60 | .todoapp input::-moz-placeholder {
61 | font-style: italic;
62 | font-weight: 300;
63 | color: #e6e6e6;
64 | }
65 |
66 | .todoapp input::input-placeholder {
67 | font-style: italic;
68 | font-weight: 300;
69 | color: #e6e6e6;
70 | }
71 |
72 | .todoapp h1 {
73 | position: absolute;
74 | top: -155px;
75 | width: 100%;
76 | font-size: 100px;
77 | font-weight: 100;
78 | text-align: center;
79 | color: rgba(175, 47, 47, 0.15);
80 | -webkit-text-rendering: optimizeLegibility;
81 | -moz-text-rendering: optimizeLegibility;
82 | text-rendering: optimizeLegibility;
83 | }
84 |
85 | .new-todo,
86 | .edit {
87 | position: relative;
88 | margin: 0;
89 | width: 100%;
90 | font-size: 24px;
91 | font-family: inherit;
92 | font-weight: inherit;
93 | line-height: 1.4em;
94 | border: 0;
95 | color: inherit;
96 | padding: 6px;
97 | border: 1px solid #999;
98 | box-shadow: inset 0 -1px 5px 0 rgba(0, 0, 0, 0.2);
99 | box-sizing: border-box;
100 | -webkit-font-smoothing: antialiased;
101 | -moz-osx-font-smoothing: grayscale;
102 | }
103 |
104 | .new-todo {
105 | padding: 16px 16px 16px 60px;
106 | border: none;
107 | background: rgba(0, 0, 0, 0.003);
108 | box-shadow: inset 0 -2px 1px rgba(0,0,0,0.03);
109 | }
110 |
111 | .main {
112 | position: relative;
113 | z-index: 2;
114 | border-top: 1px solid #e6e6e6;
115 | }
116 |
117 | .toggle-all {
118 | width: 1px;
119 | height: 1px;
120 | border: none; /* Mobile Safari */
121 | opacity: 0;
122 | position: absolute;
123 | right: 100%;
124 | bottom: 100%;
125 | }
126 |
127 | .toggle-all + label {
128 | width: 60px;
129 | height: 34px;
130 | font-size: 0;
131 | position: absolute;
132 | top: -52px;
133 | left: -13px;
134 | -webkit-transform: rotate(90deg);
135 | transform: rotate(90deg);
136 | }
137 |
138 | .toggle-all + label:before {
139 | content: '❯';
140 | font-size: 22px;
141 | color: #e6e6e6;
142 | padding: 10px 27px 10px 27px;
143 | }
144 |
145 | .toggle-all:checked + label:before {
146 | color: #737373;
147 | }
148 |
149 | .todo-list {
150 | margin: 0;
151 | padding: 0;
152 | list-style: none;
153 | }
154 |
155 | .todo-list li {
156 | position: relative;
157 | font-size: 24px;
158 | border-bottom: 1px solid #ededed;
159 | }
160 |
161 | .todo-list li:last-child {
162 | border-bottom: none;
163 | }
164 |
165 | .todo-list li.editing {
166 | border-bottom: none;
167 | padding: 0;
168 | }
169 |
170 | .todo-list li.editing .edit {
171 | display: block;
172 | width: calc(100% - 43px);
173 | padding: 12px 16px;
174 | margin: 0 0 0 43px;
175 | }
176 |
177 | .todo-list li.editing .view {
178 | display: none;
179 | }
180 |
181 | .todo-list li .toggle {
182 | text-align: center;
183 | width: 40px;
184 | /* auto, since non-WebKit browsers doesn't support input styling */
185 | height: auto;
186 | position: absolute;
187 | top: 0;
188 | bottom: 0;
189 | margin: auto 0;
190 | border: none; /* Mobile Safari */
191 | -webkit-appearance: none;
192 | appearance: none;
193 | }
194 |
195 | .todo-list li .toggle {
196 | opacity: 0;
197 | }
198 |
199 | .todo-list li .toggle + label {
200 | /*
201 | Firefox requires `#` to be escaped - https://bugzilla.mozilla.org/show_bug.cgi?id=922433
202 | IE and Edge requires *everything* to be escaped to render, so we do that instead of just the `#` - https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/7157459/
203 | */
204 | background-image: url('data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23ededed%22%20stroke-width%3D%223%22/%3E%3C/svg%3E');
205 | background-repeat: no-repeat;
206 | background-position: center left;
207 | }
208 |
209 | .todo-list li .toggle:checked + label {
210 | background-image: url('data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23bddad5%22%20stroke-width%3D%223%22/%3E%3Cpath%20fill%3D%22%235dc2af%22%20d%3D%22M72%2025L42%2071%2027%2056l-4%204%2020%2020%2034-52z%22/%3E%3C/svg%3E');
211 | }
212 |
213 | .todo-list li label {
214 | word-break: break-all;
215 | padding: 15px 15px 15px 60px;
216 | display: block;
217 | line-height: 1.2;
218 | transition: color 0.4s;
219 | }
220 |
221 | .todo-list li.completed label {
222 | color: #d9d9d9;
223 | text-decoration: line-through;
224 | }
225 |
226 | .todo-list li .destroy {
227 | display: none;
228 | position: absolute;
229 | top: 0;
230 | right: 10px;
231 | bottom: 0;
232 | width: 40px;
233 | height: 40px;
234 | margin: auto 0;
235 | font-size: 30px;
236 | color: #cc9a9a;
237 | margin-bottom: 11px;
238 | transition: color 0.2s ease-out;
239 | }
240 |
241 | .todo-list li .destroy:hover {
242 | color: #af5b5e;
243 | }
244 |
245 | .todo-list li .destroy:after {
246 | content: '×';
247 | }
248 |
249 | .todo-list li:hover .destroy {
250 | display: block;
251 | }
252 |
253 | .todo-list li .edit {
254 | display: none;
255 | }
256 |
257 | .todo-list li.editing:last-child {
258 | margin-bottom: -1px;
259 | }
260 |
261 | .footer {
262 | color: #777;
263 | padding: 10px 15px;
264 | height: 20px;
265 | text-align: center;
266 | border-top: 1px solid #e6e6e6;
267 | }
268 |
269 | .footer:before {
270 | content: '';
271 | position: absolute;
272 | right: 0;
273 | bottom: 0;
274 | left: 0;
275 | height: 50px;
276 | overflow: hidden;
277 | box-shadow: 0 1px 1px rgba(0, 0, 0, 0.2),
278 | 0 8px 0 -3px #f6f6f6,
279 | 0 9px 1px -3px rgba(0, 0, 0, 0.2),
280 | 0 16px 0 -6px #f6f6f6,
281 | 0 17px 2px -6px rgba(0, 0, 0, 0.2);
282 | }
283 |
284 | .todo-count {
285 | float: left;
286 | text-align: left;
287 | }
288 |
289 | .todo-count strong {
290 | font-weight: 300;
291 | }
292 |
293 | .filters {
294 | margin: 0;
295 | padding: 0;
296 | list-style: none;
297 | position: absolute;
298 | right: 0;
299 | left: 0;
300 | }
301 |
302 | .filters li {
303 | display: inline;
304 | }
305 |
306 | .filters li a {
307 | color: inherit;
308 | margin: 3px;
309 | padding: 3px 7px;
310 | text-decoration: none;
311 | border: 1px solid transparent;
312 | border-radius: 3px;
313 | }
314 |
315 | .filters li a:hover {
316 | border-color: rgba(175, 47, 47, 0.1);
317 | }
318 |
319 | .filters li a.selected {
320 | border-color: rgba(175, 47, 47, 0.2);
321 | }
322 |
323 | .clear-completed,
324 | html .clear-completed:active {
325 | float: right;
326 | position: relative;
327 | line-height: 20px;
328 | text-decoration: none;
329 | cursor: pointer;
330 | }
331 |
332 | .clear-completed:hover {
333 | text-decoration: underline;
334 | }
335 |
336 | .info {
337 | margin: 65px auto 0;
338 | color: #bfbfbf;
339 | font-size: 10px;
340 | text-shadow: 0 1px 0 rgba(255, 255, 255, 0.5);
341 | text-align: center;
342 | }
343 |
344 | .info p {
345 | line-height: 1;
346 | }
347 |
348 | .info a {
349 | color: inherit;
350 | text-decoration: none;
351 | font-weight: 400;
352 | }
353 |
354 | .info a:hover {
355 | text-decoration: underline;
356 | }
357 |
358 | /*
359 | Hack to remove background from Mobile Safari.
360 | Can't use it globally since it destroys checkboxes in Firefox
361 | */
362 | @media screen and (-webkit-min-device-pixel-ratio:0) {
363 | .toggle-all,
364 | .todo-list li .toggle {
365 | background: none;
366 | }
367 |
368 | .todo-list li .toggle {
369 | height: 40px;
370 | }
371 | }
372 |
373 | @media (max-width: 430px) {
374 | .footer {
375 | height: 50px;
376 | }
377 |
378 | .filters {
379 | bottom: 10px;
380 | }
381 | }
382 |
--------------------------------------------------------------------------------
/examples/htdocs/todo.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Vue.js - TodoMVC
6 |
7 |
8 |
9 |
10 |
11 |
56 |
61 |
62 |
63 |
64 |
--------------------------------------------------------------------------------
/examples/htdocs/todo.js:
--------------------------------------------------------------------------------
1 | // See https://github.com/tastejs/todomvc/tree/gh-pages/examples/vue
2 |
3 | // IE on file does not provide localStorage
4 | var windowLocalStorage = window.localStorage;
5 | if (!windowLocalStorage) {
6 | windowLocalStorage = {
7 | _data : {},
8 | setItem : function(id, val) { return this._data[id] = String(val); },
9 | getItem : function(id) { return this._data.hasOwnProperty(id) ? this._data[id] : undefined; },
10 | removeItem : function(id) { return delete this._data[id]; },
11 | clear : function() { return this._data = {}; }
12 | };
13 | }
14 |
15 | // Full spec-compliant TodoMVC with localStorage persistence
16 | // and hash-based routing in ~120 effective lines of JavaScript.
17 |
18 | // localStorage persistence
19 | var STORAGE_KEY = 'todos-vuejs-2.0'
20 | var todoStorage = {
21 | fetch: function () {
22 | var todos = JSON.parse(windowLocalStorage.getItem(STORAGE_KEY) || '[]')
23 | todos.forEach(function (todo, index) {
24 | todo.id = index
25 | })
26 | todoStorage.uid = todos.length
27 | return todos
28 | },
29 | save: function (todos) {
30 | windowLocalStorage.setItem(STORAGE_KEY, JSON.stringify(todos))
31 | }
32 | }
33 |
34 | // visibility filters
35 | var filters = {
36 | all: function (todos) {
37 | return todos
38 | },
39 | active: function (todos) {
40 | return todos.filter(function (todo) {
41 | return !todo.completed
42 | })
43 | },
44 | completed: function (todos) {
45 | return todos.filter(function (todo) {
46 | return todo.completed
47 | })
48 | }
49 | }
50 |
51 | // app Vue instance
52 | var app = new Vue({
53 | // app initial state
54 | data: {
55 | todos: todoStorage.fetch(),
56 | newTodo: '',
57 | editedTodo: null,
58 | visibility: 'all'
59 | },
60 |
61 | // watch todos change for localStorage persistence
62 | watch: {
63 | todos: {
64 | handler: function (todos) {
65 | todoStorage.save(todos)
66 | },
67 | deep: true
68 | }
69 | },
70 |
71 | // computed properties
72 | // http://vuejs.org/guide/computed.html
73 | computed: {
74 | filteredTodos: function () {
75 | return filters[this.visibility](this.todos)
76 | },
77 | remaining: function () {
78 | return filters.active(this.todos).length
79 | },
80 | allDone: {
81 | get: function () {
82 | return this.remaining === 0
83 | },
84 | set: function (value) {
85 | this.todos.forEach(function (todo) {
86 | todo.completed = value
87 | })
88 | }
89 | }
90 | },
91 |
92 | filters: {
93 | pluralize: function (n) {
94 | return n === 1 ? 'item' : 'items'
95 | }
96 | },
97 |
98 | // methods that implement data logic.
99 | // note there's no DOM manipulation here at all.
100 | methods: {
101 | addTodo: function () {
102 | var value = this.newTodo && this.newTodo.trim()
103 | if (!value) {
104 | return
105 | }
106 | this.todos.push({
107 | id: todoStorage.uid++,
108 | title: value,
109 | completed: false
110 | })
111 | this.newTodo = ''
112 | },
113 |
114 | removeTodo: function (todo) {
115 | this.todos.splice(this.todos.indexOf(todo), 1)
116 | },
117 |
118 | editTodo: function (todo) {
119 | this.beforeEditCache = todo.title
120 | this.editedTodo = todo
121 | },
122 |
123 | doneEdit: function (todo) {
124 | if (!this.editedTodo) {
125 | return
126 | }
127 | this.editedTodo = null
128 | todo.title = todo.title.trim()
129 | if (!todo.title) {
130 | this.removeTodo(todo)
131 | }
132 | },
133 |
134 | cancelEdit: function (todo) {
135 | this.editedTodo = null
136 | todo.title = this.beforeEditCache
137 | },
138 |
139 | removeCompleted: function () {
140 | this.todos = filters.active(this.todos)
141 | }
142 | },
143 |
144 | // a custom directive to wait for the DOM to be updated
145 | // before focusing on the input field.
146 | // http://vuejs.org/guide/custom-directive.html
147 | directives: {
148 | 'todo-focus': function (el, binding) {
149 | if (binding.value) {
150 | el.focus()
151 | }
152 | }
153 | }
154 | })
155 |
156 | // handle routing
157 | function onHashChange () {
158 | var visibility = window.location.hash.replace(/#\/?/, '')
159 | if (filters[visibility]) {
160 | app.visibility = visibility
161 | } else {
162 | window.location.hash = ''
163 | app.visibility = 'all'
164 | }
165 | }
166 |
167 | window.addEventListener('hashchange', onHashChange)
168 | onHashChange()
169 |
170 | // mount
171 | app.$mount('.todoapp')
172 |
--------------------------------------------------------------------------------
/examples/htdocs/winexp.css:
--------------------------------------------------------------------------------
1 | body {
2 | font: 12px Helvetica, Arial, sans-serif;
3 | line-height: 1.4em;
4 | background: #f5f5f5;
5 | color: #4d4d4d;
6 | }
7 | div.toolbar {
8 | text-align: right;
9 | }
10 | .strikeout {
11 | text-decoration: line-through;
12 | }
13 |
--------------------------------------------------------------------------------
/examples/htdocs/winexp.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Window Explorer
6 |
7 |
8 |
9 |
37 |
38 |
39 |
40 |
41 |
42 |
--------------------------------------------------------------------------------
/examples/htdocs/winexp.js:
--------------------------------------------------------------------------------
1 | var filters = {
2 | all: function (list) {
3 | return list;
4 | },
5 | visible: function (list) {
6 | return list.filter(function(item) {
7 | return item.visible;
8 | });
9 | },
10 | hidden: function (list) {
11 | return list.filter(function(item) {
12 | return !item.visible;
13 | });
14 | }
15 | }
16 |
17 | var main = new Vue({
18 | el: '#main',
19 | data: {
20 | list: [],
21 | search: '',
22 | seen: false,
23 | visibility: 'all'
24 | },
25 | methods: {
26 | refresh: function() {
27 | this.value = '0';
28 | }
29 | },
30 | computed: {
31 | filteredWindows: function () {
32 | var searchLowerCase = this.search.toLowerCase();
33 | return filters[this.visibility](this.list.filter(function(item) {
34 | return item.text.toLowerCase().indexOf(searchLowerCase) >= 0;
35 | }));
36 | }
37 | }
38 | });
39 |
40 | function addWindows(list, clean) {
41 | main.list = clean ? list : main.list.concat(list);
42 | main.list.sort(function(a, b) {
43 | return a.text === b.text ? 0 : (a.text > b.text ? 1 : -1);
44 | })
45 | }
46 |
47 | // sample data
48 | /*
49 | main.list = [
50 | {handle: 123, text: 'Title', visible: false, width: 100, height: 100,left: 0, top: 0},
51 | {handle: 123, text: 'Hidden title', visible: true, width: 100, height: 100,left: 0, top: 0},
52 | {handle: 456, text: 'long long long long long long long long long long long long long long long long title', visible: true, width: 100, height: 100,left: 0, top: 0}
53 | ];
54 | */
--------------------------------------------------------------------------------
/examples/htdocs/winexp.lua:
--------------------------------------------------------------------------------
1 | local winapiLib = require('winapi')
2 |
3 | local viewportLeft, viewportTop = -10240, -10240
4 | local viewportRight, viewportBottom = 10240, 10240
5 | local minimizedLeft, minimizedTop = -32000, -32000
6 | local minWidth, minHeight = 5, 5
7 |
8 | local currentWindow
9 |
10 | local WINDOW_TEXT = 'Window Explorer'
11 | local HIDDEN_WINDOW_TEXT = 'Hidden '..WINDOW_TEXT
12 |
13 | local winexp = {}
14 |
15 | local function isViewableWindow(w)
16 | local width, height = w:get_bounds()
17 | if width > minWidth and height > minHeight then
18 | local left, top = w:get_position()
19 | if left == minimizedLeft and top == minimizedTop then -- special posision when minimized
20 | return true
21 | end
22 | local right, bottom = left + width, top + height
23 | return left < viewportRight and top < viewportBottom and right > viewportLeft and bottom > viewportTop
24 | end
25 | return false
26 | end
27 |
28 | local function isTopLevelWindow(w)
29 | local pw = w:get_parent()
30 | return pw and pw:get_handle()
31 | end
32 |
33 | local function isUserWindow(w)
34 | return isTopLevelWindow(w) and isViewableWindow(w) and w:get_text() ~= nil
35 | end
36 |
37 | local function isVisibleWindow(w)
38 | return w:is_visible() and isViewableWindow(w)
39 | end
40 |
41 | local function getWindowInfos(w)
42 | if not w then
43 | return nil
44 | end
45 | local width, height = w:get_bounds()
46 | local left, top = w:get_position()
47 | local text = w:get_text()
48 | if text and #text > 128 then
49 | text = string.sub(text, 1, 125)..'...'
50 | end
51 | local p = w:get_process()
52 | return {
53 | handle = w:get_handle(),
54 | text = text,
55 | visible = w:is_visible(),
56 | width = width,
57 | height = height,
58 | left = left,
59 | top = top,
60 | pid = p and p:get_pid(),
61 | }
62 | end
63 |
64 | function winexp.printWindow(w)
65 | if w then
66 | local i = getWindowInfos(w)
67 | print(i.handle, string.format('0x%08x', i.pid), i.visible, string.format('%dx%d', i.left, i.top), string.format('%dx%d', i.width, i.height), i.text)
68 | end
69 | end
70 |
71 | function winexp.findWindow(m)
72 | return winapiLib.find_window_ex(m or isViewableWindow)
73 | end
74 |
75 | function winexp.listWindows(m, t)
76 | if not m then
77 | m = isViewableWindow
78 | end
79 | if not t then
80 | return winapiLib.find_all_windows(m)
81 | end
82 | local list = {}
83 | winapiLib.enum_windows(function(w)
84 | if m(w) then
85 | table.insert(list, t(w))
86 | end
87 | end)
88 | return list
89 | end
90 |
91 | function winexp.listWindowsInfos()
92 | return winexp.listWindows(isUserWindow, getWindowInfos)
93 | end
94 |
95 | local SW_HIDE = 0 -- Hides the window and activates another window
96 | local SW_SHOW = 5 -- Activates the window and displays it in its current size and position
97 | local SW_SHOWNA = 8 -- Displays the window in its current size and position
98 |
99 | function winexp.getWindowByHandle(h)
100 | return winapiLib.find_window_ex(function(w)
101 | return w:get_handle() == h
102 | end)
103 | end
104 |
105 | function winexp.toggleWindowByHandle(h)
106 | local w = winexp.getWindowByHandle(h)
107 | if w then
108 | w:show(w:is_visible() and SW_HIDE or SW_SHOWNA)
109 | end
110 | end
111 |
112 | function winexp.foregroundWindowByHandle(h)
113 | local w = winexp.getWindowByHandle(h)
114 | if w then
115 | w:set_foreground()
116 | end
117 | end
118 |
119 | function winexp.lookForCurrentWindow()
120 | if not currentWindow then
121 | -- we could not see our own window so we check the foreground one with our pid
122 | local w = winapiLib.get_foreground_window()
123 | if w then
124 | local pid = winapiLib.get_current_pid()
125 | local p = w:get_process()
126 | if p and p:get_pid() == pid then
127 | currentWindow = w
128 | end
129 | end
130 | end
131 | return currentWindow
132 | end
133 |
134 | function winexp.hideCurrentSession()
135 | local cw = winexp.lookForCurrentWindow()
136 | if cw then
137 | cw:show(SW_HIDE)
138 | cw:set_text(HIDDEN_WINDOW_TEXT)
139 | end
140 | end
141 |
142 | function winexp.restoreHiddenSession()
143 | local pw = winapiLib.find_window_ex(function(w)
144 | return not w:is_visible() and w:get_text() == HIDDEN_WINDOW_TEXT
145 | end)
146 | if not pw then
147 | return false
148 | end
149 | local cw = winexp.lookForCurrentWindow()
150 | if cw then
151 | --local width, height = cw:get_bounds()
152 | --local left, top = cw:get_position()
153 | --pw:resize(left, top, width, height)
154 | pw:set_text(WINDOW_TEXT)
155 | pw:show(SW_SHOW)
156 | cw:show(SW_HIDE)
157 | print('Restoring hidden session')
158 | return true
159 | end
160 | return false
161 | end
162 |
163 | winapiLib.set_encoding(winapiLib.CP_UTF8)
164 |
165 | local pmw = winapiLib.find_window('Progman', 'Program Manager')
166 | if pmw then
167 | local width, height = pmw:get_bounds()
168 | viewportLeft, viewportTop = pmw:get_position()
169 | viewportRight = viewportLeft + width
170 | viewportBottom = viewportTop + height
171 | end
172 |
173 | local function listWindows()
174 | callJs('addWindows', winexp.listWindowsInfos(), true)
175 | end
176 |
177 | winexp.lookForCurrentWindow()
178 | if winexp.restoreHiddenSession() then
179 | require('webview').terminate(webview, true)
180 | end
181 |
182 | expose('hideCurrentSession', winexp.hideCurrentSession)
183 | expose('toggleWindow', winexp.toggleWindowByHandle, true)
184 | expose('foregroundWindow', winexp.foregroundWindowByHandle, true)
185 | expose('listWindows', listWindows)
186 |
187 | listWindows()
188 |
--------------------------------------------------------------------------------
/examples/launch.lua:
--------------------------------------------------------------------------------
1 | require('webview-launcher').launchFromArgs()
--------------------------------------------------------------------------------
/examples/open.lua:
--------------------------------------------------------------------------------
1 | -- Default web content
2 | local url = [[data:text/html,
3 |
4 |
5 | Welcome !
6 | You could specify an URL to open as a command line argument.
7 |
8 |
9 | ]]
10 |
11 | -- Parse command line arguments
12 | local urlArg = arg[1]
13 | if urlArg and urlArg ~= '' then
14 | if urlArg == '-h' or urlArg == '/?' or urlArg == '--help' then
15 | print('Opens a WebView using the specified URL')
16 | print('Optional arguments: url title width height resizable')
17 | os.exit(0)
18 | end
19 | local protocol = string.match(urlArg, '^([^:]+):.+$')
20 | if protocol == 'http' or protocol == 'https' or protocol == 'file' or protocol == 'data' then
21 | url = urlArg
22 | elseif string.match(urlArg, '^.:\\.+$') or string.match(urlArg, '^/.+$') then
23 | url = 'file://'..tostring(urlArg)
24 | else
25 | print('Invalid URL, to open a file please use an absolute path')
26 | os.exit(22)
27 | end
28 | end
29 | local title = arg[2] or 'Web View'
30 | local width = arg[3] or 800
31 | local height = arg[4] or 600
32 | local resizable = arg[5] ~= 'false'
33 |
34 | -- Opens the web view
35 | require('webview').open(url, title, width, height, resizable)
36 |
--------------------------------------------------------------------------------
/examples/simple.lua:
--------------------------------------------------------------------------------
1 | local webviewLib = require('webview')
2 |
3 | local content = [[
4 |
5 |
6 | It works !
7 | Change Title
8 | Print Date
9 | Show Date
10 | User Agent
11 |
12 | ↻
13 | ⤢
14 | ✖
15 |
16 |
17 |
23 |
24 | ]]
25 |
26 | content = string.gsub(content, "[ %c!#$%%&'()*+,/:;=?@%[%]]", function(c)
27 | return string.format('%%%02X', string.byte(c))
28 | end)
29 |
30 | local webview = webviewLib.new('data:text/html,'..content, 'Example', 480, 240, true, true)
31 |
32 | webviewLib.callback(webview, function(value)
33 | if value == 'print_date' then
34 | print(os.date())
35 | elseif value == 'show_date' then
36 | webviewLib.eval(webview, 'showText("Lua date is '..os.date()..'")', true)
37 | elseif value == 'fullscreen' then
38 | webviewLib.fullscreen(webview, true)
39 | elseif value == 'exit_fullscreen' then
40 | webviewLib.fullscreen(webview, false)
41 | elseif value == 'terminate' then
42 | webviewLib.terminate(webview, true)
43 | elseif string.find(value, '^title=') then
44 | webviewLib.title(webview, string.sub(value, 7))
45 | else
46 | print('callback received', value)
47 | end
48 | end)
49 |
50 | webviewLib.loop(webview)
51 |
--------------------------------------------------------------------------------
/rock.mk:
--------------------------------------------------------------------------------
1 | #
2 | # Makefile for rockspec
3 | #
4 | # Install with Lua Binaries:
5 | # luarocks --lua-dir C:/bin/lua-5.3.5_Win64_bin MAKE=make CC=gcc LD=gcc install lua-webview
6 | #
7 | # Build with luaclibs:
8 | # luarocks --lua-dir ../../luaclibs/lua/src MAKE=make CC=gcc LD=gcc make
9 | # luarocks --lua-dir C:/bin/lua-5.4.2_Win64_bin MAKE=make CC=gcc LD=gcc make lua-webview-1.3-2.rockspec
10 | #
11 |
12 | CC ?= gcc
13 |
14 | PLAT ?= windows
15 | LIBNAME = webview
16 |
17 | ifdef LUA_LIBDIR
18 | LUA_LIBDIR_OPT=-L$(LUA_LIBDIR)
19 | else
20 | LUA_LIBDIR_OPT=
21 | endif
22 |
23 | #LUA_APP = $(LUA_BINDIR)/$(LUA)
24 | LUA_APP = $(LUA)
25 | LUA_VERSION = $(shell $(LUA_APP) -e "print(string.sub(_VERSION, 5))")
26 | LUA_LIBNAME = lua$(subst .,,$(LUA_VERSION))
27 | LUA_BITS = $(shell $(LUA_APP) -e "print(string.len(string.pack('T', 0)) * 8)")
28 |
29 | WEBVIEW_ARCH = x64
30 | ifeq ($(LUA_BITS),32)
31 | WEBVIEW_ARCH = x86
32 | endif
33 |
34 | WEBVIEW_C = webview-c
35 | MS_WEBVIEW2 = $(WEBVIEW_C)/ms.webview2
36 |
37 | CFLAGS_windows = -Wall \
38 | -Wextra \
39 | -Wno-unused-parameter \
40 | -Wstrict-prototypes \
41 | -I$(WEBVIEW_C) \
42 | -I$(MS_WEBVIEW2)/include \
43 | -I$(LUA_INCDIR) \
44 | -DWEBVIEW_WINAPI=1
45 |
46 | LIBFLAG_windows = -O \
47 | -shared \
48 | -Wl,-s \
49 | $(LUA_LIBDIR_OPT) -l$(LUA_LIBNAME) \
50 | -static-libgcc \
51 | -lole32 -lcomctl32 -loleaut32 -luuid -lgdi32
52 |
53 | TARGET_windows = $(LIBNAME).dll
54 |
55 | CFLAGS_linux = -pedantic \
56 | -Wall \
57 | -Wextra \
58 | -Wno-unused-parameter \
59 | -Wstrict-prototypes \
60 | -I$(WEBVIEW_C) \
61 | -I$(LUA_INCDIR) \
62 | -DWEBVIEW_GTK=1 \
63 | $(shell pkg-config --cflags gtk+-3.0 webkit2gtk-4.0)
64 |
65 | LIBFLAG_linux= -static-libgcc \
66 | -Wl,-s \
67 | $(LUA_LIBDIR_OPT) \
68 | $(shell pkg-config --libs gtk+-3.0 webkit2gtk-4.0)
69 |
70 | TARGET_linux = $(LIBNAME).so
71 |
72 |
73 | TARGET = $(TARGET_$(PLAT))
74 |
75 | SOURCES = webview.c
76 |
77 | OBJS = webview.o
78 |
79 | lib: $(TARGET)
80 |
81 | install: install-$(PLAT)
82 | cp $(TARGET) $(INST_LIBDIR)
83 | -cp webview-launcher.lua $(INST_LUADIR)
84 |
85 | install-linux:
86 |
87 | install-windows:
88 | cp $(MS_WEBVIEW2)/$(WEBVIEW_ARCH)/WebView2Loader.dll $(INST_BINDIR)
89 |
90 | show:
91 | @echo PLAT: $(PLAT)
92 | @echo LUA_VERSION: $(LUA_VERSION)
93 | @echo LUA_LIBNAME: $(LUA_LIBNAME)
94 | @echo CFLAGS: $(CFLAGS)
95 | @echo LIBFLAG: $(LIBFLAG)
96 | @echo LUA_LIBDIR: $(LUA_LIBDIR)
97 | @echo LUA_BINDIR: $(LUA_BINDIR)
98 | @echo LUA_INCDIR: $(LUA_INCDIR)
99 | @echo LUA: $(LUA)
100 | @echo LUALIB: $(LUALIB)
101 |
102 | show-install:
103 | @echo PREFIX: $(PREFIX) or $(INST_PREFIX)
104 | @echo BINDIR: $(BINDIR) or $(INST_BINDIR)
105 | @echo LIBDIR: $(LIBDIR) or $(INST_LIBDIR)
106 | @echo LUADIR: $(LUADIR) or $(INST_LUADIR)
107 |
108 | $(TARGET): $(OBJS)
109 | $(CC) $(OBJS) $(LIBFLAG) $(LIBFLAG_$(PLAT)) -o $(TARGET)
110 |
111 | clean:
112 | -$(RM) $(OBJS) $(TARGET)
113 |
114 | $(OBJS): %.o : %.c $(SOURCES)
115 | $(CC) $(CFLAGS) $(CFLAGS_$(PLAT)) -c -o $@ $<
116 |
--------------------------------------------------------------------------------
/test.lua:
--------------------------------------------------------------------------------
1 |
2 | local webviewLauncher = require('webview-launcher')
3 |
4 | print('-- JSON --------')
5 | local values = {
6 | 'ti/ti\nta\9ta\tto\20to "tutu" ty\\ty',
7 | '', 'Hi', true, false, 123, -123, 1.23,
8 | }
9 | for _, value in ipairs(values) do
10 | local encoded = webviewLauncher.jsonLib.encode(value)
11 | local decoded = webviewLauncher.jsonLib.decode(encoded)
12 | if value == decoded then
13 | print(encoded, type(value), 'Ok')
14 | else
15 | print('>>'..tostring(value)..'<<'..type(value))
16 | print('>>'..tostring(encoded)..'<<'..type(encoded))
17 | print('>>'..tostring(decoded)..'<<'..type(decoded))
18 | end
19 | end
20 | print('-- FS --------')
21 | print('currentdir:', webviewLauncher.fsLib.currentdir())
22 | local paths = {'webview-launcher.lua', 'not a file'}
23 | for _, path in ipairs(paths) do
24 | print(path, webviewLauncher.fsLib.attributes(path) and 'exists' or 'not found')
25 | end
26 |
--------------------------------------------------------------------------------
/webview-init.js:
--------------------------------------------------------------------------------
1 | /*
2 | This script triggers the webview launcher initialisation sequence.
3 | By default the initialisation is done on first load, this script provides support for navigation and page reload.
4 | Lua code in HTML must support multiple executions as well as onWebviewInitalized handler.
5 | */
6 | (function() {
7 | var timeoutDelay = 1;
8 |
9 | function handleLoad() {
10 | if ((typeof window.external !== 'object') || (typeof window.external.invoke !== 'function')) {
11 | if (timeoutDelay > 30000) {
12 | throw 'window.external is not available';
13 | }
14 | setTimeout(handleLoad, timeoutDelay); // Let external.invoke be registered
15 | timeoutDelay = timeoutDelay * 2;
16 | } else if (typeof window.webview !== 'object') {
17 | window.external.invoke(':init:');
18 | }
19 | }
20 |
21 | if (document.readyState === 'complete') {
22 | handleLoad();
23 | } else {
24 | window.addEventListener('load', handleLoad);
25 | }
26 | })();
27 |
--------------------------------------------------------------------------------
/webview-launcher.lua:
--------------------------------------------------------------------------------
1 | local webviewLib = require('webview')
2 |
3 | -- This module allows to launch a web page that could executes custom Lua code.
4 |
5 | -- The webview library may change locale to native and thus mislead the JSON libraries.
6 | if os.setlocale() == 'C' then
7 | -- Default locale is 'C' at startup, set native locale
8 | os.setlocale('')
9 | end
10 |
11 | -- Load JSON module
12 | local status, jsonLib = pcall(require, 'cjson')
13 | if not status then
14 | status, jsonLib = pcall(require, 'dkjson')
15 | if not status then
16 | -- provide a basic JSON implementation suitable for basic types
17 | local escapeMap = { ['\b'] = '\\b', ['\f'] = '\\f', ['\n'] = '\\n', ['\r'] = '\\r', ['\t'] = '\\t', ['"'] = '\\"', ['\\'] = '\\\\', ['/'] = '\\/', }
18 | local revertMap = {}; for c, s in pairs(escapeMap) do revertMap[s] = c; end
19 | jsonLib = {
20 | null = {},
21 | decode = function(value)
22 | if string.sub(value, 1, 1) == '"' and string.sub(value, -1, -1) == '"' then
23 | return string.gsub(string.gsub(string.sub(value, 2, -2), '\\u(%x%x%x%x)', function(s)
24 | return string.char(tonumber(s, 16))
25 | end), '\\.', function(s)
26 | return revertMap[s] or ''
27 | end)
28 | elseif string.match(value, '^%s*[%-%+]?%d[%d%.%s]*$') then
29 | return tonumber(value)
30 | elseif (value == 'true') or (value == 'false') then
31 | return value == 'true'
32 | elseif value == 'null' then
33 | return jsonLib.null
34 | end
35 | return nil
36 | end,
37 | encode = function(value)
38 | local valueType = type(value)
39 | if valueType == 'boolean' then
40 | return value and 'true' or 'false'
41 | elseif valueType == 'number' then
42 | return (string.gsub(tostring(value), ',', '.', 1))
43 | elseif valueType == 'string' then
44 | return '"'..string.gsub(value, '[%c"/\\]', function(c)
45 | return escapeMap[c] or string.format('\\u%04X', string.byte(c))
46 | end)..'"'
47 | elseif value == jsonLib.null then
48 | return 'null'
49 | end
50 | return 'undefined'
51 | end
52 | }
53 | end
54 | end
55 |
56 | -- OS file separator
57 | local fileSeparator = string.sub(package.config, 1, 1) or '/'
58 |
59 | -- Load file system module
60 | local fsLib
61 | status, fsLib = pcall(require, 'luv')
62 | if status then
63 | local uvLib = fsLib
64 | fsLib = {
65 | currentdir = uvLib.cwd,
66 | attributes = uvLib.fs_stat,
67 | }
68 | else
69 | status, fsLib = pcall(require, 'lfs')
70 | if not status then
71 | -- provide a basic file system implementation
72 | fsLib = {
73 | currentdir = function()
74 | local f = io.popen(fileSeparator == '\\' and 'cd' or 'pwd')
75 | if f then
76 | local d = f:read()
77 | f:close()
78 | return d
79 | end
80 | return '.'
81 | end,
82 | attributes = function(p)
83 | local f = io.open(p)
84 | if f then
85 | f:close()
86 | return {}
87 | end
88 | return nil
89 | end,
90 | }
91 | end
92 | end
93 |
94 | -- Lua code injected to provide default local variables
95 | local localContextLua = 'local evalJs, callJs, expose = context.evalJs, context.callJs, context.expose; '
96 |
97 | local function exposeFunctionJs(name, remove)
98 | local nameJs = "'"..name.."'"
99 | if remove then
100 | return 'delete webview['..nameJs..'];\n';
101 | end
102 | return 'webview['..nameJs..'] = function(value, callback) {'..
103 | 'webview.invokeLua('..nameJs..', value, callback);'..
104 | '};\n'
105 | end
106 |
107 | -- Initializes the web view and provides a global JavaScript webview object
108 | local function initializeJs(webview, functionMap, options)
109 | local jsContent = [[
110 | if (typeof window.webview === 'object') {
111 | console.log('webview object already exists');
112 | } else {
113 | console.log('initialize webview object');
114 | var webview = {};
115 | window.webview = webview;
116 | var refs = {};
117 | var callbackToRef = function(callback, delay) {
118 | if (typeof callback === 'function') {
119 | var ref;
120 | var id = setTimeout(function() {
121 | var cb = refs[ref];
122 | if (cb) {
123 | delete refs[ref];
124 | cb('timeout');
125 | }
126 | }, delay);
127 | ref = id.toString(36);
128 | refs[ref] = callback;
129 | return ref;
130 | }
131 | return null;
132 | };
133 | webview.callbackRef = function(ref, reason, result) {
134 | var id = parseInt(ref, 36);
135 | clearTimeout(id);
136 | var callback = refs[ref];
137 | if (callback) {
138 | delete refs[ref];
139 | callback(reason, result);
140 | }
141 | };
142 | webview.invokeLua = function(name, value, callback, delay) {
143 | var kind = ':', data = '';
144 | if (typeof value === 'string') {
145 | data = value;
146 | } else if (typeof value === 'function') {
147 | delay = callback;
148 | callback = value;
149 | } else if (value !== undefined) {
150 | kind = ';';
151 | data = JSON.stringify(value);
152 | }
153 | var message;
154 | var ref = callbackToRef(callback, delay || 30000);
155 | if (ref) {
156 | message = '#' + name + kind + ref + ';' + data;
157 | } else {
158 | message = name + kind + data;
159 | }
160 | window.external.invoke(message);
161 | };
162 | ]]
163 | if options and options.captureError then
164 | jsContent = jsContent..[[
165 | window.onerror = function(message, source, lineno, colno, error) {
166 | var message = '' + message; // Just "Script error." when occurs in different origin
167 | if (source) {
168 | message += '\n source: ' + source + ', line: ' + lineno + ', col: ' + colno;
169 | }
170 | if (error) {
171 | message += '\n error: ' + error;
172 | }
173 | window.external.invoke(':error:' + message);
174 | return true;
175 | };
176 | ]]
177 | end
178 | if options and options.useJsTitle then
179 | jsContent = jsContent..[[
180 | if (document.title) {
181 | window.external.invoke('title:' + document.title);
182 | }
183 | ]]
184 | end
185 | if functionMap then
186 | for name in pairs(functionMap) do
187 | jsContent = jsContent..exposeFunctionJs(name)
188 | end
189 | end
190 | if options and options.luaScript then
191 | jsContent = jsContent..[[
192 | var evalLuaScripts = function() {
193 | var scripts = document.getElementsByTagName('script');
194 | for (var i = 0; i < scripts.length; i++) {
195 | var script = scripts[i];
196 | if (script.getAttribute('type') === 'text/lua') {
197 | var src = script.getAttribute('src');
198 | if (src) {
199 | window.external.invoke('evalLuaSrc:' + src);
200 | } else {
201 | window.external.invoke('evalLua:' + script.text);
202 | }
203 | }
204 | }
205 | };
206 | if (document.readyState !== 'loading') {
207 | evalLuaScripts();
208 | } else {
209 | document.addEventListener('DOMContentLoaded', evalLuaScripts);
210 | }
211 | ]]
212 | end
213 | jsContent = jsContent..[[
214 | var completeInitialization = function() {
215 | if (typeof window.onWebviewInitalized === 'function') {
216 | webview.evalJs("window.onWebviewInitalized(window.webview);");
217 | }
218 | };
219 | if (document.readyState === 'complete') {
220 | completeInitialization();
221 | } else {
222 | window.addEventListener('load', completeInitialization);
223 | }
224 | }
225 | ]]
226 | webviewLib.eval(webview, jsContent, true)
227 | end
228 |
229 | -- Prints error message to the error stream
230 | local function printError(value)
231 | io.stderr:write('WebView Launcher - '..tostring(value)..'\n')
232 | end
233 |
234 | local function callbackJs(webview, ref, reason, result)
235 | webviewLib.eval(webview, 'if (webview) {'..
236 | 'webview.callbackRef("'..ref..'", '..jsonLib.encode(reason)..', '..jsonLib.encode(result)..');'..
237 | '}', true)
238 | end
239 |
240 | local function handleCallback(callback, reason, result)
241 | if callback then
242 | callback(reason, result)
243 | elseif reason then
244 | printError(reason)
245 | end
246 | end
247 |
248 | -- Executes the specified Lua code
249 | local function evalLua(value, callback, context, webview)
250 | local f, err = load('local callback, context, webview = ...; '..localContextLua..value)
251 | if f then
252 | f(callback, context, webview)
253 | else
254 | handleCallback(callback, 'Error '..tostring(err)..' while loading '..tostring(value))
255 | end
256 | end
257 |
258 | -- Toggles the web view full screen on/off
259 | local function fullscreen(value, callback, _, webview)
260 | webviewLib.fullscreen(webview, value == 'true')
261 | handleCallback(callback)
262 | end
263 |
264 | -- Sets the web view title
265 | local function setTitle(value, callback, _, webview)
266 | webviewLib.title(webview, value)
267 | handleCallback(callback)
268 | end
269 |
270 | -- Terminates the web view
271 | local function terminate(_, callback, _, webview)
272 | webviewLib.terminate(webview, true)
273 | handleCallback(callback)
274 | end
275 |
276 | -- Executes the specified Lua file relative to the URL
277 | local function evalLuaSrc(value, callback, context, webview)
278 | local content
279 | if context.luaSrcPath then
280 | local path = context.luaSrcPath..fileSeparator..string.gsub(value, '[/\\]+', fileSeparator)
281 | local file = io.open(path)
282 | if file then
283 | content = file:read('a')
284 | file:close()
285 | end
286 | end
287 | if content then
288 | evalLua(content, callback, context, webview)
289 | else
290 | handleCallback(callback, 'Cannot load Lua file from src "'..tostring(value)..'"')
291 | end
292 | end
293 |
294 | -- Evaluates the specified JS code
295 | local function evalJs(value, callback, _, webview)
296 | webviewLib.eval(webview, value, true)
297 | handleCallback(callback)
298 | end
299 |
300 | -- Calls the specified JS function name,
301 | -- the arguments are JSON encoded then passed to the JS function
302 | local function callJs(webview, functionName, ...)
303 | local argCount = select('#', ...)
304 | local args = {...}
305 | for i = 1, argCount do
306 | args[i] = jsonLib.encode(args[i])
307 | end
308 | local jsString = functionName..'('..table.concat(args, ',')..')'
309 | webviewLib.eval(webview, jsString, true)
310 | end
311 |
312 | -- Creates the webview context and sets the callback and default functions
313 | local function createContext(webview, options)
314 | local initialized = false
315 |
316 | -- Named requests callable from JS using window.external.invoke('name:value')
317 | -- Custom request can be registered using window.external.invoke('+name:Lua code')
318 | -- The Lua code has access to the string value, the evalJs() and callJs() functions
319 | local functionMap = {
320 | fullscreen = fullscreen,
321 | title = setTitle,
322 | terminate = terminate,
323 | evalLua = evalLua,
324 | evalLuaSrc = evalLuaSrc,
325 | evalJs = evalJs,
326 | }
327 |
328 | -- Defines the context that will be shared across Lua calls
329 | local context = {
330 | expose = function(name, fn)
331 | functionMap[name] = fn
332 | if initialized then
333 | webviewLib.eval(webview, exposeFunctionJs(name, not fn), true)
334 | end
335 | end,
336 | exposeAll = function(fnMap)
337 | local jsContent = ''
338 | for name, fn in pairs(fnMap) do
339 | functionMap[name] = fn
340 | jsContent = jsContent..exposeFunctionJs(name, not fn)
341 | end
342 | if initialized then
343 | webviewLib.eval(webview, jsContent, true)
344 | end
345 |
346 | end,
347 | -- Setup a Lua function to evaluates JS code
348 | evalJs = function(value)
349 | webviewLib.eval(webview, value, true)
350 | end,
351 | callJs = function(functionName, ...)
352 | callJs(webview, functionName, ...)
353 | end,
354 | callbackJs = function(ref, reason, result)
355 | callbackJs(webview, ref, reason, result)
356 | end,
357 | terminate = function()
358 | webviewLib.terminate(webview, true)
359 | end,
360 | }
361 |
362 | if options and type(options.expose) == 'table' then
363 | context.exposeAll(options.expose)
364 | end
365 |
366 | if options and type(options.context) == 'table' then
367 | for name, value in pairs(options.context) do
368 | context[name] = value
369 | end
370 | end
371 |
372 | -- Creates the web view callback that handles the JS requests coming from window.external.invoke()
373 | local handler = function(request)
374 | local flag, name, kind, value = string.match(request, '^(%A?)(%a%w*)([:;])(.*)$')
375 | if name then
376 | if flag == '' or flag == '#' then
377 | -- Look for the specified function
378 | local fn = functionMap[name]
379 | local callback, cbRef
380 | if fn then
381 | if flag == '#' then
382 | local ref, val = string.match(value, '^(%w+);(.*)$')
383 | if ref and val then
384 | value = val
385 | cbRef = ref
386 | callback = function(reason, result)
387 | callbackJs(webview, ref, reason, result)
388 | end
389 | else
390 | printError('Invalid reference request '..request)
391 | return
392 | end
393 | end
394 | local s, r
395 | if kind == ';' then
396 | s, r = pcall(jsonLib.decode, value)
397 | if s then
398 | value = r
399 | else
400 | handleCallback(callback, 'Fail to parse '..name..' JSON value "'..tostring(value)..'" due to '..tostring(r))
401 | return
402 | end
403 | end
404 | s, r = pcall(fn, value, callback, context, webview, cbRef)
405 | if not s then
406 | handleCallback(callback, 'Fail to execute '..name..' due to '..tostring(r))
407 | end
408 | else
409 | printError('Unknown function '..name)
410 | end
411 | elseif flag == '-' then
412 | context.expose(name)
413 | elseif flag == '+' then
414 | -- Registering the new function using the specified Lua code
415 | local injected = 'local value, callback, context, webview = ...; '
416 | local fn, err = load(injected..localContextLua..value)
417 | if fn then
418 | context.expose(name, fn)
419 | else
420 | printError('Error '..tostring(err)..' while loading '..tostring(value))
421 | end
422 | elseif name == 'error' and flag == ':' then
423 | printError(value)
424 | elseif name == 'init' and flag == ':' then
425 | initialized = true
426 | initializeJs(webview, functionMap, options)
427 | else
428 | printError('Invalid flag '..flag..' for name '..name)
429 | end
430 | else
431 | printError('Invalid request #'..tostring(request and #request)..' "'..tostring(request)..'"')
432 | end
433 | end
434 |
435 | if options and options.initialize then
436 | initialized = true
437 | initializeJs(webview, functionMap, options)
438 | end
439 |
440 | return handler
441 | end
442 |
443 | local function escapeUrl(value)
444 | return string.gsub(value, "[ %c!#$%%&'()*+,/:;=?@%[%]]", function(c)
445 | return string.format('%%%02X', string.byte(c))
446 | end)
447 | end
448 |
449 | local function parseArgs(args)
450 | -- Default web content
451 | local url = 'data:text/html,'..escapeUrl([[
452 |
453 |
454 | Welcome WebView
455 |
456 |
459 |
460 | Welcome !
461 | You could specify an HTML file to launch as a command line argument.
462 | Close
463 |
464 |
465 | ]])
466 |
467 | local title
468 | local width = 800
469 | local height = 600
470 | local resizable = true
471 | local debug = false
472 | local eventMode = nil
473 | local initialize = true
474 | local luaScript = true
475 | local captureError = true
476 | local luaPath = false
477 |
478 | -- Parse command line arguments
479 | args = args or arg or {}
480 | local ctxArgs = {}
481 | local luaSrcPath = nil
482 | local urlArg
483 |
484 | for i = 1, #args do
485 | local name, value = string.match(args[i], '^%-%-wv%-([^=]+)=?(.*)$')
486 | if not name then
487 | if urlArg then
488 | table.insert(ctxArgs, args[i])
489 | else
490 | urlArg = args[i]
491 | end
492 | elseif name == 'size' and value then
493 | local w, h = string.match(value, '^(%d+)[xX-/](%d+)$')
494 | width = tonumber(w)
495 | height = tonumber(h)
496 | elseif name == 'title' and value then
497 | title = value
498 | elseif name == 'width' and tonumber(value) then
499 | width = tonumber(value)
500 | elseif name == 'height' and tonumber(value) then
501 | height = tonumber(value)
502 | elseif name == 'resizable' then
503 | resizable = value ~= 'false'
504 | elseif name == 'debug' then
505 | debug = value == 'true'
506 | elseif name == 'event' and (value == 'open' or value == 'main' or value == 'thread' or value == 'http') then
507 | eventMode = value
508 | elseif name == 'initialize' then
509 | initialize = value ~= 'false'
510 | elseif name == 'script' then
511 | luaScript = value ~= 'false'
512 | elseif name == 'captureError' then
513 | captureError = value ~= 'false'
514 | elseif name == 'luaPath' then
515 | luaPath = value == 'true'
516 | else
517 | print('Invalid argument', args[i])
518 | os.exit(22)
519 | end
520 | end
521 |
522 | -- Process URL argument
523 | if urlArg and urlArg ~= '' then
524 | if urlArg == '-h' or urlArg == '/?' or urlArg == '--help' then
525 | print('Launchs a WebView using the specified URL')
526 | print('Optional arguments: url --wv-title= --wv-width='..tostring(width)..' --wv-height='..tostring(height)..' --wv-resizable='..tostring(resizable))
527 | os.exit(0)
528 | end
529 | local protocol = string.match(urlArg, '^([^:]+):.+$')
530 | if protocol == 'http' or protocol == 'https' or protocol == 'file' or protocol == 'data' then
531 | url = urlArg
532 | else
533 | local filePath
534 | if string.match(urlArg, '^.:\\.+$') or string.match(urlArg, '^/.+$') then
535 | filePath = urlArg
536 | elseif fsLib then
537 | filePath = fsLib.currentdir()..fileSeparator..urlArg
538 | end
539 | if not filePath then
540 | print('Invalid URL, to launch a file please use an absolute path')
541 | os.exit(22)
542 | end
543 | luaSrcPath = string.match(filePath, '^(.*)[/\\][^/\\]+$')
544 | url = 'file://'..filePath
545 | end
546 | end
547 |
548 | if luaSrcPath and luaPath then
549 | package.path = package.path..';'..luaSrcPath..'/?.lua'
550 | end
551 |
552 | return url, {
553 | title = title or 'Web View',
554 | width = width,
555 | height = height,
556 | resizable = resizable,
557 | debug = debug
558 | }, {
559 | eventMode = eventMode,
560 | initialize = initialize,
561 | useJsTitle = not title,
562 | luaScript = luaScript,
563 | luaPath = luaPath,
564 | captureError = captureError,
565 | context = {
566 | args = ctxArgs,
567 | luaSrcPath = luaSrcPath,
568 | },
569 | }
570 | end
571 |
572 | local function createContextAndPath(wv, opts)
573 | if opts.luaPath and opts.context and opts.context.luaSrcPath then
574 | package.path = package.path..';'..opts.context.luaSrcPath..'/?.lua'
575 | end
576 | return createContext(wv, opts)
577 | end
578 |
579 | local function launchWithOptions(url, wvOptions, options)
580 | wvOptions = wvOptions or {}
581 | options = options or wvOptions
582 | if options.eventMode then
583 | local event = require('jls.lang.event')
584 | local WebView = require('jls.util.WebView')
585 | if options.eventMode == 'thread' then
586 | --webviewLib.open('data:text/html,Close to start', 'Close to start', 320, 200)
587 | WebView.open(url, wvOptions):next(function(webview)
588 | local callback = createContextAndPath(webview._webview, options)
589 | webview:callback(callback)
590 | end)
591 | elseif options.eventMode == 'http' then
592 | if not options.context.luaSrcPath then
593 | error('Please specify a file path as URL')
594 | end
595 | local FileHttpHandler = require('jls.net.http.handler.FileHttpHandler')
596 | options.callback = true
597 | options.contexts = {
598 | ['/(.*)'] = FileHttpHandler:new(options.context.luaSrcPath or '.')
599 | }
600 | local filename = string.match(url, '^.*[/\\]([^/\\]+)$')
601 | WebView.open('http://localhost:0/'..filename, options):next(function(webview)
602 | local callback = createContextAndPath(webview._webview, options)
603 | webview:callback(callback)
604 | end)
605 | else
606 | local open = options.eventMode == 'main' and WebView.openSync or WebView.open
607 | wvOptions.fn = function(webview, data)
608 | local webviewLauncher = require('webview-launcher')
609 | local opts = webviewLauncher.jsonLib.decode(data)
610 | local callback = webviewLauncher.createContextAndPath(webview._webview, opts)
611 | webview:callback(callback)
612 | end
613 | wvOptions.data = jsonLib.encode(options)
614 | open(url, wvOptions)
615 | end
616 | event:loop()
617 | else
618 | local webview = webviewLib.new(url, wvOptions.title, wvOptions.width, wvOptions.height, wvOptions.resizable, wvOptions.debug)
619 | local callback = createContext(webview, options)
620 | webviewLib.callback(webview, callback)
621 | webviewLib.loop(webview)
622 | end
623 | end
624 |
625 | return {
626 | initializeJs = initializeJs,
627 | createContext = createContext,
628 | createContextAndPath = createContextAndPath,
629 | escapeUrl = escapeUrl,
630 | launchFromArgs = function(args, ...)
631 | if type(args) == 'string' then
632 | args = {args, ...}
633 | elseif type(args) ~= 'table' then
634 | args = arg
635 | end
636 | launchWithOptions(parseArgs(args))
637 | end,
638 | launchWithOptions = launchWithOptions,
639 | jsonLib = jsonLib,
640 | fsLib = fsLib,
641 | }
642 |
--------------------------------------------------------------------------------
/webview.c:
--------------------------------------------------------------------------------
1 | #include
2 | #include
3 |
4 | #define WEBVIEW_IMPLEMENTATION
5 |
6 | // wget https://raw.githubusercontent.com/zserge/webview/master/webview.h
7 | #include "webview.h"
8 |
9 | /*
10 | ********************************************************************************
11 | * Lua 5.1 compatibility
12 | ********************************************************************************
13 | */
14 |
15 | #if LUA_VERSION_NUM < 502
16 | // From Lua 5.3 lauxlib.c
17 | LUALIB_API void luaL_setfuncs (lua_State *L, const luaL_Reg *l, int nup) {
18 | luaL_checkstack(L, nup, "too many upvalues");
19 | for (; l->name != NULL; l++) { /* fill the table with given functions */
20 | int i;
21 | for (i = 0; i < nup; i++) /* copy upvalues to the top */
22 | lua_pushvalue(L, -nup);
23 | lua_pushcclosure(L, l->func, nup); /* closure with those upvalues */
24 | lua_setfield(L, -(nup + 2), l->name);
25 | }
26 | lua_pop(L, nup); /* remove upvalues */
27 | }
28 | LUALIB_API void *luaL_testudata (lua_State *L, int ud, const char *tname) {
29 | void *p = lua_touserdata(L, ud);
30 | if (p != NULL) { /* value is a userdata? */
31 | if (lua_getmetatable(L, ud)) { /* does it have a metatable? */
32 | luaL_getmetatable(L, tname); /* get correct metatable */
33 | if (!lua_rawequal(L, -1, -2)) /* not the same? */
34 | p = NULL; /* value is a userdata with wrong metatable */
35 | lua_pop(L, 2); /* remove both metatables */
36 | return p;
37 | }
38 | }
39 | return NULL; /* value is not a userdata with a metatable */
40 | }
41 | #endif
42 |
43 | /*
44 | ********************************************************************************
45 | * Lua reference structure and functions
46 | ********************************************************************************
47 | */
48 |
49 | typedef struct LuaReferenceStruct {
50 | lua_State *state;
51 | int ref;
52 | } LuaReference;
53 |
54 | static void initLuaReference(LuaReference *r) {
55 | if (r != NULL) {
56 | r->state = NULL;
57 | r->ref = LUA_NOREF;
58 | }
59 | }
60 |
61 | static void registerLuaReference(LuaReference *r, lua_State *l) {
62 | if ((r != NULL) && (l != NULL)) {
63 | if ((r->state != NULL) && (r->ref != LUA_NOREF)) {
64 | luaL_unref(r->state, LUA_REGISTRYINDEX, r->ref);
65 | }
66 | r->state = l;
67 | r->ref = luaL_ref(l, LUA_REGISTRYINDEX);
68 | }
69 | }
70 |
71 | static void unregisterLuaReference(LuaReference *r, lua_State *l) {
72 | if ((r != NULL) && (r->state != NULL) && (r->state == l) && (r->ref != LUA_NOREF)) {
73 | luaL_unref(r->state, LUA_REGISTRYINDEX, r->ref);
74 | r->state = NULL;
75 | r->ref = LUA_NOREF;
76 | }
77 | }
78 |
79 | /*
80 | ********************************************************************************
81 | * Lua webview structure
82 | ********************************************************************************
83 | */
84 |
85 | typedef struct LuaWebViewStruct {
86 | LuaReference cbFn;
87 | lua_State *initState;
88 | struct webview webview;
89 | } LuaWebView;
90 |
91 | #define WEBVIEW_PTR(_cp) \
92 | ((LuaWebView *) ((char *) (_cp) - offsetof(LuaWebView, webview)))
93 |
94 | /*
95 | ********************************************************************************
96 | * Lua webview functions
97 | ********************************************************************************
98 | */
99 |
100 | static int lua_webview_open(lua_State *l) {
101 | const char *url = luaL_checkstring(l, 1);
102 | const char *title = luaL_optstring(l, 2, "Lua Web View");
103 | lua_Integer width = luaL_optinteger(l, 3, 800);
104 | lua_Integer height = luaL_optinteger(l, 4, 600);
105 | lua_Integer resizable = lua_toboolean(l, 5);
106 | webview_run(title, url, width, height, resizable);
107 | return 0;
108 | }
109 |
110 | static LuaWebView *lua_webview_asudata(lua_State *l, int ud) {
111 | if (lua_islightuserdata(l, ud)) {
112 | return lua_touserdata(l, ud);
113 | }
114 | return (LuaWebView *)luaL_checkudata(l, ud, "webview");
115 | }
116 |
117 | static LuaWebView * lua_webview_newuserdata(lua_State *l) {
118 | size_t urlLen;
119 | size_t titleLen;
120 | const char *url = luaL_optlstring(l, 1, "", &urlLen);
121 | const char *title = luaL_optlstring(l, 2, "Lua Web View", &titleLen);
122 | lua_Integer width = luaL_optinteger(l, 3, 800);
123 | lua_Integer height = luaL_optinteger(l, 4, 600);
124 | lua_Integer resizable = lua_toboolean(l, 5);
125 | lua_Integer debug = lua_toboolean(l, 6);
126 | LuaWebView *lwv = (LuaWebView *)lua_newuserdata(l, sizeof(LuaWebView) + titleLen + 1 + urlLen + 1);
127 | const char *titleCopy = ((char *)lwv) + sizeof(LuaWebView);
128 | const char *urlCopy = ((char *)lwv) + sizeof(LuaWebView) + titleLen + 1;
129 | memset(lwv, 0, sizeof(LuaWebView));
130 | memcpy(titleCopy, title, titleLen + 1);
131 | memcpy(urlCopy, url, urlLen + 1);
132 | lwv->initState = NULL;
133 | lwv->webview.title = titleCopy;
134 | lwv->webview.url = urlCopy;
135 | lwv->webview.width = width;
136 | lwv->webview.height = height;
137 | lwv->webview.resizable = resizable;
138 | lwv->webview.debug = debug;
139 | initLuaReference(&lwv->cbFn);
140 | return lwv;
141 | }
142 |
143 | static int lua_webview_new(lua_State *l) {
144 | LuaWebView *lwv = lua_webview_newuserdata(l);
145 | int r = webview_init(&lwv->webview);
146 | if (r != 0) {
147 | return 0;
148 | }
149 | lwv->initState = l;
150 | luaL_getmetatable(l, "webview");
151 | lua_setmetatable(l, -2);
152 | return 1;
153 | }
154 |
155 | static int lua_webview_allocate(lua_State *l) {
156 | (void) lua_webview_newuserdata(l);
157 | luaL_getmetatable(l, "webview");
158 | lua_setmetatable(l, -2);
159 | return 1;
160 | }
161 |
162 | static int lua_webview_init(lua_State *l) {
163 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
164 | int initialized = 0;
165 | if (lwv != NULL && lwv->initState == NULL) {
166 | int r = webview_init(&lwv->webview);
167 | if (r == 0) {
168 | initialized = 1;
169 | lwv->initState = l;
170 | }
171 | }
172 | lua_pushboolean(l, initialized);
173 | return 1;
174 | }
175 |
176 | static int lua_webview_initialized(lua_State *l) {
177 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
178 | lua_pushboolean(l, lwv != NULL && lwv->initState != NULL);
179 | return 1;
180 | }
181 |
182 | static const char *const lua_webview_loop_modes[] = {
183 | "default", "once", "nowait", NULL
184 | };
185 |
186 | static int lua_webview_loop(lua_State *l) {
187 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
188 | int mode = luaL_checkoption(l, 2, "default", lua_webview_loop_modes);
189 | int r = 0;
190 | if (l == lwv->initState) {
191 | do {
192 | r = webview_loop(&lwv->webview, mode != 2);
193 | } while ((mode == 0) && (r == 0));
194 | } else {
195 | webview_debug("loop and init states differs");
196 | }
197 | lua_pushboolean(l, r);
198 | return 1;
199 | }
200 |
201 | static void invoke_callback(struct webview *w, const char *arg) {
202 | if ((w != NULL) && (arg != NULL)) {
203 | LuaWebView *lwv = WEBVIEW_PTR(w);
204 | lua_State *l = lwv->cbFn.state;
205 | int ref = lwv->cbFn.ref;
206 | if ((l != NULL) && (ref != LUA_NOREF)) {
207 | if (l == lwv->initState) {
208 | lua_rawgeti(l, LUA_REGISTRYINDEX, ref);
209 | lua_pushstring(l, arg);
210 | lua_pcall(l, 1, 0, 0);
211 | } else {
212 | webview_debug("callback and init states differs");
213 | }
214 | }
215 | }
216 | }
217 |
218 | static int lua_webview_callback(lua_State *l) {
219 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
220 | if (lua_isfunction(l, 2)) {
221 | lua_pushvalue(l, 2);
222 | registerLuaReference(&lwv->cbFn, l);
223 | lwv->webview.external_invoke_cb = &invoke_callback;
224 | } else {
225 | unregisterLuaReference(&lwv->cbFn, l);
226 | lwv->webview.external_invoke_cb = NULL;
227 | }
228 | return 0;
229 | }
230 |
231 | static void dispatched_eval(struct webview *w, void *arg) {
232 | webview_eval(w, (const char *) arg);
233 | }
234 |
235 | static int lua_webview_eval(lua_State *l) {
236 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
237 | const char *js = luaL_checkstring(l, 2);
238 | int dispatch = lua_toboolean(l, 3);
239 | if (dispatch) {
240 | // do we need to register the js code to dispatch?
241 | webview_dispatch(&lwv->webview, dispatched_eval, (void *)js);
242 | return 0;
243 | }
244 | int r = webview_eval(&lwv->webview, js);
245 | lua_pushboolean(l, r);
246 | return 1;
247 | }
248 |
249 | static int lua_webview_title(lua_State *l) {
250 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
251 | const char *title = luaL_checkstring(l, 2);
252 | webview_set_title(&lwv->webview, title);
253 | return 0;
254 | }
255 |
256 | static int lua_webview_fullscreen(lua_State *l) {
257 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
258 | int fullscreen = lua_toboolean(l, 2);
259 | webview_set_fullscreen(&lwv->webview, fullscreen);
260 | return 0;
261 | }
262 |
263 | static void dispatched_terminate(struct webview *w, void *arg) {
264 | webview_terminate(w);
265 | }
266 |
267 | static int lua_webview_terminate(lua_State *l) {
268 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
269 | int dispatch = lua_toboolean(l, 2);
270 | if (dispatch) {
271 | webview_dispatch(&lwv->webview, dispatched_terminate, NULL);
272 | return 0;
273 | }
274 | webview_terminate(&lwv->webview);
275 | return 0;
276 | }
277 |
278 | static void clean_webview(lua_State *l, LuaWebView *lwv) {
279 | if (lwv != NULL) {
280 | unregisterLuaReference(&lwv->cbFn, l);
281 | if (lwv->initState == l) {
282 | //webview_debug("clean_webview()");
283 | webview_exit(&lwv->webview);
284 | lwv->initState = NULL;
285 | }
286 | }
287 | }
288 |
289 | static int lua_webview_clean(lua_State *l) {
290 | LuaWebView *lwv = (LuaWebView *)lua_webview_asudata(l, 1);
291 | clean_webview(l, lwv);
292 | return 0;
293 | }
294 |
295 | static int lua_webview_lighten(lua_State *l) {
296 | LuaWebView *lwv = (LuaWebView *)luaL_checkudata(l, 1, "webview");
297 | lua_pushlightuserdata(l, lwv);
298 | return 1;
299 | }
300 |
301 | static int lua_webview_asstring(lua_State *l) {
302 | LuaWebView *lwv = (LuaWebView *)luaL_checkudata(l, 1, "webview");
303 | lua_pushlstring(l, (const char *) &lwv, sizeof(void *));
304 | return 1;
305 | }
306 |
307 | static int lua_webview_fromstring(lua_State *l) {
308 | size_t len = 0;
309 | const char *udata = luaL_optlstring(l, 1, NULL, &len);
310 | if (len == sizeof(void *)) {
311 | lua_pushlightuserdata(l, *((void **) udata));
312 | return 1;
313 | }
314 | return 0;
315 | }
316 |
317 | static int lua_webview_gc(lua_State *l) {
318 | LuaWebView *lwv = (LuaWebView *)luaL_testudata(l, 1, "webview");
319 | clean_webview(l, lwv);
320 | return 0;
321 | }
322 |
323 | #if defined(WEBVIEW2_MEMORY_MODULE)
324 | static int lua_webview_loadWebView2Dll(lua_State *l) {
325 | void *data;
326 | size_t len = 0;
327 | int status = 0;
328 | data = lua_tolstring(l, 1, &len);
329 | if (data != NULL && len > 0 && WebView2Load(data, len)) {
330 | status = 1;
331 | }
332 | lua_pushboolean(l, status);
333 | return 1;
334 | }
335 | #endif
336 |
337 | LUALIB_API int luaopen_webview(lua_State *l) {
338 | luaL_newmetatable(l, "webview");
339 | lua_pushstring(l, "__gc");
340 | lua_pushcfunction(l, lua_webview_gc);
341 | lua_settable(l, -3);
342 |
343 | luaL_Reg reg[] = {
344 | { "open", lua_webview_open },
345 | { "new", lua_webview_new },
346 | { "allocate", lua_webview_allocate },
347 | { "clean", lua_webview_clean },
348 | { "init", lua_webview_init },
349 | { "initialized", lua_webview_initialized },
350 | { "loop", lua_webview_loop },
351 | { "eval", lua_webview_eval },
352 | { "callback", lua_webview_callback },
353 | { "terminate", lua_webview_terminate },
354 | { "fullscreen", lua_webview_fullscreen },
355 | { "title", lua_webview_title },
356 | { "lighten", lua_webview_lighten },
357 | { "asstring", lua_webview_asstring },
358 | { "fromstring", lua_webview_fromstring },
359 | #if defined(WEBVIEW2_MEMORY_MODULE)
360 | { "loadWebView2Dll", lua_webview_loadWebView2Dll },
361 | #endif
362 | { NULL, NULL }
363 | };
364 | lua_newtable(l);
365 | luaL_setfuncs(l, reg, 0);
366 | lua_pushliteral(l, "Lua webview");
367 | lua_setfield(l, -2, "_NAME");
368 | lua_pushliteral(l, "1.0");
369 | lua_setfield(l, -2, "_VERSION");
370 | return 1;
371 | }
372 |
--------------------------------------------------------------------------------