Roll20 uses cookies to improve your experience on our site. Cookies enable you to enjoy certain features, social sharing functionality, and tailor message and display ads to your interests on our site and others. They also help us understand how our site is being used. By continuing to use our site, you consent to our use of cookies. Update your cookie preferences .
×
Create a free account

Custom Checkbox style, request for help (almost there!)

August 13 (5 months ago)

Edited August 13 (5 months ago)
Steve
Pro
Sheet Author

I've almost got it. I just need somebody to point out the obvious error my eyes won't see.

I've styled, using the label substitution method, a clickable image to function as a checkbox.

The remaining problem I have is that it doesn't play nice with multiple grid areas.

Here's how it looks as written:



While this works, It’s not how I’d prefer the layout. The shield image(checkbox) to be on the left. So the grid template areas of afattributes should be:

grid-template-areas:
    "afheader afheader afheader"
    "imgshieldcol aftype aftypeinput"
    "imgshieldcol afname afinput" 
    "shieldhidden shieldhidden shieldhidden";
}

 

Only when I do this, aftype/aftypeinput appear beside the imgshieldcol normally, but the afname/afinput shunt down to just below the imgshieldcol. It's sort of in the right place, but not exactly:


I've experimented with changing the size (heigh/width) attributes of the shield image to be more responsive to the size of the grid container/s it is in, but anything other than a rigidly defined size results in a tiny, or non-existent image. In this screenshot example I've used auto for width/height (see the tiny grey dot?):



Here's the current code snips I think are relevant:

.character
  grid-template-columns: 29% 34% 36%;
  align-items: stretch;
  justify-items: stretch;
  justify-content: stretch;
  align-content: stretch;
  grid-template-areas:
    "common common common"
    "primaryattributes afattributes finalcombatattributes"
    "atkdefattributes hpattributes finalcombatattributes"
    "evaattribute hpattributes finalcombatattributes"
    "mamdattributes mpattributes classcombatabilities"
    "steperattributes mpattributes classcombatabilities"
    "gapfillimg mpattributes classcombatabilities";
}
/* Grid layout for attack/defense attributes with image shields */
.afattributes {
  grid-area: afattributes;
  grid-template-columns: 1fr 1fr 1fr;
  grid-template-rows: auto;
  display: grid;
  border-style: double;
  border-width: 6px;
  grid-template-areas:
    "afheader afheader afheader"
    "aftype aftypeinput aftypeinput"
    "afname afinput imgshieldcol"
    "shieldhidden shieldhidden shieldhidden";
}
/* Grid layout for attack/defense attributes with image shields */
.afattributes {
  grid-area: afattributes;
  grid-template-columns: 1fr 1fr 1fr;
  grid-template-rows: auto;
  display: grid;
  border-style: double;
  border-width: 6px;
  grid-template-areas: 
    "afheader afheader afheader"
    "aftype aftypeinput aftypeinput"
    "afname afinput imgshieldcol"
    "shieldhidden shieldhidden shieldhidden"; 
}
/* Grid areas for attack/defense attributes */
.afheader {
  grid-area: afheader;
}
.aftype {
  grid-area: aftype;
}
.aftypeinput {
  grid-area: aftypeinput;
}
.afname {
  grid-area: afname;
}
.afinput {
  grid-area: afinput;
}
.shieldhidden {
  display: none;
  text-align: center;
  grid-area: shieldhidden;
  grid-template-areas: 
    "shielddie shieldbutton";
}
/* Grid areas for shield attributes */
.shielddie {
  grid-area: shielddie;
}
.shieldbutton {
  grid-area: shieldbutton;
}
/* Toggle display of shield-related elements based on shield-toggle input */
input.shield-toggle[value="1"] ~ div.shieldhidden {
  display: grid;
}
/* Ensure the image container fits within its grid area */
.imgshieldcol {
  display: grid; /* Use grid to center content */
  align-items: center;
  justify-content: space-evenly;
}
/* Style for the checkbox container */
.checkbox-container {
  position: relative;
  display: grid; 
}
/* Hide the default checkbox */
.checkbox-input {
  position: absolute;
  width: 0;
  height: 0;
  opacity: 0;
  cursor: pointer; /* Ensure the checkbox still responds to clicks */
}
/* Style the image */
.checkbox-image {
  display: grid;
  width: 51px; /* Set the image width explicitly */
  height: 118px; /* Set the image height explicitly */
  background: url('https://raw.githubusercontent.com/Roll20/roll20-character-sheets/master/Dragon%20Warriors/DW_Shieldnobg.png') no-repeat center center;
  background-size: contain; /* Ensure the image scales correctly */
}
/* Style the text overlay */
.checkbox-text {
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%, -50%);
  color: white;
  font-size: 24px;
  font-weight: bold;
  display: none; /* Hide text by default */
}
/* Show text when checkbox is checked */
.checkbox-input:checked ~ .checkbox-image ~ .checkbox-text {
  display: block;
}
/* Grid area for martial attributes */
.mpattributes {
  grid-area: mpattributes;
  border-style: double;
  border-width: 6px;}

 

<div class="afattributes">
                       <input type="hidden" class="shield-toggle" name="attr_shield" value="0" />
                       <div class="afheader">
                               <label class="boxheading">
                                      Armour
                               </label>
                       </div>
                       <div class="aftype">
                               <label>
                                      Type
                               </label>
                       </div>
                       <div class="aftypeinput">
                               <input type="text" class="flexitextinput" name="attr_Armour" value="None" readonly />
                       </div>
            <div class="imgshieldcol">
                <label class="checkbox-container">
                    <input type="checkbox" class="checkbox-input" name="attr_shield" value="1">
                    <span class="checkbox-image"></span>
                    <span class="checkbox-text"></span>
                </label>
            </div>
                       <div class="afname">
                               <label>
                                      Armour Factor
                               </label>
                       </div>
                       <div class="afinput">
                               <label>
                                      <input title="Includes Magic and Miscellaneous Bonuses" type="text" class="fourshorttextinput" name="attr_totalaf" value="0" readonly />
                               <br>
                                      <br>
                                      Click shield to equip:
                               </label>
                       </div>
                       <div class="shieldhidden">
                               <div class="shielddie">
                                      Roll
                                      <input title="Modified by Guard/Abilities" type="text" class="fourshorttextinput" name="attr_shieldsuccessvalue" value="1" readonly />
                                      on
                                      <input class="fourshorttextinput" type="text" name="attr_shielddievalue" value="1d6" readonly />
                               </div>
                               <div class="shieldbutton">
                                      <button class="d20" type="roll" value="&{template:default} {{name=@{character_name} attempts to block with a Shield}} {{Shield roll= [[@{shielddievalue}]]}} {{Target Result= [[@{shieldsuccessvalue}]] }}" name="roll_shieldcheck" >
                                      </button>
                               </div>
                       </div>
               </div>

August 14 (5 months ago)

Edited August 14 (5 months ago)
vÍnce
Pro
Sheet Author

Hi Steve,
one option is to use your "desired" grid areas

    grid-template-areas:
    "afheader afheader afheader"
    "imgshieldcol aftype aftypeinput"
    "imgshieldcol afname afinput"
    "shieldhidden shieldhidden shieldhidden";


and then just add a position and negative margin to your shield column class to allow the other areas to shift upward.

.imgshieldcol {
    display: grid;
    align-items: center;
    justify-content: space-evenly;
    position: relative;
    margin-bottom: -6em;
}

or instead of position and margin,
use a grid span to span those extra rows that are getting pushed down.

.imgshieldcol {
display: grid;
align-items: center;
justify-content: space-evenly;
grid-row: 1/5;
grid-column-start: 1;
}


August 14 (5 months ago)

Edited August 14 (5 months ago)
Steve
Pro
Sheet Author

Ah that's a big step forward in sorting out the layout, thanks vÍnce , the second approach did just the trick.

or instead of position and margin, use a grid span to span those extra rows that are getting pushed down.

.imgshieldcol {
display: grid;
align-items: center;
justify-content: space-evenly;
grid-row: 1/5;
grid-column-start: 1;
}

Now I just have to work out how to make the shield image size responsive to the size of its grid container.

August 14 (5 months ago)

Edited August 14 (5 months ago)
vÍnce
Pro
Sheet Author

You can try making your image a background-image with size cover or contain with no-repeat..?  I can never remember. lol  Just experiment.
https://developer.mozilla.org/en-US/docs/Web/CSS/background-size


August 14 (5 months ago)
Scott C.
Forum Champion
Sheet Author
API Scripter
Compendium Curator

so, the issue that is causing the sizing issue is that you are using position:absolute/relative. Absolute is going to make the size separate from the grid without some additional code. Instead of doing that additional code, you can just use the features of grid.

That CSS looks like:

/* there was a typo here such that there wasn't an opening curly brace for the character styling. This caused your first .afattributes styling to be ignored. */
.ui-dialog .tab-content .charsheet .character{
  grid-template-columns: 29% 34% 36%;
  align-items: stretch;
  justify-items: stretch;
  justify-content: stretch;
  align-content: stretch;
  grid-template-areas:
    "common common common"
    "primaryattributes afattributes finalcombatattributes"
    "atkdefattributes hpattributes finalcombatattributes"
    "evaattribute hpattributes finalcombatattributes"
    "mamdattributes mpattributes classcombatabilities"
    "steperattributes mpattributes classcombatabilities"
    "gapfillimg mpattributes classcombatabilities";
}
/* Grid layout for attack/defense attributes with image shields */
.ui-dialog .tab-content .charsheet .afattributes {
  grid-area: afattributes;
  grid-template-columns: 1fr 1fr 1fr;
  grid-template-rows: auto;
  display: grid;
  border-style: double;
  border-width: 6px;
  grid-template-areas:
  "afheader afheader afheader"
  "imgshieldcol aftype aftypeinput"
  "imgshieldcol afname afinput" 
  "shieldhidden shieldhidden shieldhidden";
}
/* you had duplicated the afattributes styling */
/* Grid layout for attack/defense attributes with image shields
.ui-dialog .tab-content .charsheet .afattributes {
  grid-area: afattributes;
  grid-template-columns: 1fr 1fr 1fr;
  grid-template-rows: auto;
  display: grid;
  border-style: double;
  border-width: 6px;
  grid-template-areas: 
    "afheader afheader afheader"
    "aftype aftypeinput aftypeinput"
    "afname afinput imgshieldcol"
    "shieldhidden shieldhidden shieldhidden"; 
} */
/* Grid areas for attack/defense attributes */
.ui-dialog .tab-content .charsheet .afheader {
  grid-area: afheader;
}
.ui-dialog .tab-content .charsheet .aftype {
  grid-area: aftype;
}
.ui-dialog .tab-content .charsheet .aftypeinput {
  grid-area: aftypeinput;
}
.ui-dialog .tab-content .charsheet .afname {
  grid-area: afname;
}
.ui-dialog .tab-content .charsheet .afinput {
  grid-area: afinput;
}
.ui-dialog .tab-content .charsheet .shieldhidden {
  display: none;
  text-align: center;
  grid-area: shieldhidden;
  grid-template-areas: 
    "shielddie shieldbutton";
}
/* Grid areas for shield attributes */
.ui-dialog .tab-content .charsheet .shielddie {
  grid-area: shielddie;
}
.ui-dialog .tab-content .charsheet .shieldbutton {
  grid-area: shieldbutton;
}
/* Toggle display of shield-related elements based on shield-toggle input */
input.shield-toggle[value="1"] ~ div.shieldhidden {
  display: grid;
}
/* Ensure the image container fits within its grid area */
.ui-dialog .tab-content .charsheet .imgshieldcol {
  display: grid;
  position: relative;
  /* assign the imgshieldcol to the appropriate grid-area */
  grid-area: imgshieldcol;
}
.ui-dialog .tab-content .charsheet .checkbox-container {
  display: grid;
  /* define an area where we want all content to go */
  grid-template-areas: 'content';
  cursor: pointer;
}
.ui-dialog .tab-content .charsheet .checkbox-container > *{
  /* place all direct children of the checkbox container in the content area */
  grid-area: content;
}
/* Hide the default checkbox */
.ui-dialog .tab-content .charsheet .checkbox-input {
  /* note that this bit of code actually doesn't have anything to do with whether the checkbox responds to clicks or not */
  /* cursor: pointer; Ensure the checkbox still responds to clicks */
  /* since the checkbox is in a label, we can just set it to display:none */
  display: none;
}
/* Style the image */
.ui-dialog .tab-content .charsheet .checkbox-image {
  /* display: grid; No need to give this a display of grid. It doesn't have any children, so nothing to put in a grid. */
  background: url('https://raw.githubusercontent.com/Roll20/roll20-character-sheets/master/Dragon%20Warriors/DW_Shieldnobg.png') no-repeat center center;
  background-size: contain; /* Ensure the image scales correctly */
}
/* Style the text overlay */
.ui-dialog .tab-content .charsheet .checkbox-text {
  /* position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%, -50%); */
  place-self: center; /* this line replaces the four above it */
  color: black;
  font-size: 24px;
  font-weight: bold;
}
/* Show text when checkbox is checked */
.ui-dialog .tab-content .charsheet .checkbox-input:not(:checked) ~ .checkbox-text {
  display: none;
  /* reverse the text display logic so that you don't have to redefine it's display when it should display. Also, we simplify the selectors so that we aren't needlessly complex. */
}

Note that I added some prepends to the css declarations to ensure that we override the Roll20 styling. I highly recommend putting .ui-dialog .tab-content .charsheet in front of all of your styling for your character sheet itself.

The code above with your original html (added "test Text" to the contents of the checkbox-text span) produces this layout:

 


Additionally, depending on what the checkbox-text element is supposed to display, you might actually be able to do this using just a checkbox without the label.

August 15 (5 months ago)
Steve
Pro
Sheet Author

Scott C. Thanks so much man, that really tidies things up. The first couple of things (missing bracket and duped section) were copy/paste errors porting sample code to here, but the other fixes now have the checkbox/shield working on click and resizing as it should.

The text overlay says 'Equipped' when the box is checked (somehow deleted that bit of text, restored now).

Only problem is now the "shieldhidden" section is not responding to the checkbox anymore, is the checkboxes value somehow isolated to within the container?


August 15 (5 months ago)
Scott C.
Forum Champion
Sheet Author
API Scripter
Compendium Curator

Yes, you'll need a duplicate of the checkbox outside of the container.

Alternatively, you could use the just a checkbox method and have it control the display of the text and the shield hidden stuff.

August 15 (5 months ago)
Steve
Pro
Sheet Author

Okay, so count me interested in the just a checkbox method. What do I need to do for that to happen?

In going down this rabbit-warren I was under the impression that the approach above was just about the only way to have a 'customised' checkbox.

August 16 (5 months ago)
Scott C.
Forum Champion
Sheet Author
API Scripter
Compendium Curator

You can style checkboxes as if they were divs by applying appearance: none to the checkbox, then you can give it a background image, border, etc just like any other element.