Skip to content

Commit 7b7616a

Browse files
authored
Merge pull request #12 from avo-hq/fix/routes-not-automatically-mounted
2 parents fa655a5 + 48d7714 commit 7b7616a

File tree

21 files changed

+226
-1637
lines changed

21 files changed

+226
-1637
lines changed

.github/workflows/ci.yml

+14-14
Original file line numberDiff line numberDiff line change
@@ -6,20 +6,20 @@ on:
66
branches: [ main ]
77

88
jobs:
9-
lint:
10-
runs-on: ubuntu-latest
11-
steps:
12-
- name: Checkout code
13-
uses: actions/checkout@v4
14-
15-
- name: Set up Ruby
16-
uses: ruby/setup-ruby@v1
17-
with:
18-
ruby-version: ruby-3.3.1
19-
bundler-cache: true
20-
21-
- name: Lint code for consistent style
22-
run: bin/rubocop -f github
9+
# lint:
10+
# runs-on: ubuntu-latest
11+
# steps:
12+
# - name: Checkout code
13+
# uses: actions/checkout@v4
14+
15+
# - name: Set up Ruby
16+
# uses: ruby/setup-ruby@v1
17+
# with:
18+
# ruby-version: ruby-3.3.1
19+
# bundler-cache: true
20+
21+
# - name: Lint code for consistent style
22+
# run: bin/rubocop -f github
2323

2424
test:
2525
runs-on: ubuntu-latest

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -17,3 +17,5 @@ node_modules
1717
*.local
1818

1919
app/assets/builds
20+
21+
vite.config.ts.timestamp-*.mjs

Gemfile

+6-1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@ source "https://rubygems.org"
33
# Specify your gem's dependencies in marksmith.gemspec.
44
gemspec
55

6+
gem "rails", ">= 7.0.0"
7+
68
gem "puma"
79

810
gem "sqlite3"
@@ -18,6 +20,9 @@ gem 'vite_rails'
1820
gem 'stimulus-rails'
1921
gem 'turbo-rails'
2022

23+
group :development do
24+
gem "web-console", ">= 3.3.0"
25+
end
26+
2127
# Access an interactive console on exception pages or by calling 'console' anywhere in the code.
22-
gem "web-console", ">= 3.3.0"
2328
gem "listen", ">= 3.5.1"

Gemfile.lock

+2-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ PATH
22
remote: .
33
specs:
44
marksmith (0.1.1)
5-
rails (>= 7.0.0)
5+
activesupport
66
redcarpet
77

88
GEM
@@ -305,6 +305,7 @@ DEPENDENCIES
305305
marksmith!
306306
propshaft
307307
puma
308+
rails (>= 7.0.0)
308309
rubocop-rails-omakase
309310
sqlite3
310311
stimulus-rails

README.md

+83-15
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,9 @@ It supports Active Storage attachments and comes with a built-in mardown preview
88

99
![Marksmith demo](./marksmith.gif)
1010

11+
> [!WARNING]
12+
> Marksmith is at the initial stage of development. It's nearing a beta release, but the API might change and bugs are expected. Please continue to use the library and report any issues in the GitHub repo.
13+
1114
Temporary live demo here, under the description field: [https://main.avodemo.com/avo/resources/projects/new](https://main.avodemo.com/avo/resources/projects/new)
1215

1316
## Usage
@@ -18,7 +21,7 @@ Temporary live demo here, under the description field: [https://main.avodemo.com
1821

1922
## Installation
2023

21-
### 1. Add `marksmith` to your `Gemfile`
24+
#### 1. Add `marksmith` to your `Gemfile`
2225

2326
Have Bundler add it by running this command:
2427

@@ -35,20 +38,27 @@ Add this line to your application's Gemfile:
3538
gem "marksmith"
3639
```
3740

38-
### 2. Install the NPM package to import the StimulusJS controller.
41+
#### 2. Install the NPM package to import the StimulusJS controller.
3942

40-
Install the package.
43+
Install the package from npmjs.org.
4144

4245
```bash
4346
$ yarn add @avo-hq/marksmith
4447
```
4548

46-
Import and register it in your application.
49+
Or pin it using importmap.
50+
51+
```bash
52+
bin/importmap pin @avo-hq/marksmith
53+
```
54+
55+
Import and register the controllers in your application. The `ListContinuationController` is optional and only needed if you want to have continued lists in your markdown.
4756

4857
```js
49-
import { MarksmithController } from '@avo-hq/marksmith'
58+
import { MarksmithController, ListContinuationController } from '@avo-hq/marksmith'
5059

5160
application.register('marksmith', MarksmithController)
61+
application.register('list-continuation', ListContinuationController)
5262
```
5363

5464
> [!NOTE]
@@ -68,28 +78,78 @@ import { MarksmithController } from '@avo-hq/marksmith/core'
6878
application.register('marksmith', MarksmithController)
6979
```
7080

71-
### 3. Add the style tag to your `application.html` layout
81+
#### 3. Add the style tag to your `application.html` layout
7282

7383
```erb
7484
<%= stylesheet_link_tag "marksmith" %>
7585
```
7686

77-
### 4. Use it
87+
#### 4. Use it
7888

7989
Use a form helper tag or attach it to your form builder.
8090

8191
```erb
8292
<%= marksmith_tag :body, value: "### This is important" %>
8393
or
84-
<%= @form.marksmith :body %>
94+
<%= form.marksmith :body %>
8595
```
8696

87-
## Options
97+
## Configuration
98+
99+
Marksmith accepts a few configuration options.
100+
101+
### Field options
88102

89103
The field supports a few of the regular options like `disabled`, `placeholder`, `autofocus`, `style`, `class`, `rows`, `data`, and `value`, but also a custom one.
90104

91105
`extra_preview_params` - Sends extra params to the preview renderer.
92106

107+
`enable_file_uploads` - Whether to enable file uploads.
108+
109+
`upload_url` - The URL to use for file uploads. If not provided, the editor will use the `rails_direct_uploads_url` helper.
110+
111+
```erb
112+
<%= marksmith_tag :body,
113+
disabled: true,
114+
placeholder: "Write your best markdown here.",
115+
extra_preview_params: { foo: "bar" },
116+
enable_file_uploads: true,
117+
upload_url: nil
118+
%>
119+
```
120+
121+
### Eject configuration file
122+
123+
Marksmith comes with a default configuration file that you can eject to your app.
124+
125+
```bash
126+
bin/rails generate marksmith:install
127+
```
128+
129+
This will create a `config/initializers/marksmith.rb` file in your app.
130+
131+
### Mount path
132+
133+
The engine is mounted by default at `/marksmith`. You can change it by setting `Marksmith.configuration.mount_path` to a different path.
134+
135+
```ruby
136+
# config/initializers/marksmith.rb
137+
Marksmith.configure do |config|
138+
config.mount_path = "/markdown"
139+
end
140+
```
141+
142+
### Mounting the engine
143+
144+
The engine is mounted by default, but you can disable it by setting `Marksmith.configuration.automatically_mount_engine` to `false` and then manually mount the engine in your `routes.rb` file.
145+
146+
```ruby
147+
# config/routes.rb
148+
Rails.application.routes.draw do
149+
mount Marksmith::Engine => Marksmith.configuration.mount_path
150+
end
151+
```
152+
93153
## Built-in preview renderer
94154

95155
The renderer is powered by [`Redcarpet`](https://github.com/vmg/redcarpet).
@@ -101,13 +161,15 @@ In your `show.html.erb` view or the place where you want to render the compiled
101161
<%== marksmithed post.body %>
102162
```
103163

104-
## Using with importmap
164+
> [!WARNING]
165+
> Using the `<%==` tag will output the raw HTML, so ensure you sanitize the content to avoid XSS attacks.
166+
>
167+
> See how we do it [here](https://github.com/avo-hq/avo/blob/main/app/views/marksmith/shared/_rendered_body.html.erb#L2).
168+
> ```ruby
169+
> # sample sanitization
170+
> sanitize(body, tags: %w(table th tr td span) + ActionView::Helpers::SanitizeHelper.sanitizer_vendor.safe_list_sanitizer.allowed_tags.to_a)
171+
> ```
105172
106-
It should be as simple as running this command and have it pinned in your `importmap.rb` file.
107-
108-
```bash
109-
bin/importmap pin @avo-hq/marksmith
110-
```
111173
112174
## Active Storage
113175
@@ -130,6 +192,12 @@ application.register('marksmith', MarksmithController)
130192
application.register('list-continuation', ListContinuationController)
131193
```
132194
195+
## Who uses Marksmith?
196+
197+
- [Avo](https://avohq.io) - Ruby on Rails Code-Based App Builder Framework
198+
199+
*Open a PR and add your project here*
200+
133201
## Contributing
134202

135203
Contribution directions go here.

app/assets/javascripts/marksmith_controller-full.esm.js

+9
Original file line numberDiff line numberDiff line change
@@ -2827,10 +2827,15 @@ var MarksmithController = (function () {
28272827
fieldId: String,
28282828
galleryEnabled: { type: Boolean, default: false },
28292829
galleryOpenPath: String,
2830+
fileUploadsEnabled: { type: Boolean, default: true },
28302831
}
28312832

28322833
static targets = ['fieldContainer', 'fieldElement', 'previewElement', 'writeTabButton', 'previewTabButton', 'toolbar']
28332834

2835+
get #fileUploadsDisabled() {
2836+
return !this.fileUploadsEnabledValue
2837+
}
2838+
28342839
connect() {
28352840
subscribe(this.fieldContainerTarget, { defaultPlainTextPaste: { urlLinks: true } });
28362841
}
@@ -2878,11 +2883,15 @@ var MarksmithController = (function () {
28782883
}
28792884

28802885
dropUpload(event) {
2886+
if (this.#fileUploadsDisabled) return
2887+
28812888
event.preventDefault();
28822889
this.#uploadFiles(event.dataTransfer.files);
28832890
}
28842891

28852892
pasteUpload(event) {
2893+
if (this.#fileUploadsDisabled) return
2894+
28862895
if (!event.clipboardData.files.length) return
28872896

28882897
event.preventDefault();

app/assets/javascripts/marksmith_controller-no-stimulus.esm.js

+9
Original file line numberDiff line numberDiff line change
@@ -2337,10 +2337,15 @@ var MarksmithController = (function (stimulus) {
23372337
fieldId: String,
23382338
galleryEnabled: { type: Boolean, default: false },
23392339
galleryOpenPath: String,
2340+
fileUploadsEnabled: { type: Boolean, default: true },
23402341
}
23412342

23422343
static targets = ['fieldContainer', 'fieldElement', 'previewElement', 'writeTabButton', 'previewTabButton', 'toolbar']
23432344

2345+
get #fileUploadsDisabled() {
2346+
return !this.fileUploadsEnabledValue
2347+
}
2348+
23442349
connect() {
23452350
subscribe(this.fieldContainerTarget, { defaultPlainTextPaste: { urlLinks: true } });
23462351
}
@@ -2388,11 +2393,15 @@ var MarksmithController = (function (stimulus) {
23882393
}
23892394

23902395
dropUpload(event) {
2396+
if (this.#fileUploadsDisabled) return
2397+
23912398
event.preventDefault();
23922399
this.#uploadFiles(event.dataTransfer.files);
23932400
}
23942401

23952402
pasteUpload(event) {
2403+
if (this.#fileUploadsDisabled) return
2404+
23962405
if (!event.clipboardData.files.length) return
23972406

23982407
event.preventDefault();

app/frontend/entrypoints/javascript/controllers/marksmith_controller.js

+9
Original file line numberDiff line numberDiff line change
@@ -17,10 +17,15 @@ export default class extends Controller {
1717
fieldId: String,
1818
galleryEnabled: { type: Boolean, default: false },
1919
galleryOpenPath: String,
20+
fileUploadsEnabled: { type: Boolean, default: true },
2021
}
2122

2223
static targets = ['fieldContainer', 'fieldElement', 'previewElement', 'writeTabButton', 'previewTabButton', 'toolbar']
2324

25+
get #fileUploadsDisabled() {
26+
return !this.fileUploadsEnabledValue
27+
}
28+
2429
connect() {
2530
subscribe(this.fieldContainerTarget, { defaultPlainTextPaste: { urlLinks: true } })
2631
}
@@ -68,11 +73,15 @@ export default class extends Controller {
6873
}
6974

7075
dropUpload(event) {
76+
if (this.#fileUploadsDisabled) return
77+
7178
event.preventDefault()
7279
this.#uploadFiles(event.dataTransfer.files)
7380
}
7481

7582
pasteUpload(event) {
83+
if (this.#fileUploadsDisabled) return
84+
7685
if (!event.clipboardData.files.length) return
7786

7887
event.preventDefault()

app/views/marksmith/shared/_editor.html.erb

+20-3
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,20 @@
1313
local_assigns[:value] || nil
1414
end
1515
extra_preview_params = local_assigns[:extra_preview_params] || {}
16+
upload_url = if local_assigns[:upload_url].present?
17+
local_assigns[:upload_url]
18+
elsif defined?(ActiveStorage)
19+
main_app.rails_direct_uploads_url
20+
end
21+
22+
enable_file_uploads = if upload_url.blank?
23+
false
24+
elsif local_assigns[:enable_file_uploads].nil?
25+
true
26+
else
27+
local_assigns[:enable_file_uploads]
28+
end
29+
1630

1731
# Used by Avo and other adapters to enable the gallery link.
1832
gallery_enabled = local_assigns.dig(:gallery, :enabled) || false
@@ -35,8 +49,9 @@
3549
",
3650
unique_selector: ".#{@input_id}", # used to pinpoint the exact element in which to insert the attachment
3751
marksmith_preview_url_value: marksmith.markdown_previews_path,
52+
marksmith_file_uploads_enabled_value: enable_file_uploads,
3853
marksmith_active_tab_class: "bg-white",
39-
marksmith_attach_url_value: main_app.rails_direct_uploads_url,
54+
marksmith_attach_url_value: upload_url,
4055
marksmith_extra_preview_params_value: extra_preview_params.as_json,
4156
**local_assigns.fetch(:controller_data_attributes, {})
4257
} do %>
@@ -85,8 +100,10 @@
85100
<%= link_to "https://docs.github.com/github/writing-on-github/getting-started-with-writing-and-formatting-on-github/basic-writing-and-formatting-syntax", target: "_blank", class: class_names("ms:flex ms:items-center ms:text-neutral-800 ms:no-underline", toolbar_button_classes) do %>
86101
<%= image_tag asset_path("marksmith/svgs/markdown.svg"), class: "ms:inline ms:size-4 ms:mr-1" %> <%= t("marksmith.markdown_is_supported").humanize %>
87102
<% end %>
88-
<%= button_tag data: { action: "click->marksmith#buttonUpload" }, class: class_names("ms:bg-none ms:border-none ms:bg-transparent ms:text-neutral-600 ms:items-center ms:flex", toolbar_button_classes) do %>
89-
<%= image_tag asset_path("marksmith/svgs/paperclip.svg"), class: "ms:inline ms:size-4 ms:mr-1" %> <%= t("marksmith.upload_files").humanize %>
103+
<% if enable_file_uploads %>
104+
<%= button_tag data: { action: "click->marksmith#buttonUpload" }, class: class_names("ms:bg-none ms:border-none ms:bg-transparent ms:text-neutral-600 ms:items-center ms:flex", toolbar_button_classes) do %>
105+
<%= image_tag asset_path("marksmith/svgs/paperclip.svg"), class: "ms:inline ms:size-4 ms:mr-1" %> <%= t("marksmith.upload_files").humanize %>
106+
<% end %>
90107
<% end %>
91108
<% if gallery_enabled %>
92109
<%= link_to gallery_full_path, data: { turbo_frame: gallery_turbo_frame }, class: class_names("ms:flex ms:items-center ms:text-neutral-800 ms:no-underline", toolbar_button_classes) do %>

0 commit comments

Comments
 (0)