mirror of
https://github.com/kennethreitz/kjvstudy.org.git
synced 2026-06-05 23:00:16 +00:00
3336863a4d
- Add KJVNav.initGridNav for standardized 2D grid navigation - Migrate books.html, topics.html, resources.html to use initGridNav - Add sidebarActive check to all templates with custom keyboard handlers - Add [ and ] shortcuts for prev/next chapter on chapter pages - Add [ and ] shortcuts for prev/next book on book pages - Update accessibility page with comprehensive keyboard shortcut docs - Add honest note about keyboard navigation complexity - Fix sidebar nav conflicting with main content selection 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
394 lines
10 KiB
HTML
394 lines
10 KiB
HTML
{% extends "base.html" %}
|
||
|
||
{% block title %}Strong's Concordance - KJV Study{% endblock %}
|
||
|
||
{% block description %}Search Strong's Exhaustive Concordance. Look up Hebrew and Greek word definitions, etymology, and KJV usage.{% endblock %}
|
||
|
||
{% block head %}
|
||
<style>
|
||
.search-form {
|
||
max-width: 600px;
|
||
margin: 2rem 0;
|
||
}
|
||
|
||
.search-input-group {
|
||
display: flex;
|
||
gap: 0.5rem;
|
||
}
|
||
|
||
.search-input {
|
||
flex: 1;
|
||
padding: 0.75rem 1rem;
|
||
font-size: 1.1rem;
|
||
border: 1px solid var(--border-color-darker);
|
||
border-radius: 4px;
|
||
background: var(--bg-color);
|
||
color: var(--text-color);
|
||
}
|
||
|
||
.search-input:focus {
|
||
outline: none;
|
||
border-color: var(--link-color);
|
||
}
|
||
|
||
.search-btn {
|
||
padding: 0.75rem 1.5rem;
|
||
font-size: 1rem;
|
||
background: var(--link-color);
|
||
color: #fff;
|
||
border: none;
|
||
border-radius: 4px;
|
||
cursor: pointer;
|
||
font-weight: 600;
|
||
}
|
||
|
||
.search-btn:hover {
|
||
background: var(--link-hover);
|
||
}
|
||
|
||
.results-count {
|
||
color: var(--text-secondary);
|
||
margin-bottom: 1rem;
|
||
}
|
||
|
||
.results-list {
|
||
list-style: none;
|
||
padding: 0;
|
||
margin: 0;
|
||
}
|
||
|
||
.result-item {
|
||
padding: 1rem;
|
||
border-bottom: 1px solid var(--border-color);
|
||
}
|
||
|
||
.result-item:last-child {
|
||
border-bottom: none;
|
||
}
|
||
|
||
.result-header {
|
||
display: flex;
|
||
align-items: baseline;
|
||
gap: 0.75rem;
|
||
margin-bottom: 0.5rem;
|
||
}
|
||
|
||
.result-strongs {
|
||
font-family: monospace;
|
||
font-size: 0.9rem;
|
||
color: var(--link-color);
|
||
text-decoration: none;
|
||
font-weight: 600;
|
||
}
|
||
|
||
.result-strongs:hover {
|
||
text-decoration: underline;
|
||
}
|
||
|
||
.result-word {
|
||
font-size: 1.4rem;
|
||
}
|
||
|
||
.result-word.hebrew {
|
||
direction: rtl;
|
||
font-family: "SBL Hebrew", "Ezra SIL", "Times New Roman", serif;
|
||
}
|
||
|
||
.result-word.greek {
|
||
font-family: "SBL Greek", "Gentium Plus", "Times New Roman", serif;
|
||
}
|
||
|
||
.result-lang {
|
||
font-size: 0.75rem;
|
||
padding: 0.15rem 0.4rem;
|
||
border-radius: 3px;
|
||
text-transform: uppercase;
|
||
letter-spacing: 0.03em;
|
||
font-weight: 600;
|
||
}
|
||
|
||
.result-lang.hebrew {
|
||
background: #8B4513;
|
||
color: #fff;
|
||
}
|
||
|
||
.result-lang.greek {
|
||
background: #4169E1;
|
||
color: #fff;
|
||
}
|
||
|
||
.result-definition {
|
||
color: var(--text-color);
|
||
margin-bottom: 0.25rem;
|
||
}
|
||
|
||
.result-kjv {
|
||
font-size: 0.9rem;
|
||
color: var(--text-secondary);
|
||
}
|
||
|
||
.quick-lookup {
|
||
margin-top: 2rem;
|
||
padding-top: 2rem;
|
||
border-top: 1px solid var(--border-color);
|
||
}
|
||
|
||
.quick-lookup h3 {
|
||
font-size: 1rem;
|
||
margin-bottom: 0.75rem;
|
||
}
|
||
|
||
.quick-links {
|
||
display: flex;
|
||
flex-wrap: wrap;
|
||
gap: 0.5rem;
|
||
}
|
||
|
||
.quick-link {
|
||
padding: 0.35rem 0.7rem;
|
||
font-size: 0.85rem;
|
||
background: var(--code-bg);
|
||
border: 1px solid var(--border-color);
|
||
border-radius: 4px;
|
||
text-decoration: none;
|
||
color: var(--text-color);
|
||
}
|
||
|
||
.quick-link:hover {
|
||
border-color: var(--link-color);
|
||
color: var(--link-color);
|
||
}
|
||
|
||
/* Browse language cards */
|
||
.browse-section {
|
||
margin: 2rem 0;
|
||
}
|
||
|
||
.browse-section h3 {
|
||
font-size: 1rem;
|
||
margin-bottom: 1rem;
|
||
}
|
||
|
||
.browse-cards {
|
||
display: grid;
|
||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||
gap: 1rem;
|
||
}
|
||
|
||
.browse-card {
|
||
background: var(--bg-color);
|
||
border: 1px solid var(--border-color);
|
||
border-radius: 8px;
|
||
padding: 1.5rem;
|
||
text-decoration: none;
|
||
color: var(--text-color);
|
||
transition: all 0.2s;
|
||
}
|
||
|
||
.browse-card:hover {
|
||
border-color: var(--link-color);
|
||
transform: translateY(-2px);
|
||
box-shadow: 0 4px 12px rgba(0,0,0,0.1);
|
||
}
|
||
|
||
.browse-card .lang-title {
|
||
font-size: 1.2rem;
|
||
font-weight: 600;
|
||
margin-bottom: 0.25rem;
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 0.5rem;
|
||
}
|
||
|
||
.browse-card .lang-badge {
|
||
font-size: 0.65rem;
|
||
padding: 0.2rem 0.5rem;
|
||
border-radius: 3px;
|
||
text-transform: uppercase;
|
||
letter-spacing: 0.03em;
|
||
font-weight: 600;
|
||
}
|
||
|
||
.browse-card .lang-badge.hebrew {
|
||
background: #8B4513;
|
||
color: #fff;
|
||
}
|
||
|
||
.browse-card .lang-badge.greek {
|
||
background: #4169E1;
|
||
color: #fff;
|
||
}
|
||
|
||
.browse-card .lang-count {
|
||
font-size: 0.9rem;
|
||
color: var(--text-secondary);
|
||
margin-bottom: 0.5rem;
|
||
}
|
||
|
||
.browse-card .lang-sample {
|
||
font-size: 1.5rem;
|
||
color: var(--text-secondary);
|
||
}
|
||
|
||
.browse-card .lang-sample.hebrew {
|
||
direction: rtl;
|
||
font-family: "SBL Hebrew", "Ezra SIL", "Times New Roman", serif;
|
||
}
|
||
|
||
.browse-card .lang-sample.greek {
|
||
font-family: "SBL Greek", "Gentium Plus", "Times New Roman", serif;
|
||
}
|
||
|
||
[data-theme="dark"] .browse-card {
|
||
background: #1a1a1a;
|
||
border-color: #333;
|
||
}
|
||
|
||
[data-theme="dark"] .browse-card:hover {
|
||
border-color: var(--link-color);
|
||
box-shadow: 0 4px 12px rgba(0,0,0,0.3);
|
||
}
|
||
</style>
|
||
{% endblock %}
|
||
|
||
{% block content %}
|
||
<h1>Strong's Concordance</h1>
|
||
<p class="subtitle">Hebrew & Greek Word Definitions</p>
|
||
|
||
<form class="search-form" action="/strongs" method="get">
|
||
<div class="search-input-group">
|
||
<input type="text" name="q" class="search-input" placeholder="Search by definition (e.g., love, faith, grace)" value="{{ query }}" {% if not query %}autofocus{% endif %}>
|
||
<button type="submit" class="search-btn">Search</button>
|
||
</div>
|
||
</form>
|
||
|
||
{% if query %}
|
||
<p class="results-count">{{ results|length }} result{% if results|length != 1 %}s{% endif %} for "{{ query }}"</p>
|
||
|
||
{% if results %}
|
||
<ul class="results-list">
|
||
{% for result in results %}
|
||
<li class="result-item">
|
||
<div class="result-header">
|
||
<a href="/strongs/{{ result.strongs }}" class="result-strongs">{{ result.strongs }}</a>
|
||
<span class="result-word {% if result.language == 'Hebrew' %}hebrew{% else %}greek{% endif %}">{{ result.word }}</span>
|
||
<span class="result-lang {% if result.language == 'Hebrew' %}hebrew{% else %}greek{% endif %}">{{ result.language }}</span>
|
||
</div>
|
||
<p class="result-definition">{{ result.definition }}</p>
|
||
{% if result.kjv_usage %}
|
||
<p class="result-kjv"><strong>KJV:</strong> {{ result.kjv_usage }}</p>
|
||
{% endif %}
|
||
</li>
|
||
{% endfor %}
|
||
</ul>
|
||
{% else %}
|
||
<p>No results found for "{{ query }}".</p>
|
||
{% endif %}
|
||
|
||
{% else %}
|
||
<div class="browse-section">
|
||
<h3>Browse All Entries</h3>
|
||
<div class="browse-cards">
|
||
<a href="/strongs/hebrew" class="browse-card">
|
||
<div class="lang-title">
|
||
Hebrew
|
||
<span class="lang-badge hebrew">OT</span>
|
||
</div>
|
||
<div class="lang-count">8,674 entries (H1–H8674)</div>
|
||
<div class="lang-sample hebrew">אָב אֱלֹהִים שָׁלוֹם</div>
|
||
</a>
|
||
<a href="/strongs/greek" class="browse-card">
|
||
<div class="lang-title">
|
||
Greek
|
||
<span class="lang-badge greek">NT</span>
|
||
</div>
|
||
<div class="lang-count">5,523 entries (G1–G5624)</div>
|
||
<div class="lang-sample greek">ἀγάπη πίστις χάρις</div>
|
||
</a>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="quick-lookup">
|
||
<h3>Quick Lookup - Common Words</h3>
|
||
<div class="quick-links">
|
||
<a href="/strongs/G26" class="quick-link">G26 - agape (love)</a>
|
||
<a href="/strongs/G4102" class="quick-link">G4102 - pistis (faith)</a>
|
||
<a href="/strongs/G5485" class="quick-link">G5485 - charis (grace)</a>
|
||
<a href="/strongs/G3056" class="quick-link">G3056 - logos (word)</a>
|
||
<a href="/strongs/G2316" class="quick-link">G2316 - theos (God)</a>
|
||
<a href="/strongs/G5547" class="quick-link">G5547 - Christos (Christ)</a>
|
||
<a href="/strongs/H430" class="quick-link">H430 - Elohim (God)</a>
|
||
<a href="/strongs/H3068" class="quick-link">H3068 - YHWH (LORD)</a>
|
||
<a href="/strongs/H7965" class="quick-link">H7965 - shalom (peace)</a>
|
||
<a href="/strongs/H2617" class="quick-link">H2617 - chesed (mercy)</a>
|
||
</div>
|
||
</div>
|
||
{% endif %}
|
||
|
||
<script>
|
||
(function() {
|
||
// For search results page
|
||
const resultItems = Array.from(document.querySelectorAll('.result-item'));
|
||
// For browse page - cards and quick links
|
||
const browseCards = Array.from(document.querySelectorAll('.browse-card'));
|
||
const quickLinks = Array.from(document.querySelectorAll('.quick-link'));
|
||
|
||
const allItems = resultItems.length > 0 ? resultItems : browseCards.concat(quickLinks);
|
||
if (allItems.length === 0) return;
|
||
|
||
let selectedIndex = -1;
|
||
|
||
function selectItem(index) {
|
||
if (selectedIndex >= 0 && selectedIndex < allItems.length) {
|
||
allItems[selectedIndex].style.outline = '';
|
||
allItems[selectedIndex].style.outlineOffset = '';
|
||
allItems[selectedIndex].classList.remove('selected');
|
||
}
|
||
selectedIndex = Math.max(0, Math.min(index, allItems.length - 1));
|
||
allItems[selectedIndex].style.outline = '2px solid #4a7c59';
|
||
allItems[selectedIndex].style.outlineOffset = '2px';
|
||
allItems[selectedIndex].classList.add('selected');
|
||
allItems[selectedIndex].scrollIntoView({ behavior: 'auto', block: 'center' });
|
||
}
|
||
|
||
document.addEventListener('keydown', function(e) {
|
||
if (e.target.tagName === 'INPUT' || e.target.tagName === 'TEXTAREA') return;
|
||
if (KJVNav.sidebarActive) return;
|
||
|
||
if (e.key === 'ArrowDown' || e.key === 'j') {
|
||
e.preventDefault();
|
||
if (KJVNav.isSelectionOffScreen(allItems, selectedIndex)) {
|
||
selectItem(KJVNav.findFirstVisibleIndex(allItems));
|
||
} else {
|
||
selectItem(selectedIndex < 0 ? 0 : selectedIndex + 1);
|
||
}
|
||
} else if (e.key === 'ArrowUp' || e.key === 'k') {
|
||
e.preventDefault();
|
||
if (KJVNav.isSelectionOffScreen(allItems, selectedIndex)) {
|
||
selectItem(KJVNav.findFirstVisibleIndex(allItems));
|
||
} else if (selectedIndex <= 0) {
|
||
selectItem(0);
|
||
} else {
|
||
selectItem(selectedIndex - 1);
|
||
}
|
||
} else if (e.key === 'ArrowLeft' || e.key === 'h') {
|
||
e.preventDefault();
|
||
history.back();
|
||
} else if (e.key === 'Enter' && selectedIndex >= 0) {
|
||
e.preventDefault();
|
||
const link = allItems[selectedIndex].tagName === 'A' ? allItems[selectedIndex] : allItems[selectedIndex].querySelector('a');
|
||
if (link) window.location.href = link.href;
|
||
} else if (e.key === 'Escape') {
|
||
e.preventDefault();
|
||
if (selectedIndex >= 0 && selectedIndex < allItems.length) {
|
||
allItems[selectedIndex].style.outline = '';
|
||
allItems[selectedIndex].style.outlineOffset = '';
|
||
allItems[selectedIndex].classList.remove('selected');
|
||
}
|
||
selectedIndex = -1;
|
||
}
|
||
});
|
||
})();
|
||
</script>
|
||
{% endblock %}
|