Skip to content

Commit f5181a4

Browse files
author
Andy Leverenz
committed
Init
1 parent 3a2fdd0 commit f5181a4

26 files changed

+563
-29
lines changed

app/controllers/posts_controller.rb

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
class PostsController < ApplicationController
2+
before_action :authenticate_user!, except: [:show, :index]
3+
before_action :set_post, only: [:show, :edit, :update, :destroy]
4+
5+
6+
def index
7+
@posts = Post.all
8+
end
9+
10+
def show
11+
end
12+
13+
def new
14+
@post = Post.new
15+
end
16+
17+
def edit
18+
end
19+
20+
def create
21+
@post = Post.new(post_params)
22+
@post.user_id = current_user.id
23+
24+
respond_to do |format|
25+
if @post.save
26+
format.html { redirect_to @post, notice: 'Post was successfully created.' }
27+
format.json { render :show, status: :created, location: @post }
28+
else
29+
format.html { render :new }
30+
format.json { render json: @post.errors, status: :unprocessable_entity }
31+
end
32+
end
33+
end
34+
35+
def update
36+
respond_to do |format|
37+
if @post.update(post_params)
38+
format.html { redirect_to @post, notice: 'Post was successfully updated.' }
39+
format.json { render :show, status: :ok, location: @post }
40+
else
41+
format.html { render :edit }
42+
format.json { render json: @post.errors, status: :unprocessable_entity }
43+
end
44+
end
45+
end
46+
47+
def destroy
48+
@post.destroy
49+
respond_to do |format|
50+
format.html { redirect_to posts_url, notice: 'Post was successfully destroyed.' }
51+
format.json { head :no_content }
52+
end
53+
end
54+
55+
private
56+
57+
def set_post
58+
@post = Post.find(params[:id])
59+
end
60+
61+
def post_params
62+
params.require(:post).permit(:title, :body, :user_id, :feature_image)
63+
end
64+
end

app/helpers/posts_helper.rb

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
module PostsHelper
2+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,155 @@
1+
import Dropzone from "dropzone";
2+
import { Controller } from "stimulus";
3+
import { DirectUpload } from "@rails/activestorage";
4+
import {
5+
getMetaValue,
6+
toArray,
7+
findElement,
8+
removeElement,
9+
insertAfter
10+
} from "helpers";
11+
12+
export default class extends Controller {
13+
static targets = ["input"];
14+
15+
connect() {
16+
this.dropZone = createDropZone(this);
17+
this.hideFileInput();
18+
this.bindEvents();
19+
Dropzone.autoDiscover = false;
20+
}
21+
22+
// Private
23+
hideFileInput() {
24+
this.inputTarget.disabled = true;
25+
this.inputTarget.style.display = "none";
26+
}
27+
28+
bindEvents() {
29+
this.dropZone.on("addedfile", file => {
30+
setTimeout(() => {
31+
file.accepted && createDirectUploadController(this, file).start();
32+
}, 500);
33+
});
34+
35+
this.dropZone.on("removedfile", file => {
36+
file.controller && removeElement(file.controller.hiddenInput);
37+
});
38+
39+
this.dropZone.on("canceled", file => {
40+
file.controller && file.controller.xhr.abort();
41+
});
42+
}
43+
44+
get headers() {
45+
return { "X-CSRF-Token": getMetaValue("csrf-token") };
46+
}
47+
48+
get url() {
49+
return this.inputTarget.getAttribute("data-direct-upload-url");
50+
}
51+
52+
get maxFiles() {
53+
return this.data.get("maxFiles") || 1;
54+
}
55+
56+
get maxFileSize() {
57+
return this.data.get("maxFileSize") || 256;
58+
}
59+
60+
get acceptedFiles() {
61+
return this.data.get("acceptedFiles");
62+
}
63+
64+
get addRemoveLinks() {
65+
return this.data.get("addRemoveLinks") || true;
66+
}
67+
}
68+
69+
class DirectUploadController {
70+
constructor(source, file) {
71+
this.directUpload = createDirectUpload(file, source.url, this);
72+
this.source = source;
73+
this.file = file;
74+
}
75+
76+
start() {
77+
this.file.controller = this;
78+
this.hiddenInput = this.createHiddenInput();
79+
this.directUpload.create((error, attributes) => {
80+
if (error) {
81+
removeElement(this.hiddenInput);
82+
this.emitDropzoneError(error);
83+
} else {
84+
this.hiddenInput.value = attributes.signed_id;
85+
this.emitDropzoneSuccess();
86+
}
87+
});
88+
}
89+
90+
createHiddenInput() {
91+
const input = document.createElement("input");
92+
input.type = "hidden";
93+
input.name = this.source.inputTarget.name;
94+
insertAfter(input, this.source.inputTarget);
95+
return input;
96+
}
97+
98+
directUploadWillStoreFileWithXHR(xhr) {
99+
this.bindProgressEvent(xhr);
100+
this.emitDropzoneUploading();
101+
}
102+
103+
bindProgressEvent(xhr) {
104+
this.xhr = xhr;
105+
this.xhr.upload.addEventListener("progress", event =>
106+
this.uploadRequestDidProgress(event)
107+
);
108+
}
109+
110+
uploadRequestDidProgress(event) {
111+
const element = this.source.element;
112+
const progress = (event.loaded / event.total) * 100;
113+
findElement(
114+
this.file.previewTemplate,
115+
".dz-upload"
116+
).style.width = `${progress}%`;
117+
}
118+
119+
emitDropzoneUploading() {
120+
this.file.status = Dropzone.UPLOADING;
121+
this.source.dropZone.emit("processing", this.file);
122+
}
123+
124+
emitDropzoneError(error) {
125+
this.file.status = Dropzone.ERROR;
126+
this.source.dropZone.emit("error", this.file, error);
127+
this.source.dropZone.emit("complete", this.file);
128+
}
129+
130+
emitDropzoneSuccess() {
131+
this.file.status = Dropzone.SUCCESS;
132+
this.source.dropZone.emit("success", this.file);
133+
this.source.dropZone.emit("complete", this.file);
134+
}
135+
}
136+
137+
function createDirectUploadController(source, file) {
138+
return new DirectUploadController(source, file);
139+
}
140+
141+
function createDirectUpload(file, url, controller) {
142+
return new DirectUpload(file, url, controller);
143+
}
144+
145+
function createDropZone(controller) {
146+
return new Dropzone(controller.element, {
147+
url: controller.url,
148+
headers: controller.headers,
149+
maxFiles: controller.maxFiles,
150+
maxFilesize: controller.maxFileSize,
151+
acceptedFiles: controller.acceptedFiles,
152+
addRemoveLinks: controller.addRemoveLinks,
153+
autoQueue: false
154+
});
155+
}

app/javascript/controllers/hello_controller.js

-18
This file was deleted.

app/javascript/controllers/index.js

+6-6
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
1-
// Load all the controllers within this directory and all subdirectories.
1+
// Load all the controllers within this directory and all subdirectories.
22
// Controller files must be named *_controller.js.
33

4-
import { Application } from "stimulus"
5-
import { definitionsFromContext } from "stimulus/webpack-helpers"
4+
import { Application } from "stimulus";
5+
import { definitionsFromContext } from "stimulus/webpack-helpers";
66

7-
const application = Application.start()
8-
const context = require.context("controllers", true, /_controller\.js$/)
9-
application.load(definitionsFromContext(context))
7+
const application = Application.start();
8+
const context = require.context("controllers", true, /_controller\.js$/);
9+
application.load(definitionsFromContext(context));

app/javascript/helpers/index.js

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
export function getMetaValue(name) {
2+
const element = findElement(document.head, `meta[name="${name}"]`);
3+
if (element) {
4+
return element.getAttribute("content");
5+
}
6+
}
7+
8+
export function findElement(root, selector) {
9+
if (typeof root == "string") {
10+
selector = root;
11+
root = document;
12+
}
13+
return root.querySelector(selector);
14+
}
15+
16+
export function toArray(value) {
17+
if (Array.isArray(value)) {
18+
return value;
19+
} else if (Array.from) {
20+
return Array.from(value);
21+
} else {
22+
return [].slice.call(value);
23+
}
24+
}
25+
26+
export function removeElement(el) {
27+
if (el && el.parentNode) {
28+
el.parentNode.removeChild(el);
29+
}
30+
}
31+
32+
export function insertAfter(el, referenceNode) {
33+
return referenceNode.parentNode.insertBefore(el, referenceNode.nextSibling);
34+
}

app/javascript/stylesheets/components/_forms.scss

+16-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
1-
/* Forms */
1+
@import "dropzone/dist/min/dropzone.min.css";
2+
@import "dropzone/dist/min/basic.min.css";
3+
24
.input {
35
@apply appearance-none block w-full bg-gray-100 text-gray-700 border border-gray-300 rounded py-3 px-4 leading-tight;
46
}
@@ -14,9 +16,21 @@
1416
.select {
1517
@apply appearance-none py-3 px-4 pr-8 block w-full bg-gray-100 border border-gray-300 text-gray-700
1618
rounded leading-tight;
17-
-webkit-appearance: none;
19+
-webkit-appearance: none;
1820
}
1921

2022
.select:focus {
2123
@apply outline-none bg-white border-gray-400;
2224
}
25+
26+
.dropzone {
27+
@apply border-2 rounded-lg border-gray-400 border-dashed;
28+
29+
&.dz-drag-hover {
30+
@apply border-2 rounded-lg border-gray-200 border-dashed;
31+
32+
.dz-message {
33+
opacity: 0.9;
34+
}
35+
}
36+
}

app/models/post.rb

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
class Post < ApplicationRecord
2+
belongs_to :user
3+
has_one_attached :feature_image
4+
end

app/views/posts/_form.html.erb

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
<%= form_with(model: post, local: true, multipart: true) do |form| %>
2+
<% if post.errors.any? %>
3+
<div id="error_explanation">
4+
<h2><%= pluralize(post.errors.count, "error") %> prohibited this post from being saved:</h2>
5+
6+
<ul>
7+
<% post.errors.full_messages.each do |message| %>
8+
<li><%= message %></li>
9+
<% end %>
10+
</ul>
11+
</div>
12+
<% end %>
13+
14+
<div class="mb-6">
15+
<%= form.label :title, class: "label" %>
16+
<%= form.text_field :title, class: "input" %>
17+
</div>
18+
19+
<div class="mb-6">
20+
<%= form.label :body, class: "label" %>
21+
<%= form.text_area :body, class: "input" %>
22+
</div>
23+
24+
<div class="mb-6">
25+
<%= form.label :feature_image, class: "label" %>
26+
<div class="dropzone dropzone-default dz-clickable" data-controller="dropzone" data-dropzone-max-file-size="2" data-dropzone-max-files="1">
27+
<%= form.file_field :feature_image, direct_upload: true, data: { target: 'dropzone.input' } %>
28+
<div class="dropzone-msg dz-message needsclick text-gray-600">
29+
<h3 class="dropzone-msg-title">Drag here to upload or click here to browse</h3>
30+
<span class="dropzone-msg-desc text-sm">2 MB file size maximum. Allowed file types png, jpg.</span>
31+
</div>
32+
</div>
33+
</div>
34+
35+
<div class="mb-6">
36+
<%= form.submit class: "btn-default btn" %>
37+
</div>
38+
<% end %>

app/views/posts/_post.json.jbuilder

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
json.extract! post, :id, :title, :body, :user_id, :created_at, :updated_at
2+
json.url post_url(post, format: :json)

app/views/posts/edit.html.erb

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
<div class="max-w-2xl m-auto">
2+
<h1 class="text-3xl font-bold text-black mb-6">Edit Post</h1>
3+
4+
<%= render 'form', post: @post %>
5+
</div>

0 commit comments

Comments
 (0)