create MetacodeSelect component
This commit is contained in:
parent
61e27a4dcb
commit
5d04d16590
13 changed files with 424 additions and 30 deletions
|
@ -164,10 +164,12 @@ jQuery.browser = browser;
|
||||||
// Add code that makes tab and shift+tab scroll through metacodes
|
// Add code that makes tab and shift+tab scroll through metacodes
|
||||||
$('.new_topic').bind('keydown',this,function(event){
|
$('.new_topic').bind('keydown',this,function(event){
|
||||||
if (event.keyCode == 9 && event.shiftKey) {
|
if (event.keyCode == 9 && event.shiftKey) {
|
||||||
|
$(container).show()
|
||||||
event.data.rotate(-1);
|
event.data.rotate(-1);
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
} else if (event.keyCode == 9) {
|
} else if (event.keyCode == 9) {
|
||||||
|
$(container).show()
|
||||||
event.data.rotate(1);
|
event.data.rotate(1);
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
|
@ -178,12 +180,12 @@ jQuery.browser = browser;
|
||||||
if (options.mouseWheel)
|
if (options.mouseWheel)
|
||||||
{
|
{
|
||||||
// START METAMAPS CODE
|
// START METAMAPS CODE
|
||||||
$('body').bind('mousewheel',this,function(event, delta) {
|
/*$('body').bind('mousewheel',this,function(event, delta) {
|
||||||
if (Metamaps.Create.newTopic.beingCreated && !Metamaps.Create.isSwitchingSet) {
|
if (Metamaps.Create.newTopic.beingCreated && !Metamaps.Create.isSwitchingSet) {
|
||||||
event.data.rotate(delta);
|
event.data.rotate(delta);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
});
|
});*/
|
||||||
// END METAMAPS CODE
|
// END METAMAPS CODE
|
||||||
/* ORIGINAL CODE
|
/* ORIGINAL CODE
|
||||||
$(container).bind('mousewheel',this,function(event, delta) {
|
$(container).bind('mousewheel',this,function(event, delta) {
|
||||||
|
@ -246,12 +248,10 @@ jQuery.browser = browser;
|
||||||
{
|
{
|
||||||
if ( items[this.frontIndex] === undefined ) { return; } // Images might not have loaded yet.
|
if ( items[this.frontIndex] === undefined ) { return; } // Images might not have loaded yet.
|
||||||
// METAMAPS CODE
|
// METAMAPS CODE
|
||||||
Metamaps.Create.newTopic.metacode = $(items[this.frontIndex].image).attr('data-id');
|
Metamaps.Create.newTopic.setMetacode($(items[this.frontIndex].image).attr('data-id'))
|
||||||
//$('img.cloudcarousel').css({"background":"none", "width":"","height":""});
|
|
||||||
//$(items[this.frontIndex].image).css({"width":"45px","height":"45px"});
|
|
||||||
// NOT METAMAPS CODE
|
// NOT METAMAPS CODE
|
||||||
$(options.titleBox).html( $(items[this.frontIndex].image).attr('title'));
|
//$(options.titleBox).html( $(items[this.frontIndex].image).attr('title'));
|
||||||
$(options.altBox).html( $(items[this.frontIndex].image).attr('alt'));
|
//$(options.altBox).html( $(items[this.frontIndex].image).attr('alt'));
|
||||||
};
|
};
|
||||||
|
|
||||||
this.go = function()
|
this.go = function()
|
||||||
|
@ -265,6 +265,9 @@ jQuery.browser = browser;
|
||||||
{
|
{
|
||||||
clearTimeout(this.controlTimer);
|
clearTimeout(this.controlTimer);
|
||||||
this.controlTimer = 0;
|
this.controlTimer = 0;
|
||||||
|
// METAMAPS CODE
|
||||||
|
$(container).hide()
|
||||||
|
// END METAMAPS CODE
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -385,7 +388,7 @@ jQuery.browser = browser;
|
||||||
}
|
}
|
||||||
// If all images have valid widths and heights, we can stop checking.
|
// If all images have valid widths and heights, we can stop checking.
|
||||||
clearInterval(this.tt);
|
clearInterval(this.tt);
|
||||||
this.showFrontText();
|
// METAMAPS COMMENT this.showFrontText();
|
||||||
this.autoRotate();
|
this.autoRotate();
|
||||||
this.updateAll();
|
this.updateAll();
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
/* global Metamaps, $ */
|
/* global Metamaps, $, ReactDOM, React */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Metamaps.Create.js
|
* Metamaps.Create.js
|
||||||
|
@ -22,6 +22,8 @@ Metamaps.Create = {
|
||||||
newSelectedMetacodeNames: [],
|
newSelectedMetacodeNames: [],
|
||||||
selectedMetacodes: [],
|
selectedMetacodes: [],
|
||||||
newSelectedMetacodes: [],
|
newSelectedMetacodes: [],
|
||||||
|
recentMetacodes: [],
|
||||||
|
mostUsedMetacodes: [],
|
||||||
init: function () {
|
init: function () {
|
||||||
var self = Metamaps.Create
|
var self = Metamaps.Create
|
||||||
self.newTopic.init()
|
self.newTopic.init()
|
||||||
|
@ -82,7 +84,6 @@ Metamaps.Create = {
|
||||||
}
|
}
|
||||||
metacodeModels.sort()
|
metacodeModels.sort()
|
||||||
|
|
||||||
$('#metacodeImg, #metacodeImgTitle').empty()
|
|
||||||
$('#metacodeImg').removeData('cloudcarousel')
|
$('#metacodeImg').removeData('cloudcarousel')
|
||||||
var newMetacodes = ''
|
var newMetacodes = ''
|
||||||
metacodeModels.each(function (metacode) {
|
metacodeModels.each(function (metacode) {
|
||||||
|
@ -90,13 +91,11 @@ Metamaps.Create = {
|
||||||
})
|
})
|
||||||
|
|
||||||
$('#metacodeImg').empty().append(newMetacodes).CloudCarousel({
|
$('#metacodeImg').empty().append(newMetacodes).CloudCarousel({
|
||||||
titleBox: $('#metacodeImgTitle'),
|
|
||||||
yRadius: 40,
|
yRadius: 40,
|
||||||
xRadius: 190,
|
xRadius: 190,
|
||||||
xPos: 170,
|
xPos: 170,
|
||||||
yPos: 40,
|
yPos: 40,
|
||||||
speed: 0.3,
|
speed: 0.3,
|
||||||
mouseWheel: true,
|
|
||||||
bringToFront: true
|
bringToFront: true
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -147,10 +146,24 @@ Metamaps.Create = {
|
||||||
},
|
},
|
||||||
newTopic: {
|
newTopic: {
|
||||||
init: function () {
|
init: function () {
|
||||||
$('#topic_name').keyup(function () {
|
var DOWN_ARROW = 40
|
||||||
|
|
||||||
|
$('#topic_name').keyup(function (e) {
|
||||||
Metamaps.Create.newTopic.name = $(this).val()
|
Metamaps.Create.newTopic.name = $(this).val()
|
||||||
|
if (e.which == DOWN_ARROW && !Metamaps.Create.newTopic.name.length) {
|
||||||
|
Metamaps.Create.newTopic.openSelector()
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
$('.selectedMetacode').click(function() {
|
||||||
|
if (Metamaps.Create.newTopic.metacodeSelectorOpen) {
|
||||||
|
Metamaps.Create.newTopic.hideSelector()
|
||||||
|
$('#topic_name').focus()
|
||||||
|
} else Metamaps.Create.newTopic.openSelector()
|
||||||
|
})
|
||||||
|
|
||||||
|
Metamaps.Create.newTopic.initSelector()
|
||||||
|
|
||||||
var topicBloodhound = new Bloodhound({
|
var topicBloodhound = new Bloodhound({
|
||||||
datumTokenizer: Bloodhound.tokenizers.obj.whitespace('value'),
|
datumTokenizer: Bloodhound.tokenizers.obj.whitespace('value'),
|
||||||
queryTokenizer: Bloodhound.tokenizers.whitespace,
|
queryTokenizer: Bloodhound.tokenizers.whitespace,
|
||||||
|
@ -183,16 +196,15 @@ Metamaps.Create = {
|
||||||
$('#topic_name').bind('typeahead:select', function (event, datum, dataset) {
|
$('#topic_name').bind('typeahead:select', function (event, datum, dataset) {
|
||||||
Metamaps.Topic.getTopicFromAutocomplete(datum.id)
|
Metamaps.Topic.getTopicFromAutocomplete(datum.id)
|
||||||
})
|
})
|
||||||
|
$('#topic_name').click(function() { Metamaps.Create.newTopic.hideSelector() })
|
||||||
|
|
||||||
// initialize metacode spinner and then hide it
|
// initialize metacode spinner and then hide it
|
||||||
$('#metacodeImg').CloudCarousel({
|
$('#metacodeImg').CloudCarousel({
|
||||||
titleBox: $('#metacodeImgTitle'),
|
|
||||||
yRadius: 40,
|
yRadius: 40,
|
||||||
xRadius: 190,
|
xRadius: 190,
|
||||||
xPos: 170,
|
xPos: 170,
|
||||||
yPos: 40,
|
yPos: 40,
|
||||||
speed: 0.3,
|
speed: 0.3,
|
||||||
mouseWheel: true,
|
|
||||||
bringToFront: true
|
bringToFront: true
|
||||||
})
|
})
|
||||||
$('.new_topic').hide()
|
$('.new_topic').hide()
|
||||||
|
@ -200,10 +212,59 @@ Metamaps.Create = {
|
||||||
name: null,
|
name: null,
|
||||||
newId: 1,
|
newId: 1,
|
||||||
beingCreated: false,
|
beingCreated: false,
|
||||||
|
metacodeSelectorOpen: false,
|
||||||
metacode: null,
|
metacode: null,
|
||||||
x: null,
|
x: null,
|
||||||
y: null,
|
y: null,
|
||||||
addSynapse: false,
|
addSynapse: false,
|
||||||
|
initSelector: function () {
|
||||||
|
ReactDOM.render(
|
||||||
|
React.createElement(Metamaps.ReactComponents.MetacodeSelect, {
|
||||||
|
onClick: function (id) {
|
||||||
|
Metamaps.Create.newTopic.setMetacode(id)
|
||||||
|
Metamaps.Create.newTopic.hideSelector()
|
||||||
|
$('#topic_name').focus()
|
||||||
|
},
|
||||||
|
close: function () {
|
||||||
|
Metamaps.Create.newTopic.hideSelector()
|
||||||
|
$('#topic_name').focus()
|
||||||
|
},
|
||||||
|
metacodes: Metamaps.Metacodes.models,
|
||||||
|
recent: Metamaps.Create.recentMetacodes,
|
||||||
|
mostUsed: Metamaps.Create.mostUsedMetacodes
|
||||||
|
}),
|
||||||
|
document.getElementById('metacodeSelector')
|
||||||
|
)
|
||||||
|
},
|
||||||
|
openSelector: function () {
|
||||||
|
Metamaps.Create.newTopic.initSelector()
|
||||||
|
$('#metacodeSelector').show()
|
||||||
|
Metamaps.Create.newTopic.metacodeSelectorOpen = true
|
||||||
|
$('.metacodeFilterInput').focus()
|
||||||
|
$('.selectedMetacode').addClass('isBeingSelected')
|
||||||
|
},
|
||||||
|
hideSelector: function () {
|
||||||
|
ReactDOM.unmountComponentAtNode(document.getElementById('metacodeSelector'))
|
||||||
|
$('#metacodeSelector').hide()
|
||||||
|
Metamaps.Create.newTopic.metacodeSelectorOpen = false
|
||||||
|
$('.selectedMetacode').removeClass('isBeingSelected')
|
||||||
|
},
|
||||||
|
setMetacode: function (id) {
|
||||||
|
Metamaps.Create.newTopic.metacode = id
|
||||||
|
var metacode = Metamaps.Metacodes.get(id)
|
||||||
|
$('.selectedMetacode img').attr('src', metacode.get('icon'))
|
||||||
|
$('.selectedMetacode span').html(metacode.get('name'))
|
||||||
|
$.ajax({
|
||||||
|
type: 'POST',
|
||||||
|
dataType: 'json',
|
||||||
|
url: '/user/update_metacode_focus',
|
||||||
|
data: { value: id },
|
||||||
|
success: function (data) {},
|
||||||
|
error: function () {
|
||||||
|
console.log('failed to save metacode focus')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
open: function () {
|
open: function () {
|
||||||
$('#new_topic').fadeIn('fast', function () {
|
$('#new_topic').fadeIn('fast', function () {
|
||||||
$('#topic_name').focus()
|
$('#topic_name').focus()
|
||||||
|
@ -215,6 +276,7 @@ Metamaps.Create = {
|
||||||
$('#new_topic').fadeOut('fast')
|
$('#new_topic').fadeOut('fast')
|
||||||
$('#topic_name').typeahead('val', '')
|
$('#topic_name').typeahead('val', '')
|
||||||
Metamaps.Create.newTopic.beingCreated = false
|
Metamaps.Create.newTopic.beingCreated = false
|
||||||
|
Metamaps.Create.newTopic.hideSelector()
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
newSynapse: {
|
newSynapse: {
|
||||||
|
|
|
@ -706,7 +706,7 @@ Metamaps.JIT = {
|
||||||
Metamaps.GlobalUI.CreateMap.submit()
|
Metamaps.GlobalUI.CreateMap.submit()
|
||||||
}
|
}
|
||||||
// this is to submit new topic creation
|
// this is to submit new topic creation
|
||||||
else if (Metamaps.Create.newTopic.beingCreated) {
|
else if (Metamaps.Create.newTopic.beingCreated && !Metamaps.Create.newTopic.metacodeSelectorOpen) {
|
||||||
Metamaps.Topic.createTopicLocally()
|
Metamaps.Topic.createTopicLocally()
|
||||||
}
|
}
|
||||||
// to submit new synapse creation
|
// to submit new synapse creation
|
||||||
|
|
|
@ -523,18 +523,48 @@ button.button.btn-no:hover {
|
||||||
left: -1000px;
|
left: -1000px;
|
||||||
display: block;
|
display: block;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
width: 340px;
|
|
||||||
margin: -40px 0 0 -35px;
|
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.selectedMetacode {
|
||||||
|
float: left;
|
||||||
|
background: #FFF;
|
||||||
|
border-top-left-radius: 2px;
|
||||||
|
border-bottom-left-radius: 2px;
|
||||||
|
padding: 5px 10px 5px 6px;
|
||||||
|
vertical-align: top;
|
||||||
|
border-right: 1px solid #DDD;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.selectedMetacode:hover, .selectedMetacode.isBeingSelected {
|
||||||
|
background: #EDEDED;
|
||||||
|
}
|
||||||
|
.selectedMetacode img {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
display: inline-block;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
.selectedMetacode span {
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
.selectedMetacode .downArrow {
|
||||||
|
display: inline-block;
|
||||||
|
vertical-align: middle;
|
||||||
|
width: 0;
|
||||||
|
height: 0;
|
||||||
|
border-style: solid;
|
||||||
|
border-width: 8px 6px 0 6px;
|
||||||
|
border-color: #777 transparent transparent transparent;
|
||||||
|
margin-left: 2px;
|
||||||
|
margin-top: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
#new_topic .twitter-typeahead {
|
#new_topic .twitter-typeahead {
|
||||||
position: absolute !important;
|
|
||||||
top: 45px;
|
|
||||||
left: 41px;
|
|
||||||
z-index: 9999;
|
z-index: 9999;
|
||||||
width: 256px;
|
width: 256px;
|
||||||
height: 34px;
|
height: 34px;
|
||||||
|
float: left;
|
||||||
}
|
}
|
||||||
.new_topic #topic_name,
|
.new_topic #topic_name,
|
||||||
.new_topic .tt-hint {
|
.new_topic .tt-hint {
|
||||||
|
@ -544,7 +574,8 @@ button.button.btn-no:hover {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 10px 6px;
|
padding: 10px 6px;
|
||||||
border: none;
|
border: none;
|
||||||
border-radius: 2px;
|
border-top-right-radius: 2px;
|
||||||
|
border-bottom-right-radius: 2px;
|
||||||
outline: none;
|
outline: none;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
line-height: 14px;
|
line-height: 14px;
|
||||||
|
@ -555,7 +586,7 @@ button.button.btn-no:hover {
|
||||||
color: #BDBDBD;
|
color: #BDBDBD;
|
||||||
}
|
}
|
||||||
.openMetacodeSwitcher {
|
.openMetacodeSwitcher {
|
||||||
display: block;
|
display: none;
|
||||||
height: 16px;
|
height: 16px;
|
||||||
width: 16px;
|
width: 16px;
|
||||||
background-image: url(<%= asset_data_uri('metacodesettings_sprite.png') %>);
|
background-image: url(<%= asset_data_uri('metacodesettings_sprite.png') %>);
|
||||||
|
@ -569,8 +600,14 @@ button.button.btn-no:hover {
|
||||||
}
|
}
|
||||||
#metacodeImg {
|
#metacodeImg {
|
||||||
height: 120px;
|
height: 120px;
|
||||||
|
width: 380px;
|
||||||
|
display: none;
|
||||||
|
position: absolute !important;
|
||||||
|
top: -30px;
|
||||||
|
z-index: -1;
|
||||||
}
|
}
|
||||||
#metacodeImgTitle {
|
#metacodeImgTitle {
|
||||||
|
display: none;
|
||||||
float: left;
|
float: left;
|
||||||
width: 120px;
|
width: 120px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
|
72
app/assets/stylesheets/metacode-select.scss.erb
Normal file
72
app/assets/stylesheets/metacode-select.scss.erb
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
#metacodeSelector {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.metacodeSelect {
|
||||||
|
border-top: 1px solid #DDD;
|
||||||
|
padding: 0;
|
||||||
|
|
||||||
|
.tabList {
|
||||||
|
float: left;
|
||||||
|
background: #FFF;
|
||||||
|
|
||||||
|
div {
|
||||||
|
border-right: 1px solid #DDD;
|
||||||
|
border-bottom: 1px solid #DDD;
|
||||||
|
}
|
||||||
|
|
||||||
|
div:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.active {
|
||||||
|
border-right: none;
|
||||||
|
background: #EEE;
|
||||||
|
|
||||||
|
.metacodeFilterInput {
|
||||||
|
background: #EEE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.metacodeFilterInput {
|
||||||
|
width: 100px;
|
||||||
|
outline: none;
|
||||||
|
border: 0;
|
||||||
|
padding: 8px;
|
||||||
|
font-size: 14px;
|
||||||
|
line-height: 14px;
|
||||||
|
color: #424242;
|
||||||
|
font-family: 'din-medium', helvetica, sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
span {
|
||||||
|
padding: 8px;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.metacodeList {
|
||||||
|
float:left;
|
||||||
|
list-style: none;
|
||||||
|
background: #FFF;
|
||||||
|
min-height: 107px;
|
||||||
|
min-width: 100px;
|
||||||
|
|
||||||
|
li {
|
||||||
|
padding: 8px;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
&:hover, &.keySelect {
|
||||||
|
background: #EEE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
display: inline-block;
|
||||||
|
vertical-align: middle;
|
||||||
|
padding-right: 6px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
class UsersController < ApplicationController
|
class UsersController < ApplicationController
|
||||||
before_action :require_user, only: [:edit, :update, :updatemetacodes]
|
before_action :require_user, only: [:edit, :update, :updatemetacodes, :update_metacode_focus]
|
||||||
|
|
||||||
respond_to :html, :json
|
respond_to :html, :json
|
||||||
|
|
||||||
|
@ -92,6 +92,16 @@ class UsersController < ApplicationController
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# PUT /user/update_metacode_focus
|
||||||
|
def update_metacode_focus
|
||||||
|
@user = current_user
|
||||||
|
@user.settings.metacode_focus = params[:value]
|
||||||
|
@user.save
|
||||||
|
respond_to do |format|
|
||||||
|
format.json { render json: { success: "success" }}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def user_params
|
def user_params
|
||||||
|
|
|
@ -16,6 +16,10 @@ module ApplicationHelper
|
||||||
@metacodes.sort! { |m1, m2| m2.name.downcase <=> m1.name.downcase }.rotate!(-1)
|
@metacodes.sort! { |m1, m2| m2.name.downcase <=> m1.name.downcase }.rotate!(-1)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def user_metacode
|
||||||
|
current_user.settings.metacode_focus ? Metacode.find(current_user.settings.metacode_focus.to_i) || user_metacodes()[0] : user_metacodes()[0]
|
||||||
|
end
|
||||||
|
|
||||||
def determine_invite_link
|
def determine_invite_link
|
||||||
"#{request.base_url}/join" + (current_user ? "?code=#{current_user.code}" : '')
|
"#{request.base_url}/join" + (current_user ? "?code=#{current_user.code}" : '')
|
||||||
end
|
end
|
||||||
|
|
|
@ -64,6 +64,28 @@ class User < ActiveRecord::Base
|
||||||
json
|
json
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def recentMetacodes
|
||||||
|
array = []
|
||||||
|
self.topics.sort{|a,b| b.created_at <=> a.created_at }.each do |t|
|
||||||
|
if array.length < 5 and array.index(t.metacode_id) == nil
|
||||||
|
array.push(t.metacode_id)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
array
|
||||||
|
end
|
||||||
|
|
||||||
|
def mostUsedMetacodes
|
||||||
|
self.topics.to_a.reduce({}) { |memo, topic|
|
||||||
|
if memo[topic.metacode_id] == nil
|
||||||
|
memo[topic.metacode_id] = 1
|
||||||
|
else
|
||||||
|
memo[topic.metacode_id] = memo[topic.metacode_id] + 1
|
||||||
|
end
|
||||||
|
|
||||||
|
memo
|
||||||
|
}.to_a.sort{ |a, b| b[1] <=> a[1] }.map{|i| i[0]}.slice(0, 5)
|
||||||
|
end
|
||||||
|
|
||||||
# generate a random 8 letter/digit code that they can use to invite people
|
# generate a random 8 letter/digit code that they can use to invite people
|
||||||
def generate_code
|
def generate_code
|
||||||
self.code ||= rand(36**8).to_s(36)
|
self.code ||= rand(36**8).to_s(36)
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
class UserPreference
|
class UserPreference
|
||||||
attr_accessor :metacodes
|
attr_accessor :metacodes, :metacode_focus
|
||||||
|
|
||||||
def initialize
|
def initialize
|
||||||
array = []
|
array = []
|
||||||
|
@ -8,5 +8,6 @@ class UserPreference
|
||||||
array.push(metacode.id.to_s) if metacode
|
array.push(metacode.id.to_s) if metacode
|
||||||
end
|
end
|
||||||
@metacodes = array
|
@metacodes = array
|
||||||
|
@metacode_focus = array[0]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,14 +1,20 @@
|
||||||
<%= form_for Topic.new, url: topics_url, remote: true do |form| %>
|
<%= form_for Topic.new, url: topics_url, remote: true do |form| %>
|
||||||
<div class="openMetacodeSwitcher openLightbox" data-open="switchMetacodes"></div>
|
<div class="openMetacodeSwitcher openLightbox" data-open="switchMetacodes"></div>
|
||||||
<div id="metacodeImg">
|
<div id="metacodeImg">
|
||||||
<% @metacodes = user_metacodes() %>
|
<% @metacodes = [user_metacode()].concat(user_metacodes()).uniq %>
|
||||||
<% set = get_metacodeset() %>
|
<% set = get_metacodeset() %>
|
||||||
<% @metacodes.each do |metacode| %>
|
<% @metacodes.each do |metacode| %>
|
||||||
<img class="cloudcarousel" width="40" height="40" src="<%= asset_path metacode.icon %>" alt="<%= metacode.name %>" title="<%= metacode.name %>" data-id="<%= metacode.id %>" />
|
<img class="cloudcarousel" width="40" height="40" src="<%= asset_path metacode.icon %>" alt="<%= metacode.name %>" title="<%= metacode.name %>" data-id="<%= metacode.id %>" />
|
||||||
<% end %>
|
<% end %>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="selectedMetacode">
|
||||||
|
<img src="<%= asset_path user_metacode().icon %>" />
|
||||||
|
<span><%= user_metacode().name %></span>
|
||||||
|
<div class="downArrow"></div>
|
||||||
|
</div>
|
||||||
<%= form.text_field :name, :maxlength => 140, :placeholder => "title..." %>
|
<%= form.text_field :name, :maxlength => 140, :placeholder => "title..." %>
|
||||||
<div id="metacodeImgTitle"></div>
|
<div class="clearfloat"></div>
|
||||||
|
<div id="metacodeSelector"></div>
|
||||||
<div class="clearfloat"></div>
|
<div class="clearfloat"></div>
|
||||||
<script>
|
<script>
|
||||||
<% @metacodes.each do |metacode| %>
|
<% @metacodes.each do |metacode| %>
|
||||||
|
@ -19,5 +25,11 @@
|
||||||
Metamaps.Create.newSelectedMetacodeNames.push("<%= metacode.name %>");
|
Metamaps.Create.newSelectedMetacodeNames.push("<%= metacode.name %>");
|
||||||
<% end %>
|
<% end %>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
<% current_user.recentMetacodes.each do |id| %>
|
||||||
|
Metamaps.Create.recentMetacodes.push(<%= id %>);
|
||||||
|
<% end %>
|
||||||
|
<% current_user.mostUsedMetacodes.each do |id| %>
|
||||||
|
Metamaps.Create.mostUsedMetacodes.push(<%= id %>);
|
||||||
|
<% end %>
|
||||||
</script>
|
</script>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|
|
@ -61,5 +61,6 @@ Metamaps::Application.routes.draw do
|
||||||
|
|
||||||
get 'users/:id/details', to: 'users#details', as: :details
|
get 'users/:id/details', to: 'users#details', as: :details
|
||||||
post 'user/updatemetacodes', to: 'users#updatemetacodes', as: :updatemetacodes
|
post 'user/updatemetacodes', to: 'users#updatemetacodes', as: :updatemetacodes
|
||||||
|
post 'user/update_metacode_focus', to: 'users#update_metacode_focus'
|
||||||
resources :users, except: [:index, :destroy]
|
resources :users, except: [:index, :destroy]
|
||||||
end
|
end
|
||||||
|
|
168
frontend/src/components/MetacodeSelect.js
Normal file
168
frontend/src/components/MetacodeSelect.js
Normal file
|
@ -0,0 +1,168 @@
|
||||||
|
/* global $ */
|
||||||
|
|
||||||
|
import React, { Component, PropTypes } from 'react'
|
||||||
|
|
||||||
|
const ENTER_KEY = 13
|
||||||
|
const LEFT_ARROW = 37
|
||||||
|
const UP_ARROW = 38
|
||||||
|
const RIGHT_ARROW = 39
|
||||||
|
const DOWN_ARROW = 40
|
||||||
|
|
||||||
|
const Metacode = (props) => {
|
||||||
|
const { m, onClick, underCursor } = props
|
||||||
|
|
||||||
|
return (
|
||||||
|
<li onClick={() => onClick(m.id) } className={ underCursor ? 'keySelect' : '' }>
|
||||||
|
<img src={ m.get('icon') } />
|
||||||
|
<span>{ m.get('name') }</span>
|
||||||
|
</li>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
class MetacodeSelect extends Component {
|
||||||
|
|
||||||
|
constructor (props) {
|
||||||
|
super(props)
|
||||||
|
this.state = {
|
||||||
|
filterText: '',
|
||||||
|
activeTab: 0,
|
||||||
|
selectingSection: true,
|
||||||
|
underCursor: 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
const self = this
|
||||||
|
setTimeout(function() {
|
||||||
|
$(document.body).on('keyup.metacodeSelect', self.handleKeyUp.bind(self))
|
||||||
|
}, 10)
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
$(document.body).off('.metacodeSelect')
|
||||||
|
}
|
||||||
|
|
||||||
|
changeFilterText (e) {
|
||||||
|
this.setState({ filterText: e.target.value, underCursor: 0 })
|
||||||
|
}
|
||||||
|
|
||||||
|
changeDisplay (activeTab) {
|
||||||
|
this.setState({ activeTab, underCursor: 0 })
|
||||||
|
}
|
||||||
|
|
||||||
|
getSelectMetacodes () {
|
||||||
|
const { metacodes, recent, mostUsed } = this.props
|
||||||
|
const { filterText, activeTab } = this.state
|
||||||
|
|
||||||
|
let selectMetacodes = []
|
||||||
|
if (activeTab == 0) { // search
|
||||||
|
selectMetacodes = filterText.length > 1 ? metacodes.filter(m => {
|
||||||
|
return m.get('name').toLowerCase().search(filterText.toLowerCase()) > -1
|
||||||
|
}) : []
|
||||||
|
} else if (activeTab == 1) { // recent
|
||||||
|
selectMetacodes = recent.map(id => {
|
||||||
|
return metacodes.find(m => m.id == id)
|
||||||
|
})
|
||||||
|
} else if (activeTab == 2) { // mostUsed
|
||||||
|
selectMetacodes = mostUsed.map(id => {
|
||||||
|
return metacodes.find(m => m.id == id)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return selectMetacodes
|
||||||
|
}
|
||||||
|
|
||||||
|
handleKeyUp (e) {
|
||||||
|
const { close } = this.props
|
||||||
|
const { activeTab, underCursor, selectingSection } = this.state
|
||||||
|
const selectMetacodes = this.getSelectMetacodes()
|
||||||
|
let nextIndex
|
||||||
|
|
||||||
|
switch (e.which) {
|
||||||
|
case ENTER_KEY:
|
||||||
|
if (selectMetacodes.length && !selectingSection) this.resetAndClick(selectMetacodes[underCursor].id)
|
||||||
|
break
|
||||||
|
case UP_ARROW:
|
||||||
|
if (selectingSection && activeTab == 0) {
|
||||||
|
close()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
else if (selectingSection) {
|
||||||
|
nextIndex = activeTab - 1
|
||||||
|
this.changeDisplay(nextIndex)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
nextIndex = underCursor == 0 ? selectMetacodes.length - 1 : underCursor - 1
|
||||||
|
this.setState({ underCursor: nextIndex })
|
||||||
|
break
|
||||||
|
case DOWN_ARROW:
|
||||||
|
if (selectingSection) {
|
||||||
|
nextIndex = activeTab == 2 ? 0 : activeTab + 1
|
||||||
|
this.changeDisplay(nextIndex)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
nextIndex = underCursor == selectMetacodes.length - 1 ? 0 : underCursor + 1
|
||||||
|
this.setState({ underCursor: nextIndex })
|
||||||
|
break
|
||||||
|
case RIGHT_ARROW:
|
||||||
|
if (selectingSection) this.setState({ selectingSection: false })
|
||||||
|
break
|
||||||
|
case LEFT_ARROW:
|
||||||
|
if (!selectingSection) this.setState({ selectingSection: true })
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
resetAndClick (id) {
|
||||||
|
const { onClick } = this.props
|
||||||
|
this.setState({ filterText: '', underCursor: 0 })
|
||||||
|
this.changeDisplay(0)
|
||||||
|
onClick(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
render () {
|
||||||
|
const { onClick, close, recent, mostUsed } = this.props
|
||||||
|
const { filterText, activeTab, underCursor, selectingSection } = this.state
|
||||||
|
const selectMetacodes = this.getSelectMetacodes()
|
||||||
|
return <div className='metacodeSelect'>
|
||||||
|
<div className='tabList'>
|
||||||
|
<div className={ activeTab == 0 ? 'active' : '' }
|
||||||
|
onClick={() => { this.changeDisplay(0) }}>
|
||||||
|
<input type='text'
|
||||||
|
className='metacodeFilterInput'
|
||||||
|
placeholder='Search...'
|
||||||
|
ref='input'
|
||||||
|
value={ filterText }
|
||||||
|
onChange={ this.changeFilterText.bind(this) } />
|
||||||
|
</div>
|
||||||
|
<div className={ activeTab == 1 ? 'active' : '' }
|
||||||
|
onClick={() => { this.changeDisplay(1) }}>
|
||||||
|
<span>Recent</span>
|
||||||
|
</div>
|
||||||
|
<div className={ activeTab == 2 ? 'active' : '' }
|
||||||
|
onClick={() => { this.changeDisplay(2) }}>
|
||||||
|
<span>Most Used</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<ul className='metacodeList'>
|
||||||
|
{ selectMetacodes.map((m, index) => {
|
||||||
|
return <Metacode underCursor={!selectingSection && underCursor == index}
|
||||||
|
key={m.id}
|
||||||
|
m={m}
|
||||||
|
onClick={this.resetAndClick.bind(this)} />
|
||||||
|
})}
|
||||||
|
</ul>
|
||||||
|
<div className='clearfloat'></div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
MetacodeSelect.propTypes = {
|
||||||
|
onClick: PropTypes.func.isRequired,
|
||||||
|
close: PropTypes.func.isRequired,
|
||||||
|
metacodes: PropTypes.array.isRequired,
|
||||||
|
recent: PropTypes.array.isRequired,
|
||||||
|
mostUsed: PropTypes.array.isRequired
|
||||||
|
}
|
||||||
|
|
||||||
|
export default MetacodeSelect
|
||||||
|
|
|
@ -3,6 +3,7 @@ import ReactDOM from 'react-dom'
|
||||||
import Backbone from 'backbone'
|
import Backbone from 'backbone'
|
||||||
import _ from 'underscore'
|
import _ from 'underscore'
|
||||||
import Maps from './components/Maps.js'
|
import Maps from './components/Maps.js'
|
||||||
|
import MetacodeSelect from './components/MetacodeSelect.js'
|
||||||
|
|
||||||
// this is optional really, if we import components directly React will be
|
// this is optional really, if we import components directly React will be
|
||||||
// in the bundle, so we won't need a global reference
|
// in the bundle, so we won't need a global reference
|
||||||
|
@ -14,5 +15,6 @@ window._ = _
|
||||||
|
|
||||||
window.Metamaps = window.Metamaps || {}
|
window.Metamaps = window.Metamaps || {}
|
||||||
window.Metamaps.ReactComponents = {
|
window.Metamaps.ReactComponents = {
|
||||||
Maps
|
Maps,
|
||||||
|
MetacodeSelect
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue