Graphite/frontend/src
0HyperCube 81c3420470 Keyboard menu/widget navigation (#628)
* Keyboard menu navigation

* Fix dropdown keyboard navigation

* Fix merge error

* Some code review

* Interactive dropdowns

* Query by data attr not class name

* Add locking behaviour

* Change query selector style

* Change query selector style (again)

* Code review feedback

* Fix highlighted entry regression

* Styling and disabling checkbox tabindex in MenuLists

* Don't redirect space off canvas to backend

* Do not emit update if value same

* Escape closes all floating menus

* Close dropdowns on blur

Co-authored-by: Keavon Chambers <keavon@keavon.com>
2022-05-25 22:10:18 +01:00
..
components Keyboard menu/widget navigation (#628) 2022-05-25 22:10:18 +01:00
io-managers Keyboard menu/widget navigation (#628) 2022-05-25 22:10:18 +01:00
state-providers Add frontend file structure docs and some related cleanup 2022-05-23 19:13:51 -07:00
utility-functions Update the assets directory to prefix icon folders with "icon-" 2022-05-22 00:06:10 -07:00
wasm-communication Refactor source of Git commit build info (closes #661) 2022-05-24 14:33:58 -07:00
App.vue Keyboard menu/widget navigation (#628) 2022-05-25 22:10:18 +01:00
main.ts Vue initialization and FloatingMenu codebase refactoring and cleanup (#649) 2022-05-21 19:46:15 -07:00
README.md Add frontend file structure docs and some related cleanup 2022-05-23 19:13:51 -07:00
TwoViewTest.vue Rework wasm initialization and reduce global state (#379) 2021-12-20 07:37:19 +01:00
types.d.ts Improve web infrastructure 2021-08-26 19:20:38 -07:00
volar.d.ts Reorganize Vue components and add documentation explaining each folder 2022-05-23 19:03:34 -07:00

Overview of /frontend/src/

Vue components: components/

Vue components that build the Graphite editor GUI, which are mounted in App.vue. These are Vue SFCs (single-file components) which each contain a Vue-templated HTML section, an SCSS (Stylus CSS) section, and a script section. The aim is to avoid implementing much editor business logic here, just enough to make things interactive and communicate to the backend where the real business logic should occur.

I/O managers: io-managers/

TypeScript files which manage the input/output of browser APIs and link this functionality with the editor backend. These files subscribe to backend events to execute JS APIs, and in response to these APIs or user interactions, they may call functions into the backend (defined in /frontend/wasm/editor_api.rs).

Each I/O manager is a self-contained module where one instance is created in App.vue when it's mounted to the DOM at app startup.

During development when HMR (hot-module replacement) occurs, these are also unmounted to clean up after themselves, so they can be mounted again with the updated code. Therefore, any side-effects that these managers cause (e.g. adding event listeners to the page) need a destructor function that cleans them up. The destructor function, when applicable, is returned by the module and automatically called in App.vue on unmount.

State providers: state-providers/

TypeScript files which provide reactive state and importable functions to Vue components. Each module defines a Vue reactive state object const state = reactive({ ... }); and exports this from the module in the returned object as the key-value pair state: readonly(state) as typeof state, using Vue's readonly() wrapper. Other functions may also be defined in the module and exported after state, which provide a way for Vue components to call functions to manipulate the state.

In App.vue, an instance of each of these are given to Vue's provide() function. This allows any component to access the state provider instance by specifying it in its inject: [...] array. The state is accessed in a component with this.stateProviderName.state.someReactiveVariable and any exposed functions are accessed with this.stateProviderName.state.someExposedVariable(). They can also be used in the Vue HTML template (sans the this. prefix).

I/O managers vs. state providers

Some state providers, similarly to I/O managers, may subscribe to backend events, call functions from editor_api.rs into the backend, and interact with browser APIs and user input. The difference is that state providers are meant to injected by components to use them for reactive state, while I/O managers are meant to be self-contained systems that operate for the lifetime of the application and aren't touched by Vue components.

Utility functions: utility-functions/

TypeScript files which define and export individual helper functions for use elsewhere in the codebase. These files should not persist state outside each function.

WASM communication: wasm-communication/

TypeScript files which serve as the JS interface to the WASM bindings for the editor backend.

WASM editor: editor.ts

Instantiates the WASM and editor backend instances. The function initWasm() asynchronously constructs and initializes an instance of the WASM bindings JS module provided by wasm-bindgen/wasm-pack. The function createEditor() constructs an instance of the editor backend. In theory there could be multiple editor instances sharing the same WASM module instance. The function returns an object where raw is the WASM module, instance is the editor, and subscriptions is the subscription router (described below).

initWasm() occurs in main.ts right before the Vue application exists, then createEditor() is run in App.vue during the Vue app's creation. Similarly to the state providers described above, the editor is provided so other components can inject it and call functions on this.editor.raw, this.editor.instance, or this.editor.subscriptions.

Message definitions: messages.ts

Defines the message formats and data types received from the backend. Since Rust and JS support different styles of data representation, this bridges the gap from Rust into JS land. Messages (and the data contained within) are serialized in Rust by serde into JSON, and these definitions are manually kept up-to-date to parallel the message structs and their data types. (However, directives like #[serde(skip)] or #[serde(rename = "someOtherName")] may cause the TypeScript format to look slightly different from the Rust structs.) These definitions are basically just for the sake of TypeScript to understand the format, although in some cases we may perform data conversion here using translation functions that we can provide.

Subscription router: subscription-router.ts

Associates messages from the backend with subscribers in the frontend, and routes messages to subscriber callbacks. This module provides a subscribeJsMessage(messageType, callback) function which JS code throughout the frontend can call to be registered as the exclusive handler for a chosen message type. This file's other exported function, handleJsMessage(messageType, messageData, wasm, instance), is called in editor.ts by the associated editor instance when the backend sends a FrontendMessage. When this occurs, the subscription router delivers the message to the subscriber for given messageType by executing its registered callback function. As an argument to the function, it provides the messageData payload transformed into its TypeScript-friendly format defined in messages.ts.

Vue app: App.vue

The entry point for the Vue application. This is where we define global CSS style rules, construct the editor,construct/destruct the editor and I/O managers, and construct/provide state providers.

Entry point: main.ts

The entry point for the entire project. Here we simply initialize the WASM module with await initWasm(); then initialize the Vue application with createApp(App).mount("#app");.