alse,"selling_plan_allocations":[],"quantity_rule":{"min":1,"max":null,"increment":1}},{"id":43936694993122,"title":"Single Gripshift \/ Black","option1":"Single Gripshift","option2":"Black","option3":null,"sku":"EG-42411397","requires_shipping":true,"taxable":true,"featured_image":{"id":39096928469218,"product_id":8055186358498,"position":1,"created_at":"2023-03-18T19:46:07+00:00","updated_at":"2023-03-18T19:46:10+00:00","alt":null,"width":776,"height":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30366-pm.png?v=1679168770","variant_ids":[43936694993122]},"available":true,"name":"Ergon GA2 - Single Gripshift \/ Black","public_title":"Single Gripshift \/ Black","options":["Single Gripshift","Black"],"price":2799,"weight":0,"compare_at_price":null,"inventory_management":"shopify","barcode":"","featured_media":{"alt":null,"id":31695345352930,"position":1,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30366-pm.png?v=1679168770"}},"requires _selling_plan":false,"selling_plan_allocations":[],"quantity_rule":{"min":1,"max":null,"increment":1}}],"images":["\/\/projektride.co.uk\/cdn\/shop\/products\/30366-pm.png?v=1679168770","\/\/projektride.co.uk\/cdn\/shop\/products\/30367-pm.png?v=1679168771","\/\/projektride.co.uk\/cdn\/shop\/products\/30369-pm.png?v=1679168770","\/\/projektride.co.uk\/cdn\/shop\/products\/30371-pm.png?v=1679168769","\/\/projektride.co.uk\/cdn\/shop\/products\/30373-pm.png?v=1679168770","\/\/projektride.co.uk\/cdn\/shop\/products\/30375-pm.png?v=1679168769","\/\/projektride.co.uk\/cdn\/shop\/products\/30377-pm.png?v=1679168770","\/\/projektride.co.uk\/cdn\/shop\/products\/26509-pm_d0fed167-a486-446a-98ab-ba43f3a40eb4.png?v=1679168963"],"featured_image":"\/\/projektride.co.uk\/cdn\/shop\/products\/30366-pm.png?v=1679168770","options":["Size","Color"],"media":[{"alt":null,"id":31695345352930,"position":1,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30366-pm.png?v
=1679168770"},"aspect_ratio":1.0,"height":776,"media_type":"image","src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30366-pm.png?v=1679168770","width":776},{"alt":null,"id":31695345385698,"position":2,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30367-pm.png?v=1679168771"},"aspect_ratio":1.0,"height":776,"media_type":"image","src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30367-pm.png?v=1679168771","width":776},{"alt":null,"id":31695345418466,"position":3,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30369-pm.png?v=1679168770"},"aspect_ratio":1.0,"height":776,"media_type":"image","src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30369-pm.png?v=1679168770","width":776},{"alt":null,"id":31695345451234,"position":4,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30371-pm.png?v=1679168769"},"aspect_ratio":1.0,"hei ght":776,"media_type":"image","src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30371-pm.png?v=1679168769","width":776},{"alt":null,"id":31695345484002,"position":5,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30373-pm.png?v=1679168770"},"aspect_ratio":1.0,"height":776,"media_type":"image","src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30373-pm.png?v=1679168770","width":776},{"alt":null,"id":31695345516770,"position":6,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30375-pm.png?v=1679168769"},"aspect_ratio":1.0,"height":776,"media_type":"image","src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30375-pm.png?v=1679168769","width":776},{"alt":null,"id":31695345549538,"position":7,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/30377-pm.png?v=1679168770"},"aspect_ratio":1.0,"height":776,"media_type":"image","src":"
\/\/projektride.co.uk\/cdn\/shop\/products\/30377-pm.png?v=1679168770","width":776},{"alt":null,"id":31695353151714,"position":8,"preview_image":{"aspect_ratio":1.0,"height":776,"width":776,"src":"\/\/projektride.co.uk\/cdn\/shop\/products\/26509-pm_d0fed167-a486-446a-98ab-ba43f3a40eb4.png?v=1679168963"},"aspect_ratio":1.0,"height":776,"media_type":"image","src":"\/\/projektride.co.uk\/cdn\/shop\/products\/26509-pm_d0fed167-a486-446a-98ab-ba43f3a40eb4.png?v=1679168963","width":776}],"requires_selling_plan":false,"selling_plan_groups":[],"content":"\u003cmeta charset=\"utf-8\"\u003e\n\u003cdiv class=\"col\"\u003e\n\u003cdiv class=\"description\"\u003e\n\u003cp\u003eAll-Mountain grip with a superb tactile feel thanks to the super soft, UV-stable rubber compound, exclusively developed and manufactured in Germany for Ergon. This allows for greater control on demanding trails.\u003c\/p\u003e\n\u003cp\u003eThe ergonomic shape fits the contours of the hand and significantly reduced arm pump on long trails. The inner construction varies in thickness allowing greater damping.\u003c\/p\u003e\n\u003cul\u003e\n\u003cli\u003eCarbon friendly CNC Aluminium clamps\u003c\/li\u003e\n\u003cli\u003eStandard or Single Gripshift versions\u003c\/li\u003e\n\u003cli\u003eReplaceable bar end plugs\u003c\/li\u003e\n\u003cli\u003e110g per pair\u003c\/li\u003e\n\u003c\/ul\u003e\n\u003c\/div\u003e\n\u003c\/div\u003e\n\u003cdiv class=\"col\"\u003e\n\u003cdiv class=\"features\"\u003e\u003c\/div\u003e\n\u003c\/div\u003e"}; window._RestockRocketConfig.variantsInventoryPolicy = {43936694763746 : "deny",43936694796514 : "deny",43936694829282 : "deny",43936694862050 : "deny",43936694894818 : "deny",43936694927586 : "deny",43936694960354 : "deny",43936694993122 : "deny",}; window._RestockRocketConfig.variantsInventoryQuantity = {43936694763746 : parseInt("3"),43936694796514 : parseInt("3"),43936694829282 : parseInt("3"),43936694862050 : parseInt("3"),43936694894818 : parseInt("3"),43936694927586 : parseInt("3"),43936694960354 : parseInt("3"),43936694993122 : parseInt("3"),}; window._RestockRocketConfig.variantsPreorderCount = {43936694763746 : parseInt(""),43936694796514 : parseInt(""),43936694829282 : parseInt(""),43936694862050 : parseInt(""),43936694894818 : parseInt(""),43936694927586 : parseInt(""),43936694960354 : parseInt(""),43936694993122 : parseInt(""),}; window._RestockRocketConfig.variantsPreorderCountForMarket = {43936694763746 : null,43936694796514 : null,43936694829282 : null,43936694862050 : null,43936694894818 : null,43936694927586 : null,43936694960354 : null,43936694993122 : null,}; window._RestockRocketConfig.variantsPreorderMaxCount = {43936694763746 : parseInt(""),43936694796514 : parseInt(""),43936694829282 : parseInt(""),43936694862050 : parseInt(""),43936694894818 : parseInt(""),43936694927586 : parseInt(""),43936694960354 : parseInt(""),43936694993122 : parseInt(""),}; window._RestockRocketConfig.variantsPreorderMaxCountForMarket = {43936694763746 : null,43936694796514 : null,43936694829282 : null,4393669 4862050 : null,43936694894818 : null,43936694927586 : null,43936694960354 : null,43936694993122 : null,}; window._RestockRocketConfig.variantsShippingText = {43936694763746 : "",43936694796514 : "",43936694829282 : "",43936694862050 : "",43936694894818 : "",43936694927586 : "",43936694960354 : "",43936694993122 : "",}; window._RestockRocketConfig.variantsShippingTextForMarket = {43936694763746 : null,43936694796514 : null,43936694829282 : null,43936694862050 : null,43936694894818 : null,43936694927586 : null,43936694960354 : null,43936694993122 : null,}; window._RestockRocketConfig.selected_variant_id = 43936694763746; window._RestockRocketConfig.selected_variant_available = window._RestockRocketConfig.product.variants.find(function(variant) { return variant.id == window._RestockRocketConfig.selected_variant_id }).available;window._RestockRocketConfig.scriptUrlProduct = 'https://cdn.shopify.com/extensions/019ecfe3-4ce2-7592-b7ce-9ed4a1b98146/restockrocket-1-530/assets/restockrocket-product.js'
window._RestockRocketConfig.scriptUrlCollection = 'https://cdn.shopify.com/extensions/019ecfe3-4ce2-7592-b7ce-9ed4a1b98146/restockrocket-1-530/assets/restockrocket-collection.js' window._RestockRocketConfig.scriptUrlProductBis = 'https://cdn.shopify.com/extensions/019ecfe3-4ce2-7592-b7ce-9ed4a1b98146/restockrocket-1-530/assets/restockrocket-product-bis.js' window._RestockRocketConfig.scriptUrlCollectionBis = 'https://cdn.shopify.com/extensions/019ecfe3-4ce2-7592-b7ce-9ed4a1b98146/restockrocket-1-530/assets/restockrocket-collection-bis.js' window._RestockRocketConfig.scriptHost = window._RestockRocketConfig.scriptUrlProduct.substring(0, window._RestockRocketConfig.scriptUrlProduct.lastIndexOf('/') + 1) window._RestockRocketConfig.host = 'https://app.restockrocket.io' // Deployed extension build number, read from the CDN asset host Shopify generates: // https://cdn.shopify.com/extensions/ <Header Imageuuid>/ <
handle>- <version>/assets/... // Trailing digits (e.g. ".../restockrocket-1-521/assets/" -> "521"). Kept numeric to // match ParseStoqData, so funnel app_version lines up with the order-attribution // app_version. Reflects the ACTUAL deployed build. This is the SINGLE source of the // parsed version — preorder.js getAppVersion() reads it back off config rather than // re-parsing, so the regex lives in exactly one place. try { const _stoqVersionMatch = window._RestockRocketConfig.scriptHost.match(/(\d+)\/?(?:assets\/?)?$/); window._RestockRocketConfig.appVersion = (_stoqVersionMatch && _stoqVersionMatch[1]) || ''; } catch (e) { window._RestockRocketConfig.appVersion = ''; } const SETTINGS_CACHE_DURATION = 15 * 60 * 1000; // 15 minutes in milliseconds const LIQUID_CACHE_MAX_AGE = 15 * 60; // 15 minutes in seconds // Calculate Liquid cache freshness once at initialization const liquidRenderedAt = window._RestockRocketConfig.liquidRenderedAt; // Validate timestamp and calculate ca che age if (!liquidRenderedAt || typeof liquidRenderedAt !== 'number' || isNaN(liquidRenderedAt)) { console.debug('STOQ - Invalid or missing liquidRenderedAt timestamp, assuming fresh'); window._RestockRocketConfig.isLiquidCacheFresh = true; window._RestockRocketConfig.liquidCacheAge = null; } else { const now = Math.floor(Date.now() / 1000); // Current time in seconds const liquidCacheAge = now - liquidRenderedAt; // Age in seconds // Surfaced into funnel events: a stale cache means the app rendered with // outdated inventory/selling-plan data — a real "had the opportunity but // failed" cause. Negative (client clock ahead) clamps to 0. window._RestockRocketConfig.liquidCacheAge = Math.max(0, liquidCacheAge); // Handle client clock ahead of server if (liquidCacheAge
< 0) { console.debug(`STOQ - Client clock appears ahead of server by ${Math.abs(Math.round(liquidCacheAge / 60))} minutes, assuming cache fresh`); window._RestockRocketConfig.isLiquidCacheFresh = true; } else if (liquidCacheAge
< = LIQUID_CACHE_MAX_AGE) { console.debug(`STOQ - Liquid cache is fresh (${Math.round(liquidCacheAge / 60)} minutes old)`); window._RestockRocketConfig.isLiquidCacheFresh = true; } else { console.debug(`STOQ - Liquid cache is stale (${Math.round(liquidCacheAge / 60)} minutes old, max ${Math.round(LIQUID_CACHE_MAX_AGE / 60)} minutes)`); window._RestockRocketConfig.isLiquidCacheFresh = false; } } 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 if translations are enabled but missing from cache // This handles the backfill period where DB has translations but metafield doesn't if (settings.multi_language_enabled) { if (!settings.translations) { // Translations enabled but no ormalizedLocale = 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]; if (value !== null && value !== undefined && value !== '') { settings[key] = value; } }); } else { console.debug('STOQ - No translated fields found for locale:', normalizedL ocale); } delete settings.translations; return settings; } catch (e) { console.debug('STOQ - error applying translations:', e); return settings; } } // Setup event listener for cart selling plan updates // This must be called before any scripts are loaded to avoid race conditions function setupCartSellingPlanUpdater(settings) { // Setup listener regardless - updateCartSellingPlans has its own guards // This ensures cleanup happens even when preorders are disabled globally // Listen for stoq:inventory-data-loaded event dispatched by api.js window.addEventListener('stoq:inventory-data-loaded', function(event) { console.debug('STOQ - Inventory data loaded, updating cart selling plans'); if (window._RestockRocket && window._RestockRocket.updateCartSellingPlans) { window._RestockRocket.updateCartSellingPlans() .then(hasUpdates => { if (hasUpdates) { console.debug('STOQ - cart selling plans update
d successfully'); } else { console.debug('STOQ - no cart selling plan updates needed'); } }) .catch(error => { console.error('STOQ - error updating cart selling plans:', error); }); } }); } // 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 - using cached settings'); initializeScripts(validCachedSettings); } else { console.debug('STOQ - fetching fresh settings'); const headers = { 'X-Shopify-Shop-Domain': window._RestockRocketConfig.shop || window.Shopify.shop, 'ngrok-skip-browser-warning': 'skip' }; if (window.Shopify?.theme?.role === 'main') { headers['X-Shopify-Theme-Schema-Name'] = window.Shopify.theme.schema_name; headers['X-Shopify-Theme-Sche ma-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) { 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 fetchEmbedConfig(endpoint, apply) { return fetch( `${window._RestockRocketConfig.host}/api/v1/embed/${endpoint}.json`, { headers: { 'X-Shopify-Shop-Domain': window._RestockRocketConfig.shop || window.Shopify.shop, 'ngrok-skip-browser-warning': 'skip' } } ) .then(function(response) { if (!response.ok) throw new Error(`Failed to fetch ${endpoint}`); return response.json(); }) .then(function(data) { try { apply(data); } catch (applyError) { // Apply failures are programming bugs (e.g. response shape changed // server-side and the assignment threw). Surface them as console.error // so they're visible in browser logs, then re-throw to fall through // to the same Liquid-cached fallback as a fetch failure. console.error('STOQ - apply failed for ' + endpoint + ':', applyError); throw applyError; } }) .catch(function(error) { console.debug(`STOQ - using cached ${endpoint}:`, error.message); } ); } function initializeScripts(settings) { settings = applyTranslations(settings); window._RestockRocketConfig.settings = settings; console.debug(`STOQ - settings configured for ${window._RestockRocketConfig.pageType}`); // Stale-Liquid resilience (default-on, per-shop opt-out via the // `disable_refresh_on_stale_liquid` Toggle, surfaced as the negative // `disable_refresh_on_stale_liquid` flag in settings.json so that // `undefined` -- in CDN-cached metafield payloads that predate this // key -- reads as `!undefined === true` and gets default-on behavior // immediately, no metafield rewrite required). // When the Liquid CDN cache is older than LIQUID_CACHE_MAX_AGE the in-page // selling_plans / integrations metafields can be wrong; refresh both from // the API before launching scripts. Race against a 1000ms timeout so a slow // API can't block init indefinitely. If the timeout wins, the in-flight // fetches still complete and update window._Resto
ckRocketConfig — the // bundle re-reads sellingPlans/integrations on every interaction, so the // late-arriving values benefit subsequent renders even though the first // paint may use the Liquid-cached values. On any failure the existing // Liquid-loaded values stay in place via fetchEmbedConfig's catch. if (!window._RestockRocketConfig.isLiquidCacheFresh && !settings.disable_refresh_on_stale_liquid) { console.debug('STOQ - Liquid cache stale, refreshing selling_plans + integrations'); Promise.race([ Promise.all([ fetchEmbedConfig('selling_plans', function(data) { if (data && Array.isArray(data.plans)) { window._RestockRocketConfig.sellingPlans = data.plans; window._RestockRocketConfig.disabledSellingPlanIds = data.disabled_plan_ids || []; } }), fetchEmbedConfig('integrations', function(data) { if (Array.isArray(data)) { window._RestockRocketConfig.integrations = data; } }) ]), new Promise(function(resolve) { setTimeout(resolve, 1000); }) ]).then(function() { loadScripts(settings); }); return; } loadScripts(settings); } function loadScripts(settings) { // Setup cart selling plan updater BEFORE loading any scripts to avoid race conditions setupCartSellingPlanUpdater(settings); 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); }) // STOQ-1520: serve the lean back-in-stock-only build (no preorder/hijack code) // only to shops with NO preorder plans. Use the full build if preorder is on, // an enabled offer exists, or a disabled-but-kept plan id remains (cart sweep // must still strip those). Rationale in the PR. const hasEnabledOffer = Arra
    y.isArray(window._RestockRocketConfig.sellingPlans) && window._RestockRocketConfig.sellingPlans.some(function(plan) { return plan && plan.enabled; }); const hasDisabledPlanIds = Array.isArray(window._RestockRocketConfig.disabledSellingPlanIds) && window._RestockRocketConfig.disabledSellingPlanIds.length > 0; const usePreorderBuild = settings.preorder_enabled || hasEnabledOffer || hasDisabledPlanIds; const collectionScriptUrl = usePreorderBuild ? window._RestockRocketConfig.scriptUrlCollection : window._RestockRocketConfig.scriptUrlCollectionBis; const productScriptUrl = usePreorderBuild ? window._RestockRocketConfig.scriptUrlProduct : window._RestockRocketConfig.scriptUrlProductBis; const pageType = window._RestockRocketConfig.pageType; const collectionPageTypes = ['collection', 'index', 'search', 'page']; if(collectionPageTypes.indexOf(pageType) !== -1 && (settings[`show_button_on_${pageType}`] || settings[`preorder_ ${pageType}_enabled`])) { createRestockRocketScript(collectionScriptUrl); } else if(pageType === 'product') { createRestockRocketScript(productScriptUrl); } else if(hijackIntegration) { createRestockRocketScript(window._RestockRocketConfig.scriptUrlCollection); } else if(usePreorderBuild) { // cart/article/blog/list-collections: full build so the cart sweep runs. createRestockRocketScript(window._RestockRocketConfig.scriptUrlCollection); } else { console.debug(`STOQ - no scripts enabled for ${pageType}`); } // Dispatch custom event when app is loaded // Cart selling plan updates will be triggered by stoq:inventory-data-loaded event const appLoadedEvent = new CustomEvent('stoq:loaded', { detail: { pageType: window._RestockRocketConfig.pageType, enabled: settings.enable_app, settings: settings, preorderEnabled: settings.preorder_enabled } }); consol

Kona Coco HD

Size
Quantity
Looking to purchase with a cycle to work scheme?

Click here to get in touch and get the ball rolling!

There’s style and then there’s smart style. The Coco HD is the best of both worlds. With an efficient hub drive motor that makes every ride enjoyable, it’s an easy solution to a daily commute. Its step-through design and head-turning bold colour is the perfect accessory for someone looking to enjoy their rides without the hassle of a tall top tube. Take it to brunch, to the park, or take it to work. The Coco HD is a game-changer for the stylish rider.

SPECS & GEOMETRY


SPECIFICATIONS

FRAME MATERIAL Kona 6061 Aluminum Butted
SIZES S, M, L
REAR SHOCK n/a
FORK Kona Project Two Aluminum Disc
CRANKARMS Suntour
CHAINRINGS 38t
B/B Suntour
PEDALS Kona JS2
CHAIN KMC Z8.3
CASSETTE Shimano Altus 11-32t 8spd
CHAINGUIDE n/a
F/D n/a
R/D Shimano TX
SHIFTERS Shimano Altus
BRAKE CALIPERS Tektro HDM275 Hydraulic
FRONT BRAKE ROTOR Tektro TR180 180mm
REAR BRAKE ROTOR Tektro TR180 180mm
BRAKE LEVERS Tektro HDM275 Hydraulic
HEADSET Feimin FP-H863
HANDLEBAR Kona Handplant
STEM Kona Commuter
SEATPOST Kona Commuter 27.2mm
SEAT CLAMP Kona Clamp
GRIPS Kona Key Grip Slip on
SADDLE Kona Commuter
FRONT HUB Shimano 100x9mm
REAR HUB SR Suntour HD 20MPH
SPOKES Stainless Black 14g/13g
RIMS WTB SX19
FRONT TIRE WTB Horizon Comp 650x47c
REAR TIRE WTB Horizon Comp 650x47c
DRIVEUNIT SR Suntour HD 20MPH
BATTERY Phylion 418Wh
DISPLAY SR Suntour OLED Display
CHARGER Phylion 2A
BATTERYLOCK SR Suntour
PAINT COLOR Gloss Metallic Yellow w/ Charcoal Decals
EXTRAS Kickstand
All specifications are subject to change without notice



GEOMETRY

  SIZE S M L
1 ST LENGTH 420 470 520
2 TT LENGTH 553 597 636
3 REACH 395 423 439
4 STACK 591 606 643
5 STANDOVER 584 584 591
6 HT ANGLE 70.0 70.0 70.0
7 HT LENGTH 170 185 225
8 ST ANGLE 75.0 74.0 73.0
9 CS LENGTH 450 450 450
10 BB DROP 60 60 60
11 BB HEIGHT 282 282 282
12 WHEELBASE 1088 1121 1150
13 FRONT CENTER 644 677 707
14 FORK LENGTH 409 409 409
15 FORK OFFSET 50 50 50
Bike Geometry

Frequently Asked Questions

  • Please get in touch with a member of the team either by phone (01313745324) or email ([email protected]) where on of the team will be more than happy to help.

    ProjektRide Bike Shop Edinburgh

  • If the item is showing in stock, we aim to post the product within 24 hours. Please allow 5 working days to receive the item.

    Postage is free on orders over £50. Orders under £50, our postage charge is £3.99.

    We also have a physical store, if you are local please pop in -

    ProjektRide Bike Shop Edinburgh

    82 Newington Road, EH91QN, Edinburgh.