动态嵌套form,使用Stimulus Js库(前后端不分离)
我的git代码:https://github.com/chentianwei411/nested_form-Stimulus-
Stimulus: https://www.cnblogs.com/chentianwei/p/9806875.html
开始:
rails new -m ../jumpstart,\ Gorails视频\(创建一个rails模版\)/template.rb -d postgresql nested_forms
rails webpacker:install:stimulus
在_header.html.erb内添加:
使用javascript_pack_tag方法添加JS pack到Rails views
<%= javascript_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
然后修改文件名:
mv app/javascript/controllers/{hello.nested_form}_controller.js
创建手脚架和模型:
rails g scaffold Project name description
rails g model Task description project:belongs_to
rails db:migrate
产生:
create_table "projects", force: :cascade do |t|
t.string "name"
t.string "description"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
end create_table "tasks", force: :cascade do |t|
t.string "description"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.bigint "project_id"
t.index ["project_id"], name: "index_tasks_on_project_id"
end
嵌套结构的类方法使用:
class Project < ApplicationRecord
has_many :tasks, inverse_of: :project
accepts_nested_attributes_for :tasks, reject_if: :all_blank, allow_destroy: true
end
具体解释见博客:
ActiveRecord Nested Atrributes 关联记录,对嵌套属性进行CURD
Rails-Treasure chest2 嵌套表单;
controller内添加属性白名单:
def project_params
params.require(:project).permit(:name, :description, tasks_attributes: [:id, :description, :_destroy])
end
然后在view视图_form.html.erb上添加一个嵌套的form builder:
<h4>Tasks</h4> <%= form.fields_for :tasks do |task| %>
<div class="form-group">
<%= task.label :description %>
<%= task.text_field :description, class: 'form-control'%>
</div>
<% end %>
使用#fields_for(record_name, record_object = nil, &block),创建一个scope在一个指定的model对象如form_for,但是不创建自身的form tags。它用于在一个form内创建额外的model对象。(具体见api文档)
修改project_controller,添加语句@project.tasks.new。或者设置视图中的#fields_for方法的第2个参数record_object为Task.new
def new
@project = Project.new
@project.tasks.new
end
⚠️:重构视图,可以把fields_for方法的块block提取出来。_nest.html.erb
使用template标签,让JavaScript决定是否显示在页面。配合使用stimulus.js。
<template>
<%= form.fields_for :tasks, Task.new, child_index: 'New_RECORD' do |task| %>
<%= render 'nest', form: task%>
<% end %>
</template>
child_index: 'New_RECORD' 即子索引的名字,会在label,input标签的for,name, id属性上使用到:
<div class="form-group">
<label for="project_tasks_attributes_New_RECORD_description">Description</label>
<input class="form-control" type="text" name="project[tasks_attributes][New_RECORD][description]" id="project_tasks_attributes_New_RECORD_description">
</div>
视图的最终代码:
- data-controller调用js文件中的类对象实例化的对象
- data-target, 用于取对应元素。
- data-action,用于绑定事件。这里使用了link_to视图方法。
<div data-controller="nested-form">
<template data-target="nested-form.template">
<%= form.fields_for :tasks, Task.new, child_index: 'New_RECORD' do |task| %>
<%= render 'nest', form: task%>
<% end %>
</template> <%= form.fields_for :tasks do |task| %>
<%= render 'nest', form: task%>
<% end %> <div class="mb-3" data-target='nested-form.links'>
<%= link_to 'Add Task', '#', class: 'btn btn-outline-primary', data: {action: 'click->nested-form#add_association'} %>
</div>
</div>
nested_form_controller.rb
export default class extends Controller {
static targets = [ "links", "template" ]
connect() {
}
add_association(event) {
event.preventDefault()
var content = this.templateTarget.innerHTML.replace(/New_RECORD/g, new Date().getTime())
this.linksTargets.insertAdjacentHTML('beforebegin', content)
}
}
⚠️这里用到js库的2个方法replace和insertAdjacentHTML用于对元素内容操作和元素节点的插入。
视图上的最终代码:
<input class="form-control" type="text" name="project[tasks_attributes][1554196211709][description]" id="project_tasks_attributes_1554196211709_description">
功能:增加task,还可以删除task。
在_nest.html.erb表格内增加一个"Remove"连接按钮。
<%= content_tag :div, class:'nested-fields', data: {new_record: form.object.new_record?} do %>
<div class="form-group">
<%= form.label :description %>
<%= form.text_field :description, class: 'form-control'%>
<small><%= link_to "REMOVE", "#", data: {action: "click->nested-form#remove_association"}%></small>
<!-- <%= form.hidden_field :_destroy%> -->
</div>
<% end %>
data-new-record用于判断是不是新建数据。
添加一个data-action绑定事件remove_association。
再看nested_form_controller.js,添加事件:
remove_association(event) {
event.preventDefault()
let wrapper = event.target.closest(".nested-fields") #有nest-fileds类的元素
if (wrapper.dataset.newRecord == "true") {
wrapper.remove()
} else {
console.log("不能删除")
}
}
注意:
dataset属性提供了读写所有客制化的data属性 data-*
考虑到edit界面,有已经添加的task和新增的task,所以要区分,新增的直接从nom树移除,已经添加的则要发出删除请求。
修改上面的代码:
_nest.html.erb内添加一个input标签,并隐藏。它的用途是标记作用!
在js中添加2行代码:
- 找到这个input标签,并设置value等于1或者true, 这样更新时,会自动判断是否删除。
- 从节点树上隐藏这个元素。CSS#display属性设置none。
remove_association(event) {
event.preventDefault()
let wrapper = event.target.closest(".nested-fields")
if (wrapper.dataset.newRecord == "true") {
wrapper.remove()
} else {
wrapper.querySelector("input[name*='_destroy']").value = true
wrapper.style.display = 'none'
}
}
动态嵌套form,使用Stimulus Js库(前后端不分离)的更多相关文章
- Vue 应用 nginx 配置 前后端不分离模式
一.先在官网下载nginx 软件,解压后放在软件盘中如D盘 将nginx 文件夹拖到编译器中,打开conf 文件夹中的 nginx.conf 文件,找到其中的server {} 配置项,默认35 行. ...
- List多个字段标识过滤 IIS发布.net core mvc web站点 ASP.NET Core 实战:构建带有版本控制的 API 接口 ASP.NET Core 实战:使用 ASP.NET Core Web API 和 Vue.js 搭建前后端分离项目 Using AutoFac
List多个字段标识过滤 class Program{ public static void Main(string[] args) { List<T> list = new List& ...
- Node.js实现前后端交互——用户注册
我之前写过一篇关于使用Node.js作为后端实现用户登陆的功能,现在再写一下node.js做后端实现简单的用户注册实例吧.另外需要说的是,上次有大佬提醒需要加密数据传输,不应该使用明文传输用户信息.在 ...
- 前后端不分离部署教程(基于Vue,Nginx)
有小伙伴私信问我vue项目是如何进行前后端不分离打包发布的,那我岂能坐视不管,如此宠粉的我肯定是要给发一篇教程的,话不多说,开始操作 前端假如我们要发布我们的Vue项目,假设我们前端用的是histor ...
- 在IDEA中使用Maven将SpringBoot项目打成jar包、同时运行打成的jar包(前后端项目分离)
1.maven教程官网 https://m.runoob.com/maven/ 2.理解Maven的构建生命周期(clean.Package) 3.在项目中使用maven进行打包 4.运行打包好的ja ...
- ASP.NET Core 实战:使用 ASP.NET Core Web API 和 Vue.js 搭建前后端分离项目
一.前言 这几年前端的发展速度就像坐上了火箭,各种的框架一个接一个的出现,需要学习的东西越来越多,分工也越来越细,作为一个 .NET Web 程序猿,多了解了解行业的发展,让自己扩展出新的技能树,对自 ...
- .Net Core与Vue.js模块化前后端分离快速开发解决方案(NetModular)
NetModular是什么? NetModular不仅仅是一个框架,它也是一整套的模块化与前后端分离的快速开发的解决方案,目标是致力于开箱即用,让开发人员完全专注于业务开发,不需要关心底层封装和实现. ...
- 一行代码实现Vue微信支付,无需引用wexin-sdk库,前后端分离HTML微信支付,无需引用任何库
前后端分离项目实现微信支付的流程: 1:用户点击支付 2:请求服务端获取支付参数 3:客户端通过JS调起微信支付(微信打开的网页) * 本文主要解决的是第3步,视为前两步已经完成,能正确拿到支付参数, ...
- 【转载】java前后端 动静分离,JavaWeb项目为什么我们要放弃jsp?
原文:http://blog.csdn.net/piantoutongyang/article/details/50878214 今天看到两篇文章,讲解 为什么web开发启用jsp,确实挺有道理,整理 ...
随机推荐
- Apache的功能模块
本人这几天一直在看apache相关的书籍,稍微说下apache的结构 本人的制图: Apache一共有五层功能结构. 从底层到上依次为: 第一层: 名称:操作系统支持层 功能:操作系统可以提供底层功能 ...
- 设置VS2017背景图片
设置方法很简单:安装扩展ClaudiaIDE 1.在这里下载扩展,https://visualstudiogallery.msdn.microsoft.com/9ba50f8d-f30c-4e33-a ...
- Linux常用操作指令(面试专用)
Linux:免费开源,多用户多任务,衍生出很多附属版本,例如常用的RedHat... 常用指令 ls 显示文件或目录 -l 列出文件详细信息l(list) -a ...
- Delphi调用API函数获取Windows目录信息、获取System目录信息、获取Temp临时文件目录信息
var Str1, Str2: Array[..Max_Path]of Char;//开辟缓冲区 Str3: Array[..]of Char; begin GetWindowsDirectory(@ ...
- OxyPlot Controller OxyPlot控制器
Default input bindings The default input bindings in the PlotController are: Action Gesture Pan* Rig ...
- Winform 图片预览列表+分页显示
针对图片列表展示信息,一开始没有做过相关类似的功能,大多都是以表格行显示为主,所以刚开始实现这个功能的时候是懵逼的.无从下口.在网上搜索一时半会也没找到合适的解决方案.大致就是类似于下图这样,每条数据 ...
- 2018-2019-20175205 实验三敏捷开发与XP实践《Java开发环境的熟悉》实验报告
2018-2019-20175205 实验三敏捷开发与XP实践<Java开发环境的熟悉>实验报告 实验要求 没有Linux基础的同学建议先学习<Linux基础入门(新版)>&l ...
- storm-sql-kafka问题情况
首先上官方文档:http://storm.apache.org/releases/1.2.2/storm-sql.html 解决的问题 1.kafka版本不对 开始测试时采用storm1.2.2+ka ...
- Kafka启动报错 : ERROR Processor got uncaught exception
参照我之前的一篇博文Kafka学习之(二)Centos下安装Kafka安装了kafka并启动,状况并不像我之前最初的预期,报错了,并且我在当前Linux环境下安装的Java版本.Kafka版本都是和之 ...
- 面试-java反射
问题:简述Java中的反射使用 答: 1.作用: 可以通过配置文件来动态配置和加载类,以实现软件工程理论里所提及的类与类,模块与模块之间的解耦.反射最经典的应用是spring框架. 2. 定义 反射简 ...