{
"@context": "http://schema.org/",
"@type": "Product",
"name": "Hope RX24 PRO 5 - Rear",
"url": "https:\/\/projektride.co.uk\/products\/hope-rx24-pro-5-rear-j-bend","image": [
"https:\/\/projektride.co.uk\/cdn\/shop\/files\/Screenshot2025-02-01at10.54.24_1880x.png?v=1738407279"
],"description": "\n\n\n\n\nOur new RX24 tubeless ready gravel alloy wheelset allows you to adventure further into the world of gravel and bikepacking, with the comfort of wider tyres and higher system weight than our existing 20FIVE wheels. The 24mm inner rim width allows for compatibility with high volume tyres up to a maximum width of 60mm. The wheels are supplied fully taped with our new tubeless tape and fitted with our brand-new alloy tubeless valves, colour matched to your hub, so they are ready for tyre fitting straight out of the box.\n\n\n\n\n\nKey Features\n\u0026gt;\u0026gt; Available in standard and boost widths built around our Pro 5 hubs \u0026gt;\u0026gt; 700C diameter in 24 hole, 28 hole and 32 hole \u0026gt;\u0026gt; Premium welded rim construction in Al 6069 \u0026gt;\u0026gt; Shot peened and black anodised \u0026gt;\u0026gt; Neutral grey water slide decals \u0026gt;\u0026gt; Brass nipples for long term durability \u0026gt;\u0026gt; Black Sapim Race double butted spokes \u0026gt;\u0026gt; 24mm inner rim width \u0026gt;\u0026gt; 23mm rim height \u0026gt;\u0026gt; Disc brake only \u0026gt;\u0026gt; Recommended maximum system weight: 130kg \u0026gt;\u0026gt; Weight: From 1780g per pair \n\n\n\n\n\n \n\n\n\n\n\n\n\n\n\n","brand": {
"@type": "Thing",
"name": "Hope"
},"offers": [{
"@type" : "Offer","availability" : "http://schema.org/InStock",
"price" : 365.0,
"priceCurrency" : "GBP",
"url" : "https:\/\/projektride.co.uk\/products\/hope-rx24-pro-5-rear-j-bend?variant=55055681913215"
}
]
}
03cbr\u003e\u0026gt;\u0026gt; Premium welded rim construction in Al 6069 \u003cbr\u003e\u0026gt;\u0026gt; Shot peened and black anodised \u003cbr\u003e\u0026gt;\u0026gt; Neutral grey water slide decals \u003cbr\u003e\u0026gt;\u0026gt; Brass nipples for long term durability \u003cbr\u003e\u0026gt;\u0026gt; Black Sapim Race double butted spokes \u003cbr\u003e\u0026gt;\u0026gt; 24mm inner rim width \u003cbr\u003e\u0026gt;\u0026gt; 23mm rim height \u003cbr\u003e\u0026gt;\u0026gt; Disc brake only \u003cbr\u003e\u0026gt;\u0026gt; Recommended maximum system weight: 130kg \u003cbr\u003e\u0026gt;\u0026gt; Weight: From 1780g per pair \u003c\/h5\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003cdiv class=\"three columns\"\u003e\n\u003cdiv class=\"new-download-items\"\u003e\n\u003ch3\u003e\u003cbr\u003e\u003c\/h3\u003e\n\u003ch5\u003e \u003c\/h5\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003c\span class="visually-hidden">Genesis Equilibrium Disc Frameset
available;window._RestockRocketConfig.scriptUrlProduct = 'https://cdn.shopify.com/extensions/019b1405-52e0-7e89-a6c1-1ac7fea6dd8f/restock-rocket-shopify-410/assets/restockrocket-product.js'
window._RestockRocketConfig.scriptUrlCollection = 'https://cdn.shopify.com/extensions/019b1405-52e0-7e89-a6c1-1ac7fea6dd8f/restock-rocket-shopify-410/assets/restockrocket-collection.js'
window._RestockRocketConfig.scriptHost = window._RestockRocketConfig.scriptUrlProduct.substring(0, window._RestockRocketConfig.scriptUrlProduct.lastIndexOf('/') + 1)
window._RestockRocketConfig.host = 'https://app.restockrocket.io'
const SETTINGS_CACHE_DURATION = 15 * 60 * 1000; // 15 minutes in milliseconds
function checkSettingsExpiry(settings) {
try {
if (!settings || !settings.updated_at) {
console.debug('STOQ - Invalid settings data structure');
return null;
}
if (!settings.cache) {
console.debug('STOQ - settings caching disabled');
return null;
}
// Check >
using cache even for untranslated locales
}
const updatedAt = new Date(settings.updated_at);
if (isNaN(updatedAt.getTime())) {
console.debug('STOQ - Invalid updated_at date format in settings');
return null;
}
const age = Date.now() - updatedAt.getTime();
if (age >
SETTINGS_CACHE_DURATION) {
console.debug('STOQ - settings changed recently, skipping cache');
return null;
}
return settings;
} catch (error) {
console.debug('STOQ - Error checking settings cache:', error);
return null;
}
}
function createRestockRocketContainer() {
const restockRocketContainer = document.createElement('div');
restockRocketContainer.id = 'restock-rocket';
document.body.appendChild(restockRocketContainer);
}
function createRestockRocketScript(scriptUrl) {
const restockRocketScriptElement = document.createElement('script');
restockRocketScriptElement.setAttribute('defer', 'defer');
restockRocketScriptElement.src = scriptUrl;
document.body.appendChild(restockRocketScriptElement);
}
createRestockRocketContainer()
console.debug('STOQ - extension activated')
function applyTranslations(settings) {
try {
// Skip translation logic entirely if multi-language is not enabled
if (!settings ||
];
if (value !== null && value !== undefined && value !== '') {
settings[key] = value;
}
});
} else {
console.debug('STOQ - No translated fields found for locale:', normalizedLocale);
}
delete settings.translations;
return settings;
} catch (e) {
console.debug('STOQ - error applying translations:', e);
return settings;
}
}
// First try to get settings from metafields with expiry check
const cachedSettings = window._RestockRocketConfig.cachedSettings;
const validCachedSettings = cachedSettings ? checkSettingsExpiry(cachedSettings) : null;
if (validCachedSettings) {
console.debug('STOQ - us
// If request failed and we have cached settings (even if expired), use them as fallback
if (cachedSettings) {
console.debug('STOQ - using expired cached settings as fallback');
initializeScripts(cachedSettings);
} else {
console.error('STOQ - failed to load settings:', error);
}
})
.catch(function(e) {
console.error(e)
})
}
function initializeScripts(settings) {
settings = applyTranslations(settings);
window._RestockRocketConfig.settings = settings;
console.debug(`STOQ - settings configured for ${window._RestockRocketConfig.pageType}`);
if(settings.enable_app) {
const hijackIntegration = window._RestockRocketConfig.integrations.find(function(integration) {
return integration.type === 'hijack' && integration.enabled && integration.page_types.includes(window._RestockRocketConfig.pageType);
})
if(window._RestockRocketConfig.pageType === 'collection' && (settings.show_button_on_collection || settin
cketConfig.scriptUrlCollection);
} else {
console.debug(`STOQ - no scripts enabled for ${window._RestockRocketConfig.pageType}`);
}
// Check and update cart selling plans after scripts are loaded
if (settings.preorder_enabled) {
updateCartSellingPlans();
}
// Dispatch custom event when app is loaded
const appLoadedEvent = new CustomEvent('stoq:loaded', {
detail: {
pageType: window._RestockRocketConfig.pageType,
enabled: settings.enable_app,
settings: settings
}
});
console.debug('STOQ - dispatching app loaded event');
window.dispatchEvent(appLoadedEvent);
}
}
function updateCartSellingPlans() {
// Wait for the API to be available with retries
const maxRetries = 10;
const retryDelay = 500;
function attemptCartCheck(attempt = 1) {
if (window._RestockRocket && window._RestockRocket.updateCartSellingPlans) {
window._RestockRocket.updateCartSellingPlans
div class="product__media-hover-img product__media" style="background-image: url(//projektride.co.uk/cdn/shop/products/gn21810_detail_2_600x.jpg?v=1709549998)">
n't
if (settings.multi_language_enabled) {
if (!settings.translations) {
// Translations enabled but no translation data in metafield
// Metafield hasn't been backfilled yet - force refresh
console.debug('STOQ - multi-language enabled but no translation data in cache, fetching fresh');
return null;
}
// Translations object exists in metafield - cache is valid
// If current locale isn't translated, applyTranslations will gracefully use default locale from base fields
if (window._RestockRocketConfig.normalizedLocale &&
!Object.prototype.hasOwnProperty.call(settings.translations, window._RestockRocketConfig.normalizedLocale)) {
console.debug('STOQ - locale not explicitly translated, will use default language from cache');
}
// Don't return null - continue using cache even for untranslated locales
}
const updatedAt = new Date(settings.updated_at);
if (isNaN(updatedAt.getT<!settings.multi_language_enabled) {
return settings;
}
if (!settings.translations) {
console.debug('STOQ - No translations found, skipping translation');
return settings;
}
const normalizedLocale = window._RestockRocketConfig.normalizedLocale;
const translations = settings.translations;
if (!normalizedLocale) {
// No matching locale has translations; drop payload to save memory
console.debug('STOQ - No matching locale for translations. Available:', Object.keys(translations || {}));
delete settings.translations;
return settings;
}
console.debug(`STOQ - Applying translations for normalized locale: ${normalizedLocale} (original: ${window._RestockRocketConfig.locale})`);
const translatedFields = translations[normalizedLocale];
if (translatedFields && typeof translatedFields === 'object') {
Object.keys(translatedFields).forEach(function(key) {
const value = translatedFields[key
e?.role === 'main') {
headers['X-Shopify-Theme-Schema-Name'] = window.Shopify.theme.schema_name;
headers['X-Shopify-Theme-Schema-Version'] = window.Shopify.theme.schema_version;
headers['X-Shopify-Theme-Store-Id'] = window.Shopify.theme.theme_store_id;
}
fetch(
`${window._RestockRocketConfig.host}/api/v1/setting.json?translation_locale=${window._RestockRocketConfig.normalizedLocale}`,
{ headers }
)
.then(function(response) {
if (!response.ok) {
throw new Error('Network response was not ok');
}
return response.json();
})
.then(function(settings) {
initializeScripts(settings);
})
.catch(function(error) {
// If request failed and we have cached settings (even if expired), use them as fallback
if (cachedSettings) {
conso
lly');
} else {
console.debug('STOQ - no cart selling plan updates needed');
}
})
.catch(error => {
console.error('STOQ - error updating cart selling plans:', error);
});
} else if (attempt
maxRetries) {
console.debug(`STOQ - API not ready, retry ${attempt}/${maxRetries}`);
setTimeout(() => attemptCartCheck(attempt + 1), retryDelay);
} else {
console.debug('STOQ - API not loaded after max retries, skipping cart selling plan check');
}
}
attemptCartCheck();
}