diff options
-rw-r--r-- | app/controllers/categories_controller.rb | 77 | ||||
-rw-r--r-- | app/helpers/categories_helper.rb | 2 | ||||
-rw-r--r-- | app/models/category.rb | 3 | ||||
-rw-r--r-- | app/models/resource_type.rb | 1 | ||||
-rw-r--r-- | app/views/categories/_category.html.erb | 17 | ||||
-rw-r--r-- | app/views/categories/_category.json.jbuilder | 2 | ||||
-rw-r--r-- | app/views/categories/_form.html.erb | 32 | ||||
-rw-r--r-- | app/views/categories/edit.html.erb | 8 | ||||
-rw-r--r-- | app/views/categories/index.html.erb | 21 | ||||
-rw-r--r-- | app/views/categories/index.json.jbuilder | 1 | ||||
-rw-r--r-- | app/views/categories/new.html.erb | 7 | ||||
-rw-r--r-- | app/views/categories/show.html.erb | 15 | ||||
-rw-r--r-- | app/views/categories/show.json.jbuilder | 1 | ||||
-rw-r--r-- | config/routes.rb | 1 | ||||
-rw-r--r-- | db/migrate/20241114170349_create_categories.rb | 11 | ||||
-rw-r--r-- | db/schema.rb | 10 | ||||
-rw-r--r-- | test/controllers/categories_controller_test.rb | 48 | ||||
-rw-r--r-- | test/fixtures/categories.yml | 11 | ||||
-rw-r--r-- | test/models/category_test.rb | 7 | ||||
-rw-r--r-- | test/system/categories_test.rb | 45 |
20 files changed, 319 insertions, 1 deletions
diff --git a/app/controllers/categories_controller.rb b/app/controllers/categories_controller.rb new file mode 100644 index 0000000..5fc872b --- /dev/null +++ b/app/controllers/categories_controller.rb @@ -0,0 +1,77 @@ +class CategoriesController < ApplicationController + before_action :set_category, only: %i[ show edit update destroy ] + before_action :require_admin, only: %i[ new create update destroy ] + + # GET /categories or /categories.json + def index + @categories = Category.all + end + + # GET /categories/1 or /categories/1.json + def show + end + + # GET /categories/new + def new + @category = Category.new + end + + # GET /categories/1/edit + def edit + end + + # POST /categories or /categories.json + def create + @category = Category.new(category_params) + + respond_to do |format| + if @category.save + format.html { redirect_to @category, notice: "Category was successfully created." } + format.json { render :show, status: :created, location: @category } + else + format.html { render :new, status: :unprocessable_entity } + format.json { render json: @category.errors, status: :unprocessable_entity } + end + end + end + + # PATCH/PUT /categories/1 or /categories/1.json + def update + respond_to do |format| + if @category.update(category_params) + format.html { redirect_to @category, notice: "Category was successfully updated." } + format.json { render :show, status: :ok, location: @category } + else + format.html { render :edit, status: :unprocessable_entity } + format.json { render json: @category.errors, status: :unprocessable_entity } + end + end + end + + # DELETE /categories/1 or /categories/1.json + def destroy + @category.destroy! + + respond_to do |format| + format.html { redirect_to categories_path, status: :see_other, notice: "Category was successfully destroyed." } + format.json { head :no_content } + end + end + + private + # Use callbacks to share common setup or constraints between actions. + def set_category + @category = Category.find(params.expect(:id)) + end + + # Only allow a list of trusted parameters through. + def category_params + params.expect(category: [ :name, :colour, :badge_foreground_colour ]) + end + + def require_admin + unless Current.session.user&.is_admin + redirect_to root_path, notice: "You must be an admin to perform this action." + end + end +end diff --git a/app/helpers/categories_helper.rb b/app/helpers/categories_helper.rb new file mode 100644 index 0000000..e06f315 --- /dev/null +++ b/app/helpers/categories_helper.rb @@ -0,0 +1,2 @@ +module CategoriesHelper +end diff --git a/app/models/category.rb b/app/models/category.rb new file mode 100644 index 0000000..296e7d6 --- /dev/null +++ b/app/models/category.rb @@ -0,0 +1,3 @@ +class Category < ApplicationRecord + validates :name, presence: true, uniqueness: true +end diff --git a/app/models/resource_type.rb b/app/models/resource_type.rb index 066a734..414e71e 100644 --- a/app/models/resource_type.rb +++ b/app/models/resource_type.rb @@ -1,3 +1,4 @@ class ResourceType < ApplicationRecord has_many :pdfresources + validates :name, presence: true, uniqueness: true end diff --git a/app/views/categories/_category.html.erb b/app/views/categories/_category.html.erb new file mode 100644 index 0000000..3c51d8f --- /dev/null +++ b/app/views/categories/_category.html.erb @@ -0,0 +1,17 @@ +<div id="<%= dom_id category %>"> + <p class="my-5"> + <strong class="block font-medium mb-1">Name:</strong> + <%= category.name %> + </p> + + <p class="my-5"> + <strong class="block font-medium mb-1">Colour:</strong> + <%= category.colour %> + </p> + + <p class="my-5"> + <strong class="block font-medium mb-1">Badge foreground colour:</strong> + <%= category.badge_foreground_colour %> + </p> + +</div> diff --git a/app/views/categories/_category.json.jbuilder b/app/views/categories/_category.json.jbuilder new file mode 100644 index 0000000..c0950c8 --- /dev/null +++ b/app/views/categories/_category.json.jbuilder @@ -0,0 +1,2 @@ +json.extract! category, :id, :name, :colour, :badge_foreground_colour, :created_at, :updated_at +json.url category_url(category, format: :json) diff --git a/app/views/categories/_form.html.erb b/app/views/categories/_form.html.erb new file mode 100644 index 0000000..fec6c3b --- /dev/null +++ b/app/views/categories/_form.html.erb @@ -0,0 +1,32 @@ +<%= form_with(model: category, class: "contents") do |form| %> + <% if category.errors.any? %> + <div id="error_explanation" class="bg-red-50 text-red-500 px-3 py-2 font-medium rounded-lg mt-3"> + <h2><%= pluralize(category.errors.count, "error") %> prohibited this category from being saved:</h2> + + <ul> + <% category.errors.each do |error| %> + <li><%= error.full_message %></li> + <% end %> + </ul> + </div> + <% end %> + + <div class="my-5"> + <%= required_label_tag(form, :name) %> + <%= form.text_field :name, class: "block shadow rounded-md border border-gray-400 outline-none px-3 py-2 mt-2 w-full" %> + </div> + + <div class="my-5"> + <%= form.label :colour, class: "font-bold" %> + <%= form.text_field :colour, class: "block shadow rounded-md border border-gray-400 outline-none px-3 py-2 mt-2 w-full" %> + </div> + + <div class="my-5"> + <%= form.label :badge_foreground_colour, class: "font-bold" %> + <%= form.text_field :badge_foreground_colour, class: "block shadow rounded-md border border-gray-400 outline-none px-3 py-2 mt-2 w-full" %> + </div> + + <div class="inline"> + <%= form.submit class: "rounded-lg py-3 px-5 bg-blue-600 text-white inline-block font-medium cursor-pointer" %> + </div> +<% end %> diff --git a/app/views/categories/edit.html.erb b/app/views/categories/edit.html.erb new file mode 100644 index 0000000..57da660 --- /dev/null +++ b/app/views/categories/edit.html.erb @@ -0,0 +1,8 @@ +<div class="mx-auto md:w-2/3 w-full"> + <h1 class="font-bold text-4xl">Editing category</h1> + + <%= render "form", category: @category %> + + <%= link_to "Show this category", @category, class: "ml-2 rounded-lg py-3 px-5 bg-gray-100 inline-block font-medium" %> + <%= link_to "Back to categories", categories_path, class: "ml-2 rounded-lg py-3 px-5 bg-gray-100 inline-block font-medium" %> +</div> diff --git a/app/views/categories/index.html.erb b/app/views/categories/index.html.erb new file mode 100644 index 0000000..68911e6 --- /dev/null +++ b/app/views/categories/index.html.erb @@ -0,0 +1,21 @@ +<div class="w-full"> + <% if notice.present? %> + <p class="py-2 px-3 bg-green-50 mb-5 text-green-500 font-medium rounded-lg inline-block" id="notice"><%= notice %></p> + <% end %> + + <% content_for :title, "Categories" %> + + <div class="flex justify-between items-center"> + <h1 class="font-bold text-4xl">Categories</h1> + <%= link_to "New category", new_category_path, class: "rounded-lg py-3 px-5 bg-blue-600 text-white block font-medium" %> + </div> + + <div id="categories" class="min-w-full"> + <% @categories.each do |category| %> + <%= render category %> + <p> + <%= link_to "Show this category", category, class: "ml-2 rounded-lg py-3 px-5 bg-gray-100 inline-block font-medium" %> + </p> + <% end %> + </div> +</div> diff --git a/app/views/categories/index.json.jbuilder b/app/views/categories/index.json.jbuilder new file mode 100644 index 0000000..aa5baad --- /dev/null +++ b/app/views/categories/index.json.jbuilder @@ -0,0 +1 @@ +json.array! @categories, partial: "categories/category", as: :category diff --git a/app/views/categories/new.html.erb b/app/views/categories/new.html.erb new file mode 100644 index 0000000..fae8770 --- /dev/null +++ b/app/views/categories/new.html.erb @@ -0,0 +1,7 @@ +<div class="mx-auto md:w-2/3 w-full"> + <h1 class="font-bold text-4xl">New category</h1> + + <%= render "form", category: @category %> + + <%= link_to "Back to categories", categories_path, class: "ml-2 rounded-lg py-3 px-5 bg-gray-100 inline-block font-medium" %> +</div> diff --git a/app/views/categories/show.html.erb b/app/views/categories/show.html.erb new file mode 100644 index 0000000..e38975f --- /dev/null +++ b/app/views/categories/show.html.erb @@ -0,0 +1,15 @@ +<div class="mx-auto md:w-2/3 w-full flex"> + <div class="mx-auto"> + <% if notice.present? %> + <p class="py-2 px-3 bg-green-50 mb-5 text-green-500 font-medium rounded-lg inline-block" id="notice"><%= notice %></p> + <% end %> + + <%= render @category %> + + <%= link_to "Edit this category", edit_category_path(@category), class: "mt-2 rounded-lg py-3 px-5 bg-gray-100 inline-block font-medium" %> + <%= link_to "Back to categories", categories_path, class: "ml-2 rounded-lg py-3 px-5 bg-gray-100 inline-block font-medium" %> + <div class="inline-block ml-2"> + <%= button_to "Destroy this category", @category, method: :delete, class: "mt-2 rounded-lg py-3 px-5 bg-gray-100 font-medium" %> + </div> + </div> +</div> diff --git a/app/views/categories/show.json.jbuilder b/app/views/categories/show.json.jbuilder new file mode 100644 index 0000000..30e6b47 --- /dev/null +++ b/app/views/categories/show.json.jbuilder @@ -0,0 +1 @@ +json.partial! "categories/category", category: @category diff --git a/config/routes.rb b/config/routes.rb index 1e8b325..3455716 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -1,4 +1,5 @@ Rails.application.routes.draw do + resources :categories resources :resource_types resources :pdfresources resources :users diff --git a/db/migrate/20241114170349_create_categories.rb b/db/migrate/20241114170349_create_categories.rb new file mode 100644 index 0000000..983f85c --- /dev/null +++ b/db/migrate/20241114170349_create_categories.rb @@ -0,0 +1,11 @@ +class CreateCategories < ActiveRecord::Migration[8.0] + def change + create_table :categories do |t| + t.string :name + t.string :colour + t.string :badge_foreground_colour + + t.timestamps + end + end +end diff --git a/db/schema.rb b/db/schema.rb index e09d8c2..8c534b7 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -10,7 +10,7 @@ # # It's strongly recommended that you check this file into your version control system. -ActiveRecord::Schema[8.0].define(version: 2024_11_14_152810) do +ActiveRecord::Schema[8.0].define(version: 2024_11_14_170349) do create_table "active_storage_attachments", force: :cascade do |t| t.string "name", null: false t.string "record_type", null: false @@ -39,6 +39,14 @@ ActiveRecord::Schema[8.0].define(version: 2024_11_14_152810) do t.index ["blob_id", "variation_digest"], name: "index_active_storage_variant_records_uniqueness", unique: true end + create_table "categories", force: :cascade do |t| + t.string "name" + t.string "colour" + t.string "badge_foreground_colour" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + end + create_table "pdfresources", force: :cascade do |t| t.string "name" t.string "stripe_product_id" diff --git a/test/controllers/categories_controller_test.rb b/test/controllers/categories_controller_test.rb new file mode 100644 index 0000000..a21af3b --- /dev/null +++ b/test/controllers/categories_controller_test.rb @@ -0,0 +1,48 @@ +require "test_helper" + +class CategoriesControllerTest < ActionDispatch::IntegrationTest + setup do + @category = categories(:one) + end + + test "should get index" do + get categories_url + assert_response :success + end + + test "should get new" do + get new_category_url + assert_response :success + end + + test "should create category" do + assert_difference("Category.count") do + post categories_url, params: { category: { badge_foreground_colour: @category.badge_foreground_colour, colour: @category.colour, name: @category.name } } + end + + assert_redirected_to category_url(Category.last) + end + + test "should show category" do + get category_url(@category) + assert_response :success + end + + test "should get edit" do + get edit_category_url(@category) + assert_response :success + end + + test "should update category" do + patch category_url(@category), params: { category: { badge_foreground_colour: @category.badge_foreground_colour, colour: @category.colour, name: @category.name } } + assert_redirected_to category_url(@category) + end + + test "should destroy category" do + assert_difference("Category.count", -1) do + delete category_url(@category) + end + + assert_redirected_to categories_url + end +end diff --git a/test/fixtures/categories.yml b/test/fixtures/categories.yml new file mode 100644 index 0000000..9074eff --- /dev/null +++ b/test/fixtures/categories.yml @@ -0,0 +1,11 @@ +# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html + +one: + name: MyString + colour: MyString + badge_foreground_colour: MyString + +two: + name: MyString + colour: MyString + badge_foreground_colour: MyString diff --git a/test/models/category_test.rb b/test/models/category_test.rb new file mode 100644 index 0000000..869357c --- /dev/null +++ b/test/models/category_test.rb @@ -0,0 +1,7 @@ +require "test_helper" + +class CategoryTest < ActiveSupport::TestCase + # test "the truth" do + # assert true + # end +end diff --git a/test/system/categories_test.rb b/test/system/categories_test.rb new file mode 100644 index 0000000..acff020 --- /dev/null +++ b/test/system/categories_test.rb @@ -0,0 +1,45 @@ +require "application_system_test_case" + +class CategoriesTest < ApplicationSystemTestCase + setup do + @category = categories(:one) + end + + test "visiting the index" do + visit categories_url + assert_selector "h1", text: "Categories" + end + + test "should create category" do + visit categories_url + click_on "New category" + + fill_in "Badge foreground colour", with: @category.badge_foreground_colour + fill_in "Colour", with: @category.colour + fill_in "Name", with: @category.name + click_on "Create Category" + + assert_text "Category was successfully created" + click_on "Back" + end + + test "should update Category" do + visit category_url(@category) + click_on "Edit this category", match: :first + + fill_in "Badge foreground colour", with: @category.badge_foreground_colour + fill_in "Colour", with: @category.colour + fill_in "Name", with: @category.name + click_on "Update Category" + + assert_text "Category was successfully updated" + click_on "Back" + end + + test "should destroy Category" do + visit category_url(@category) + click_on "Destroy this category", match: :first + + assert_text "Category was successfully destroyed" + end +end |