ChatGPT解决这个技术问题 Extra ChatGPT

how to make a whole row in a table clickable as a link?

I'm using Bootstrap and the following doesn't work:

<tbody>
    <a href="#">
        <tr>
            <td>Blah Blah</td>
            <td>1234567</td>
            <td>£158,000</td>
        </tr>
    </a>
</tbody>
JavaScript might be better for this
He's using bootstrap so I think he should be able to do it :)
How about putting a transparent A element before the row? That way it would have all the properties of a real link (status bar, middle click etc...) and also not change the look of the table (if that is desired).

C
Community

Author's note I:

Please look at other answers below, especially ones that do not use jquery.

Author's note II:

Preserved for posterity but surely the wrong approach in 2020. (Was non idiomatic even back in 2017)

Original Answer

You are using Bootstrap which means you are using jQuery :^), so one way to do it is:

<tbody>
    <tr class='clickable-row' data-href='url://'>
        <td>Blah Blah</td> <td>1234567</td> <td>£158,000</td>
    </tr>
</tbody>


jQuery(document).ready(function($) {
    $(".clickable-row").click(function() {
        window.location = $(this).data("href");
    });
});

Of course you don't have to use href or switch locations, you can do whatever you like in the click handler function. Read up on jQuery and how to write handlers;

Advantage of using a class over id is that you can apply the solution to multiple rows:

<tbody>
    <tr class='clickable-row' data-href='url://link-for-first-row/'>
        <td>Blah Blah</td> <td>1234567</td> <td>£158,000</td>
    </tr>
    <tr class='clickable-row' data-href='url://some-other-link/'>
        <td>More money</td> <td>1234567</td> <td>£800,000</td>
    </tr>
</tbody>

and your code base doesn't change. The same handler would take care of all the rows.

Another option

You can use Bootstrap jQuery callbacks like this (in a document.ready callback):

$("#container").on('click-row.bs.table', function (e, row, $element) {
    window.location = $element.data('href');
});

This has the advantage of not being reset upon table sorting (which happens with the other option).

Note

Since this was posted window.document.location is obsolete (or deprecated at the very least) use window.location instead.


You may want to also change the cursor with the css: tr.clickableRow { cursor: pointer; }
Just wanted to add that you shouldn't use href attribute on a tr, since it's not a valid attribute for this element. Use data attributes instead: data-url="{linkurl}" and in js code: $(this).data('url')
You can avoid using a class on each row, and decorate the table with a class name of clickable_row (CamelCase is against HTML standards and should be lower case (I had issues with cross browser a few times with this)) and then the jQuery is $('.clickable_row tr').click(function ... But really you should be using data- to hold data, instead of href since that is the specification for custom data. data-url and jquery will access it like $(this).data(url);
FYI: If you can, you should leave a link somewhere in the tr for people who have Javascript disabled.
Not a big fan of this solution because users don't see the link URL at the bottom of the browser when hovering, and more importantly can't wheel-click on the link to open it in a new tab. Inserting an a tag with display: block; inside each cell is a bit annoying but it seems like the most usable way to solve this issue.
D
Davy de Vries

You can't do that. It is invalid HTML. You can't put a <a> in between a <tbody> and a <tr>. Try this instead:

<tr onclick="window.location='#';">
   ...
</tr>

add style for pointer view

[data-href] { cursor: pointer; }

When you work up to it, you'd want to use JavaScript to assign the click handler outside the HTML.


@Aquillo the OP specifically states that s/he's using bootstrap. so a jQuery solution would be better within the framework
@AhmedMasud In this case, yes. Though for any future readers plain JS might be a better solution. Since that's the purpose of Stackoverflow, I still consider my comment to be a valid one.
The separation of behavior and markup is certainly to be preferred. If anything, I would say that href is not a valid attribute on a tr. It should be replaced with a data-href, and replace $(this).attr with $(this).data.
Or you could make the ".clickableRow" class event binding approach look for the first <a> inside the clicked row. (that would also degrade gracefully)
If you're using ES6 syntax, you can execute the link click like so: onClick={() => window.open('https://stackoverflow.com/', '_blank')}
C
Community

You could include an anchor inside every <td>, like so:

<tr>
  <td><a href="#">Blah Blah</a></td>
  <td><a href="#">1234567</a></td>
  <td><a href="#">more text</a></td>
</tr>

You could then use display:block; on the anchors to make the full row clickable.

tr:hover { 
   background: red; 
}
td a { 
   display: block; 
   border: 1px solid black;
   padding: 16px; 
}

Example jsFiddle here.

This is probably about as optimum as you're going to get it unless you resort to JavaScript.


This is the best solution, no jquery no javascript only CSS, i.e. my link could be different not href, with multiple parameters like so I keep it and change only the style. Many thanks.
@daniel good answer but If the cell extends down, doesn't work properly like this jsfiddle.net/RFFy9/253
Great! This solution enables open the link in a new tab if the user wants. But empty columns will not have links.
@AnnaRouben I'm going to have to disagree: having multiple similar links is a very small downside compared to the upsides (works without JS, screen reader can read where the link is leading to, can ctrl-click to open in new tab...) Pure-HTML solutions work almost always better in assistive software than JS solutions.
I like this solution because of how accessible it is. Users can tab through the links, plus you can open links in new tabs. What I've done in my app is put tabindex="-1" on all but the first <td> link, so the user tabs row to row, not cell to cell. Plus, you can still highlight/outline the entire row if you want using the :focus-within CSS pseudo-class.
m
mfluehr

A linked table row is possible, but not with the standard <table> elements. You can do it using the display: table style properties. Here and here are some fiddles to demonstrate.

This code should do the trick:

.table { display: table; } .row { display: table-row; } .cell { display: table-cell; padding: 10px; } .row:hover { background-color: #cccccc; } .cell:hover { background-color: #e5e5e5; }

1.1
1.2
1.3
2.1
2.2
2.3

Note that ARIA roles are needed to ensure proper accessibility since the standard <table> elements are not used. You may need to add additional roles like role="columnheader" if applicable. Find out more at the guide here.


Now try to apply dataTables plugin to this one! :)
This is the correct answer IMHO, because it retains all user agent tools related to links: search (eg. / key in Firefox), navigation (tab key), Copy link location, Open in new tab/window, etc. even for accessible / special needs browsers. My only nitpick is that I wouldn't nest a <div> inside an <a>, so the cells would be better marked up as <span>.
Applying third-party plugins and libraries to such CSS tables might involve a bit more work, but it's usually copy-paste and replace. For example, I just duplicated Bootstrap's own _tables.scss and a few other random pieces, replacing all occurrences of table elements with the CSS classes I chose to use (which are simply the same as the elements: th.th) Not much work compared to the benefits, really.
Nice answer, but this is only a solution if the table was just used for proper aligning (which is the case in many situations!). For tabular data, the semantic <table> is there for a reason. But this still beats any javascript solution :)
Although this is a very clever work-around, it is just that. It is not actually a table and so, as the author mentions, will present problems to accessibility. It will very likely cause issues with other things that are expecting a table but are getting one.
B
Břetislav Hájek

Achieved using standard Bootstrap 4.3+ as follows - no jQuery nor any extra css classes needed!

The key is to use stretched-link on the text within the cell and defining <tr> as a containing block.

1 Mark Otto @mdo
2 Jacob Thornton @fat
3 Larry the Bird @twitter

You can define containing block in different ways for example setting transform to not none value (as in example above).

For more information read here's the Bootstrap documentation for stretched-link.


This answer deserves more vote. No javascript at all. And what is that scope="row" for? I don't use scope and the row still behaves as a link.
The downside of this is because the link will be on top of the columns, you can't copy the text, and attribute title in td is not showing if hovered by mouse. So unfortunately, I have to go back to old way, put the link only to one column.
I liked this solution, but sadly, it is not cross-browser (Chrome/Firefox are OK but not Safari). The problem is due to tr tag that is not recognized as the rules for the containing block among browsers... stackoverflow.com/questions/61342248/…
Yes, this does seem to break table links in Safari – the hover behavior goes completely haywire. This comment on #28608 seems to indicate that it's not possible to get the desired behavior on tables with stretched-link.
This doesn't seem to be working (Firefox 78.7.0, Chrome 89.0.4389.114), all rows would point to the same href from the last row
A
AbdelElrafa

A much more flexible solution is to target anything with the data-href attribute. This was you can reuse the code easily in different places.

<tbody>
    <tr data-href="https://google.com">
        <td>Col 1</td>
        <td>Col 2</td>
    </tr>
</tbody>

Then in your jQuery just target any element with that attribute:

jQuery(document).ready(function($) {
    $('*[data-href]').on('click', function() {
        window.location = $(this).data("href");
    });
});

And don't forget to style your css:

[data-href] {
    cursor: pointer;
}

Now you can add the data-href attribute to any element and it will work. When I write snippets like this I like them to be flexible. Feel free to add a vanilla js solution to this if you have one.


M
Murad

Here is simple solution..

<tr style='cursor: pointer; cursor: hand;' onclick="window.location='google.com';"></tr>

p
passatgt

One solution that was not mentioned earlier is to use a single link in a cell and some CSS to extend this link over the cells:

table { border: 1px solid; width: 400px; overflow: hidden; } tr:hover { background: gray; } tr td { border: 1px solid; } tr td:first-child { position:relative; } a:before { content: ''; position:absolute; left: 0; top: 0; bottom: 0; display: block; width: 400px; }

First column Second column Third column
First column Second column Third column


This looks rather elegant. BUT the google link demo does not work. Stackoverflow one does. Probably just Google being arses as they often are. Maybe link to Bing :-) Serves them right. Anyhow v nice and tidy solution.
This works me!! Nice solution. One thing to note is that the width for a:before has to be long enough to cover the table.
Great, but my table doesn't have a fixed width. So I just made the anchor really wide -- 4000px. Voila!
@BrettDonald you could use 100vh as the width, so its the same width as the viewport :)
Thanks @passatgt, I came to the same conclusion last week. 4000px had some unwanted side effects, but 100vw (not 100vh) works beautifully.
p
phlegx

You can use this bootstrap component:

http://jasny.github.io/bootstrap/javascript/#rowlink

Jasny Bootstrap

The missing components for your favorite front-end framework.

<table class="table table-striped table-bordered table-hover">
  <thead>
    <tr><th>Name</th><th>Description</th><th>Actions</th></tr>
  </thead>
  <tbody data-link="row" class="rowlink">
    <tr><td><a href="#inputmask">Input mask</a></td><td>Input masks can be used to force the user to enter data conform a specific format.</td><td class="rowlink-skip"><a href="#">Action</a></td></tr>
    <tr><td><a href="http://www.jasny.net/" target="_blank">jasny.net</a></td><td>Shared knowledge of Arnold Daniels aka Jasny.</td><td class="rowlink-skip"><a href="#">Action</a></td></tr>
    <tr><td><a href="#rowlinkModal" data-toggle="modal">Launch modal</a></td><td>Toggle a modal via JavaScript by clicking this row.</td><td class="rowlink-skip"><a href="#">Action</a></td></tr>
  </tbody>
</table>

Usage

Via data attributes

Add class .rowlink and attribute data-link="row" to a <table> or <tbody> element. For other options append the name to data-, as in data-target="a.mainlink" A cell can be excluded by adding the .rowlink-skip class to the <td>.

Via JavaScript

Call the input mask via javascript:

$('tbody.rowlink').rowlink()

T
Todd Skelton

You can add the button role to a table row and Bootstrap will change the cursor without any css changes. I decided to use that role as a way to easily make any row clickable with very little code.

Html

<table class="table table-striped table-hover">
     <tbody>
          <tr role="button" data-href="#">
               <td>Cell 1</td>
               <td>Cell 2</td>
               <td>Cell 3</td>
          </tr>
     </tbody>
</table>

jQuery

$(function(){
     $(".table").on("click", "tr[role=\"button\"]", function (e) {
          window.location = $(this).data("href");
     });
});

You can apply this same principle to add the button role to any tag.


wow it actually worked with ui-sref instead data-href without the jquery code. need to just change cursor to finger.
R
Ronen

There is a nice way to technically do it with <a> tag inside <tr>, which might be semantically incorrect (might give you a browser warning), but will work with no JavaScript/jQuery required:

<!-- HTML -->
<tbody>
  <tr class="bs-table-row">
     <td>Blah Blah</td>
     <td>1234567</td>
     <td>£158,000</td>
     <a class="bs-row-link" href="/your-link-here"></a>
  </tr>
</tbody>

/* CSS */
.bs-table-row {
  position: 'relative';
}

.bs-row-link {
  position: 'absolute';
  left: 0;
  height: '36px'; // or different row height based on your requirements
  width: '100%';
  cursor: 'pointer';
}

PS: Notice the trick here is to put <a> tag as last element, otherwise it will try to take the space of the first <td> cell.

PPS: Now your entire row will be clickable and you can use this link to open in new tab as well (Ctrl/CMD+click)


Isn't this basically the same as my answer ?
While this answers looks perfect, it doesn't actually work, because <tr> elements cannot have an <a> element as a child. Only <td> and <th> are valid children. See: stackoverflow.com/questions/12634715/…
W
Wendell Pereira

A very easy option is just use on-click, and more correct, in my point of view, because this separate the view and controller, and you don't need to hard code the URL or whatever more you need do accomplish with the click. It works with Angular ng-click too.

<table>
  <tr onclick="myFunction(this)">
    <td>Click to show rowIndex</td>
  </tr>
</table>

<script>
function myFunction(x) {
    alert("Row index is: " + x.rowIndex);
}
</script>

Exemple working here


Surely this is the correct answer. The others are crazily complex for such a simple thing. If you want to change the mouse pointer you can add tr:hover { cursor: pointer; } to your CSS class
@TomClose Depending on what you want the row click to do, this can be problematic as e.g. navigation won't work normally - no shift clicks for open in new tab/window, no hover to see destination, etc.
A
Anahit Serobyan

You can use onclick javascript method in tr and make it clickable, also if you need to build your link due to some details you can declare a function in javascript and call it in onclick, passing some values.


D
David Balažic

Here is a way by putting a transparent A element over the table rows. Advantages are:

is a real link element: on hover changes pointer, shows target link in status bar, can be keyboard navigated, can be opened in new tab or window, the URL can be copied, etc

the table looks the same as without the link added

no changes in table code itself

Disadvantages:

size of the A element must be set in a script, both on creation and after any changes to the size of the row it covers (otherwise it could be done with no JavaScript at all, which is still possible if the table size is also set in HTML or CSS)

The table stays as is:

<table id="tab1">
<tbody>
        <tr>
            <td>Blah Blah</td>
            <td>1234567</td>
            <td>£158,000</td>
        </tr>
</tbody>
</table>

Add the links (for each row) via jQuery JavaScript by inserting an A element into each first column and setting the needed properties:

// v1, fixed for IE and Chrome
// v0, worked on Firefox only
// width needed for adjusting the width of the A element
var mywidth=$('#tab1').width();

$('#tab1 tbody>tr>td:nth-child(1)').each(function(index){
    $(this).css('position',  'relative');// debug: .css('background-color', '#f11');
    // insert the <A> element
    var myA=$('<A>');
    $(this).append(myA);
    var myheight=$(this).height();

    myA.css({//"border":'1px solid #2dd', border for debug only
            'display': 'block',
            'left': '0',
            'top': '0',
            'position':  'absolute'
        })
        .attr('href','the link here')
        .width(mywidth)
        .height(myheight)
        ;
    });

The width and height setting can be tricky, if many paddings and margins are used, but in general a few pixels off should not even matter.

Live demo here: http://jsfiddle.net/qo0dx0oo/ (works in Firefox, but not IE or Chrome, there the link is positioned wrong)

Fixed for Chrome and IE (still works in FF too): http://jsfiddle.net/qo0dx0oo/2/


Clever solution. I found creating the element lazily, only once the row is hovered, brought huge performance boost for big tables.
@levi can you give some feedback on how to lazyload this?
@KemalEmin I would assume that he would use javascript's onmouseover
A
Aydan Aleydin

This code bellow will make your whole table clickable. Clicking the links in this example will show the link in an alert dialog instead of following the link.

The HTML:

Here's the HTML behind the above example:

    <table id="example">
    <tr>
     <th>&nbsp;</th>
     <th>Name</th>
     <th>Description</th>
     <th>Price</th>
   </tr>
   <tr>
     <td><a href="apples">Edit</a></td>
     <td>Apples</td>
     <td>Blah blah blah blah</td>
     <td>10.23</td>
   </tr>
    <tr>
     <td><a href="bananas">Edit</a></td>
     <td>Bananas</td>
     <td>Blah blah blah blah</td>
     <td>11.45</td>
   </tr>
   <tr>
     <td><a href="oranges">Edit</a></td>
     <td>Oranges</td>
     <td>Blah blah blah blah</td>
     <td>12.56</td>
   </tr>
    </table>

The CSS

And the CSS:

    table#example {
    border-collapse: collapse;   
}
#example tr {
    background-color: #eee;
    border-top: 1px solid #fff;
}
#example tr:hover {
    background-color: #ccc;
}
#example th {
    background-color: #fff;
}
#example th, #example td {
    padding: 3px 5px;
}
#example td:hover {
    cursor: pointer;
}

The jQuery

And finally the jQuery which makes the magic happen:

    $(document).ready(function() {

    $('#example tr').click(function() {
        var href = $(this).find("a").attr("href");
        if(href) {
            window.location = href;
        }
    });

});

What it does is when a row is clicked, a search is done for the href belonging to an anchor. If one is found, the window's location is set to that href.


S
Sidupac

I know someone has written pretty much the same already, however my way is the correct way (div cannot be child of A) and also it's better to use classes.

You can imitate a table using CSS and make an A element the row

<div class="table" style="width:100%;">
  <a href="#" class="tr">
    <span class="td">
      cell 1
    </span>
    <span class="td">
      cell 2
    </span>
  </a>
</div>

css:

.table{display:table;}
.tr{display:table-row;}
.td{display:table-cell;}
.tr:hover{background-color:#ccc;}

A
Ali Raza

i would prefer to use onclick="" attribute as it is easy to use and understand for newbie like

 <tr onclick="window.location='any-page.php'">
    <td>UserName</td>
    <td>Email</td>
    <td>Address</td>
</tr>

R
Y
Yisela

Another option using an <a>, CSS positions and some jQuery or JS:

HTML:

<table>
<tr>
    <td>
        <span>1</span>
        <a href="#" class="rowLink"></a>
    </td>
    <td><span>2</span></td>
</tr>
</table>

CSS:

table tr td:first-child {
    position: relative;
}
a.rowLink {
    position: absolute;
    top: 0; left: 0;
    height: 30px;
}
a.rowLink:hover {
    background-color: #0679a6;
    opacity: 0.1;
}

Then you need to give the a width, using for example jQuery:

    $(function () {
        var $table = $('table');
            $links = $table.find('a.rowLink');

        $(window).resize(function () {
            $links.width($table.width());
        });

        $(window).trigger('resize');
    });

T
Thomas Williams

The accepted answer is great, but I propose a small alternative if you don't want to repeat the url on every tr. So you put the url or href in the table data-url and not the tr.

<table data-click data-url="href://blah">    
    <tbody>
        <tr id ="2">
            <td>Blah Blah</td> <td>1234567</td> <td>£158,000</td>
        </tr>
        <tr id ="3">
            <td>Blah Blah</td> <td>1234567</td> <td>£158,000</td>
        </tr>
    </tbody>
    </table

jQuery(document).ready(function($) {
    $('[data-click] tbody tr').click(function() {
        var url = $(this).closest('table').data("url");
        var id = $(this).closest('tr').attr('id');
        window.location = url+"?id="+id);
    });
});

This is also good because you don't need to add the click data attribute to every tr either. The other good thing is that we are not using a class to trigger a click as classes should only really be used for styling.


g
gamliela

I've invested a lot of time trying to solve this problem.

There are 3 approaches:

Use JavaScript. The clear drawbacks: it's not possible to open a new tab natively, and when hovering over the row there will be no indication on status bar like regular links have. Accessibility is also a question. Use HTML/CSS only. This means putting nested under each . A simple approach like this fiddle doesn't work - Because the clickable surface is not necessarily equal for each column. This is a serious UX concern. Also, if you need a

C
Christopher Vickers
<tbody>
    <tr data-href='www.bbc.co.uk'>
        <td>Blah Blah</td>
        <td>1234567</td>
        <td>£158,000</td>
    </tr>
    <tr data-href='www.google.com'>
        <td>Blah Blah</td>
        <td>1234567</td>
        <td>£158,000</td>
    </tr>
</tbody>

<script>
    jQuery(document).ready(function ($) {
        $('[data-href]').click(function () {
            window.location = $(this).data("href");
        });
    });
</script>

Whilst the main solution on here is great, my solution removes the need for classes. All you need to do is add the data-href attribute with the URL in it.


k
klewis

Here is another way...

The HTML:

<table>
<tbody>
       <tr class='clickableRow'>
       <td>Blah Blah</td>
       <td>1234567</td>
       <td>£158,000</td>
        </tr>
</tbody>
</table>

The jQuery:

$(function() {
      $(".clickableRow").on("click", function() {
          location.href="http://google.com";

      });

});

How can we open it in new tab? @klewis
s
stevec

Here's a generic approach. Define this css:

// css
td a.linker {
    color:#212529;
    display: block;
    padding: 16px;
    text-decoration: none;
}

Then place this inside each td:

<td>
  <a class="linker" href="www.google.com"> 
    Cell content goes here 
  </a>
</td>

P
Peter Berg

You could give the row an id, e.g.

<tr id="special"> ... </tr>

and then use jquery to say something like:

$('#special').onclick(function(){ window="http://urltolinkto.com/x/y/z";})


I believe you want to use class instead of id because you're most likely going to be having multiple rows and id should only be used for one item.
R
Ramesh J

Why should we don't use "div" tags....

<div>

  <a href="" >     <div>  Table row  of content here..  </div>    </a>

</div>

you should improve your answer with css display:table
Improve answer with adding css