Here is a list of tasks to do (moved to CHANGELOG.md once done).
- Support Component Attribute Update Listeners (nest a
on_someattribute_update
listener inside the content block of a consumed Glimmer Web Component for any attribute/option on the Component) - Support Glimmer Web Component Attribute Data-Binding (by nesting data-binding declarations within consumed components)
- Support formatting paragraph elements as stand-alone elements (not inside p)
- Automate Rails 7 setup instructions
- Automate Rails 6 setup instructions
- Consider renaming
element-ID
css classes toglimmer_element_ID
for a more unique class name that nobody else would be using - Consider removing
element
css class from elements except the root, and maybe rename the css class toglimmer_element_root
. Or maybe replace withglimmer-element
where element comes from ElementProxy (it would beglimmer-formatting-element
if it's FormattingProxy) - Implement
inspect
method for library classes likeElementProxy
,ListenerProxy
, andEventProxy
(especially) to help with troubleshooting. - Alert user if they attempt to build a component or component slot that shadows an HTML element
- Prepend/append/insert element operations
-
Router support to enable friendly URLs when needed Example: route '#hello-world' do require 'glimmer-dsl-web/samples/hello/hello_world.rb' end
route do div { button('Run Hello, World! Sample') { onclick do Glimmer::Web::Router.visit('#hello-world') end } } end
- Generate backend Rails views with Glimmer DSL for XML syntax as a Ruby alternative to ERB
- Server-Side Rendering of Glimmer Web Components that were designed originally for the Frontend, with automatic hydration upon rendering (hooking event listeners and data-bindings) [enable isomorphism by either supporting a server-side code path checker (
server?
andclient?
), or by enabling the import of js libraries in a flexible way that would make the code still work if a js library is not available in the backend). Start by supporting component options that are primitive data types (e.g.String
,Boolean
,Integer
, etc...)
- Automatically generate Rails Model forms with an authenticity token correctly
- Support Content Data-Binding progress circle (and ability to update with any image)
- Smart form components that can automatically generate the name attributes for a Rails model just like the Rails form helpers
- Support Ruby WASM: https://github.com/ruby/ruby.wasm
- Consider removing Opal-jQuery if possible switching to pure JS Document calls via Opal, to allow supporting WASM with technology neutral code
- Support integrating with official HTML Web Components to allow reusing Web Component libraries from Glimmer DSL for Web
- JavaScript Canvas API
- Contribute to Opal-Rails change to create app/assets/opal/application.rb instead of app/assets/javascript/application.js.rb as the latter is confusing (or at least an option)
- Model Proxies (Use Backend Models in the Frontend through Automatically Generated REST Controllers for ActiveRecord models with secure whitelisting of the attributes/instance-methods/class-methods that need to be exposed only. For example, calling Purchases.limit(5) in the Frontend would call a Backend Purchase model indirectly via a PurchaseProxy that securely whitelists all available attributes/methods on Purchase)
- Model Proxy Observers (Observe Backend Model events like creation, update, destruction, etc... via automatically generated Websocket-based channels for observing Backend Models view Proxy Observers)
- Consider optimizing performance of registering dom listeners by including real onevent (e.g. onclick) attributes (instead of registering via JS calls) where that works tgat would register the listener in first use. This works in tandem witj building html for all dom elements at once for much faster initial rendering
- Enhance performance optimization of registering dom listeners by including real onevent (e.g. onclick) attributes by having it handle data-binding listener registrations too
- Optimize performance of listener expressions by memoizing listeners or by creating Glimmer module methods on the fly
- Optimize performance of style expressions by turning into a static expression that behaves as both a property and an element
- Optimize performance of content expressions by turning into a static expression that behaves as both an open content block adder and a content data-binding block
- Consider optimizing Content Data-Binding with Virtual DOM
- As an optimization in Content Data-Binding, consider saving rendered DOMs per model attribute values and reshowing them instead of recreating them.
- As an optimization in Content Data-Binding, consider diffing, removing all listeners and re-installing listeners on the same elements.
- Optimize performance of Shine Data-Binding syntax by having
<=>
and<=
invoke data-binding logic directly without going throughbind
method (this optimization has to happen in the glimmer gem with a new version number if it breaks APIs) - Optimize performance of Glimmer CSS DSL with shortcut methods
- Optimize performance of a and span formatting html elements by processing them conditionally in their static expressions
- Optimize performance of
observe(*args)
keyword through memoization or some other solution - Render styles produced by
Glimmer::Web::Component
style {}
blocks in bulk - Optimize performance of
ElementProxy
respond_to_missing? & method_missing by memoizing results
- Implement
DateTime#strptime
&Date#strptime
(unless the Opal project beats me to it) - Support data-binding fine-grained
style
CSS properties on element more conveniently like withstyle.background somevalue
,style-background somevalue
,style { background something }
, or something similar. labeled_input
custom control (or simply support label as an option ininput
)- Consider supporting higher abstraction flavors of data-binding for various elements like
table
andselect
- Consider supporting input and textarea data-binding changes on focus out (onblur) instead of direct onchange, by specifying an extra option
- Consider supporting input and textarea built-in debounce (not firing change to model till the user stopped typing for a period of time) [and explore a way to allow people to support any write/read strategy as part of this]
- Consider preventing
ElementProxy
from returningnil
if an invalid method name was invoked (likevali
). It seems today, it works, but returnsnil
. Maybe, have it error out instead. - Consider using element.method format in DSL to generate an element with a class (or element_class__id)
- Consider supporting an automatic router (auto-generates history based on user actions). Not sure if we have to remember the full DOM or full Ruby Glimmer GUI DSL structure of every change and replay different pages based on that information when a route is entered or the user hits the back/forward buttons.
- Consider having form validation auto-setup by model validations using ActiveModel::Model, ActiveRecord (porting to the frontend), HyperStack, or something similar (provided by Glimmer if needed).
- Consider idea of setting a different default ID for rendering other than
body
(that could be set globally or temporarily in a scope) - Consider optionally supporting 2nd arg of
option
inComponent
as:default
- Support setting element
style
CSS properties with Glimmer DSL for CSS when using the nested property version (assuming Glimmer DSL for CSS underwent some improvements as per its TODO next items) - Build a Rails generator for installing this gem properly in a Rails app
- Consider the idea of having Formatting Elements actually return elements, not strings, but with a properly implemented
to_s
method. - Consider optimizing component rendering by pre-generating a component template, and copying (if/else conditions are simulated with show/hide on an element)
- Consider generating a nonce for style tags (elements)
- Consider ensuring display of elements immediately upon render by using setTimeout to render them (and perhaps limit that to 2nd level under root to avoid being too fine grained). Or you can have it render by accumulation count setting (like every 100 elements or every 1000 elements). Or you can do special rendering for controls that are known to contain a lot of things like
table
. - Generate Rails scaffolding directly with Glimmer GUI DSL code
- Support backend server-side rendering of Glimmer Web components to help with SEO purposes on certain pages
- Use backend server-side rendering of Glimmer Web components in glimmer_component Rails helper with an option that would pre-render the HTML in the backend, and then augment it with JS behavior using Glimmer GUI DSL on the frontend
- Consider idea of auto-routing with components by embedding component keyword and its args into URL path parameters (or serializing paramters and deserializing them)
- Support automatically displaying a progress bar or circle when rendering a component (like a giant one that might take more than the tolerable 500ms.. although that should be rare)
- Provide form helpers similar to ERB form helpers, but in Glimmer DSL
- Consider providing Serializer support in the backend that would yield the same models used in the Frontend
- Provide a generic Api class to enable frontend to pull data from backend with as little arguments as possible (e.g.
Api.show('sample', id) {|sample| }
invokes show action on samples_controller under/samples
). It could become customizable in the future. - Support setting
style
on Glimmer Web Components and having its value override thestyle
value on the markup root (unless it made more sense to concatenate it) - Support setting
class
on Glimmer Web Components and having its value override theclass
value on the markup root (unless it made more sense to concatenate it) - Display exceptions in a Modal (perhaps make it an option)
- See why element_proxy.css('something', 'something') doesn't work and requires .dom_element.css() instead.
- Consider providing an alternative CSS style syntax that is Hash based as it could be good enough for some applications.
- Support content_for_each as an optimized version of
content
data-binding that would diff models first before updating elements. content_for_each(presenter, :todos) { |todo| todo_list_item(presenter:, todo:) } - Supporting embedding consumer markup anywhere in a used component by supporting block properties
- Support integration with standard HTML Web Components
- Resolve namespaced components by preferring the current namespace module we are in first if no namespace was specified (glimmer library must be aware of current namespace by checking in method_missing and passing that information to the Glimmer DSL Engine in some stack)
- Use descendants_tracker gem instead of implementing descendants manually
- Provide an example for integrating with React components using https://github.com/bitovi/react-to-web-component or https://lit.dev/docs/frameworks/react/
- Consider cashing DOM element in every ElementProxy to avoid looking it up again with jQuery (in case that speeds up performance)
- Auto-generate CSS for inline styles in
head
of document by assigning them to CSS classes and adding the CSS classes to elements instead - Consider adding a way to simplify data-binding selection, like adding a 'selected' CSS class to an element based on whether something matches another value as selection
- Consider adding a way to simplify data-binding enablement
- Consider supporting this convenience syntax
class_name(:completed, :active, :editing) <= todo
, which expectstodo
to have methods matching the class names data-bound to it - Support
style 'string'
block inGlimmer::Web::Component
- Auto-generate CSS for embedded (internal) styles in
head
of document via CSS classes (as a performance optimization similar to that of Styled Components). This might not be really needed. Performance is generally very fast without this. - Support always being able to pass id, class, and string content to glimmer web component as an extra argument
- Implement
methods
forEventProxy
given that it pulls most of its method names dynamically through method_missing - Implement
methods
forElementProxy
given that it pulls most of its method names dynamically through method_missing - Consider printing exception.message before re-raising exception if exceptions are encountered in rendering with Glimmer DSL. This makes it easier to troubleshoot than reading exception in browser, which sometimes hides it when displaying stack trace.
- Consider adding element.children_components to avoid having to go through element.children.map(&:component) manually
- Consider automatically escaping the element inner_html and String content while providing a
raw
method to allow setting unescaped HTML in a similar way to Rails.
- It seems
after_render
inside components does not run after component elements are rendered. Look into it
- Sample showing a progress bar or progress circle (or add a web component to support it)
- Add an example of form submission via ajax request
- Add an example of a multi-page form wizard with breadcrumbs and step numbers while submitting a form via Ajax after filling every page
- Add nice CSS styling to some samples
- svg sample
- Document in README how to troubleshoot Opal code, including 3 cases, Ruby interpretation issues, Ruby syntax issues, and crazy issue that requires
rm -rf tmp/cache
and restart of server - Document all exceptions that occur during rendering of Glimmer Web Components instead of silently dying (or the Glimmer DSL in general if that is not already happening)
- Document how to reuse Ruby code both server-side (backend) and client-side (frontend)
- Document Style Guide of Glimmer DSL for Web (partially borrowed from Glimmer libui library)