Merge branch 'master' into arunoda-fr2

Conflicts:
	lib/router/users.js
This commit is contained in:
Sacha Greif 2014-12-08 10:30:31 +09:00
commit 4f9c8295d6
152 changed files with 2440 additions and 4925 deletions

1
.gitignore vendored
View file

@ -19,6 +19,7 @@ dump/
dump/*
settings.json
settings.json.not-used
.idea
scratch

View file

@ -24,7 +24,7 @@ meteorhacks:fast-render
meteorhacks:subs-manager
meteorhacks:npm
aldeed:autoform
aldeed:autoform@4.0.0-rc1
aldeed:collection2
aldeed:simple-schema
@ -64,7 +64,7 @@ telescope-search
telescope-tags
telescope-theme-hubble
telescope-email
telescope-module-embedly
telescope-embedly
telescope-newsletter
telescope-daily
telescope-update-prompt
@ -74,4 +74,6 @@ telescope-notifications
# Accounts Templates
useraccounts:unstyled
telescope-datetimepicker
tsega:bootstrap3-datetimepicker@3.1.3_1

View file

@ -66,7 +66,7 @@ matb33:collection-hooks@0.7.6
meteor-platform@1.2.0
meteor@1.1.3
meteorhacks:async@1.0.0
meteorhacks:fast-render@2.0.2
meteorhacks:fast-render@1.2.1
meteorhacks:kadira-binary-deps@1.3.1
meteorhacks:kadira@2.14.0
meteorhacks:meteorx@1.2.0

View file

@ -1,3 +1,19 @@
## v0.10.0
* Renaming Errors to Messages (thanks @yourcelf!).
## v0.9.11 “FormScope”
* Now using [Autoform](https://github.com/aldeed/meteor-autoform/)'s **quickform** feature to generate post submit and edit forms.
* Various fixes by [@anthonymayer](https://github.com/anthonymayer).
* Now using [fourseven:scss](https://github.com/fourseven/meteor-scss) to directly compile SCSS files.
* Renamed `post` method to `submitPost`.
* Post editing now happens via a `postEdit` method.
* Categories are now normalized (only the `_id` is stored on the post object, not the whole category object).
* Refactored Embedly package; now fills in description as well (thanks [@kvindasAB](https://github.com/kvindasAB)!).
* Thumbnail height and width are now customizable in settings panel.
* Settings and Post forms now i18n'ized.
## v0.9.10 “i18nScope”
* Now using [tap:i18n](https://github.com/TAPevents/tap-i18n) for internationalization (thanks a ton to @theosp).

View file

@ -2,20 +2,6 @@ Accounts.ui.config({
passwordSignupFields: 'USERNAME_AND_EMAIL'
});
EpicEditorOptions={
container: 'editor',
basePath: '/editor',
clientSideStorage: false,
autogrow: {
minHeight: 100
},
theme: {
base:'/themes/base/epiceditor.css',
preview:'/themes/preview/github.css',
editor:'/themes/editor/epic-light3.css'
}
};
SharrreOptions={
share: {
googlePlus: true,

View file

@ -42,7 +42,7 @@ UI.registerHelper('isAdmin', function(showError) {
return true;
}else{
if((typeof showError === "string") && (showError === "true"))
throwError(i18n.t('sorry_you_do_not_have_access_to_this_page'));
flashMessage(i18n.t('sorry_you_do_not_have_access_to_this_page'), "error");
return false;
}
});
@ -78,4 +78,4 @@ UI.registerHelper("sanitize", function(content) {
UI.registerHelper('pluralize', function(count, string) {
string = count === 1 ? string : string + 's';
return i18n.t(string);
});
});

File diff suppressed because it is too large Load diff

View file

@ -2,5 +2,5 @@
<title>Loading...</title>
<meta name="viewport" content="initial-scale=1.0">
<link rel="shortcut icon" href="/img/favicon.ico"/>
<link id="rss-link" rel="alternate" type="application/rss+xml" title="New Posts" href="/feed.xml"/>
<link id="rss-link" rel="alternate" type="application/rss+xml" title="New Posts" href="{{pathFor route='feed'}}"/>
</head>

View file

@ -37,4 +37,6 @@ postMeta = _.sortBy(postMeta, 'order');
Meteor.startup(function () {
$('#rss-link').attr('title', i18n.t('new_posts'));
});
});
// AutoForm.debug();

View file

@ -1,9 +1,9 @@
<template name="settings">
<div class="grid-small grid-module dialog settings">
{{#if this.hasSettings}}
{{> quickForm collection="Settings" id="updateSettingsForm" type="update" doc=this.settings label-class="control-label" input-col-class="controls" template="settings"}}
{{> quickForm collection="Settings" id="updateSettingsForm" type="update" doc=this.settings label-class="control-label" input-col-class="controls" template="telescope"}}
{{else}}
{{> quickForm collection="Settings" id="updateSettingsForm" type="insert" template="settings" label-class="control-label" input-col-class="controls"}}
{{> quickForm collection="Settings" id="updateSettingsForm" type="insert" template="telescope" label-class="control-label" input-col-class="controls"}}
{{/if}}
</div>
</template>

View file

@ -1,21 +1,7 @@
Template[getTemplate('comment_edit')].rendered = function(){
if(this.data){ // XXX
var comment = this.data.comment;
if(comment && Meteor.user() && !this.editor){
this.editor = new EpicEditor(EpicEditorOptions).load();
this.editor.importFile('editor', comment.body);
$(this.editor.editor).bind('keydown', 'meta+return', function(){
$(window.editor).closest('form').find('input[type="submit"]').click();
});
}
}
};
Template[getTemplate('comment_edit')].events({
'click input[type=submit]': function(e, instance){
var comment = this;
var content = cleanUp(instance.editor.exportFile());
var content = instance.$('#body').val();
e.preventDefault();
@ -29,18 +15,17 @@ Template[getTemplate('comment_edit')].events({
});
trackEvent("edit comment", {'postId': comment.postId, 'commentId': comment._id});
Router.go("/posts/"+comment.postId+"/comment/"+comment._id);
Router.go('post_page_comment', {_id: comment.postId, commentId: comment._id});
},
'click .delete-link': function(e){
var comment = this;
e.preventDefault();
if(confirm(i18n.t("are_you_sure"))){
Meteor.call('removeComment', comment._id);
Router.go("/posts/"+comment.postId);
throwError("Your comment has been deleted.");
// Router.go("/comments/deleted");
Router.go('post_page', {_id: comment.postId});
flashMessage("Your comment has been deleted.", "success");
}
}
});
});

View file

@ -6,7 +6,7 @@
<textarea id="comment" rows="3" autofocus="autofocus"></textarea>
</div>
<div class="comment-submit">
<input type="submit" class="button" value="{{_ "add_comment"}}" title="(⌘+enter)" />
<input type="submit" class="btn btn-primary" value="{{_ "add_comment"}}" title="(⌘+enter)" />
</div>
</form>
</div>

View file

@ -4,31 +4,26 @@ Template[getTemplate('comment_form')].helpers({
}
});
Template[getTemplate('comment_form')].rendered = function(){
if(Meteor.user() && !this.editor){
this.editor = new EpicEditor(EpicEditorOptions).load();
$(this.editor.editor).bind('keydown', 'meta+return', function(){
$(window.editor).closest('form').find('input[type="submit"]').click();
});
}
};
Template[getTemplate('comment_form')].events({
'submit form': function(e, instance){
var $commentForm = instance.$('#comment');
e.preventDefault();
$(e.target).addClass('disabled');
clearSeenErrors();
var content = instance.editor.exportFile();
clearSeenMessages();
var content = $commentForm.val();
if(getCurrentTemplate() == 'comment_reply'){
// child comment
var parentComment = this.comment;
Meteor.call('comment', parentComment.postId, parentComment._id, content, function(error, newComment){
if(error){
console.log(error);
throwError(error.reason);
flashMessage(error.reason, "error");
}else{
trackEvent("newComment", newComment);
Router.go('/posts/'+parentComment.postId+'/comment/'+newComment._id);
Router.go('post_page_comment', {
_id: parentComment.postId,
commentId: newComment._id
});
}
});
}else{
@ -38,13 +33,13 @@ Template[getTemplate('comment_form')].events({
Meteor.call('comment', post._id, null, content, function(error, newComment){
if(error){
console.log(error);
throwError(error.reason);
flashMessage(error.reason, "error");
}else{
trackEvent("newComment", newComment);
Session.set('scrollToCommentId', newComment._id);
instance.editor.importFile('editor', '');
$commentForm.val('');
}
});
}
}
});
});

View file

@ -21,9 +21,9 @@
<a class="comment-username" href="{{profileUrl}}">{{authorName}}</a>
<span class="comment-time">{{timeAgo ago}},</span>
<span class="points">{{upvotes}}</span> <span class="unit">points </span>
<a href="/comments/{{_id}}" class="comment-permalink icon-link goto-comment">{{_ "link"}}</a>
<a href="{{pathFor route='comment_reply' _id=_id}}" class="comment-permalink icon-link goto-comment">{{_ "link"}}</a>
{{#if can_edit}}
| <a class="edit-link" href="/comments/{{_id}}/edit">{{_ "edit"}}</a>
| <a class="edit-link" href="{{pathFor route='comment_edit' _id=_id}}">{{_ "edit"}}</a>
{{/if}}
{{#if isAdmin}}
| <span>{{full_date}}</span>
@ -31,7 +31,7 @@
</div>
<div class="comment-text markdown">{{{htmlBody}}}</div>
{{#if getSetting "nestedComments" true}}
<a href="/comments/{{_id}}" class="comment-reply goto-comment">{{_ "reply"}}</a>
<a href="{{pathFor route='comment_reply' _id=_id}}" class="comment-reply goto-comment">{{_ "reply"}}</a>
{{/if}}
</div>
</div>

View file

@ -153,7 +153,7 @@ Template[getTemplate('comment_item')].events({
e.preventDefault();
if(!Meteor.user()){
Router.go(getSigninUrl());
throwError(i18n.t("please_log_in_first"));
flashMessage(i18n.t("please_log_in_first"), "info");
}
Meteor.call('upvoteComment', this, function(error, result){
trackEvent("post upvoted", {'commentId':instance.data._id, 'postId': instance.data.post, 'authorId':instance.data.userId});
@ -163,7 +163,7 @@ Template[getTemplate('comment_item')].events({
e.preventDefault();
if(!Meteor.user()){
Router.go(getSigninUrl());
throwError(i18n.t("please_log_in_first"));
flashMessage(i18n.t("please_log_in_first"), "info");
}
Meteor.call('cancelUpvoteComment', this, function(error, result){
trackEvent("post upvote cancelled", {'commentId':instance.data._id, 'postId': instance.data.post, 'authorId':instance.data.userId});
@ -173,7 +173,7 @@ Template[getTemplate('comment_item')].events({
e.preventDefault();
if(!Meteor.user()){
Router.go(getSigninUrl());
throwError(i18n.t("please_log_in_first"));
flashMessage(i18n.t("please_log_in_first"), "info");
}
Meteor.call('downvoteComment', this, function(error, result){
trackEvent("post downvoted", {'commentId':instance.data._id, 'postId': instance.data.post, 'authorId':instance.data.userId});
@ -183,7 +183,7 @@ Template[getTemplate('comment_item')].events({
e.preventDefault();
if(!Meteor.user()){
Router.go(getSigninUrl());
throwError(i18n.t("please_log_in_first"));
flashMessage(i18n.t("please_log_in_first"), "info");
}
Meteor.call('cancelDownvoteComment', this, function(error, result){
trackEvent("post downvote cancelled", {'commentId':instance.data._id, 'postId': instance.data.post, 'authorId':instance.data.userId});

View file

@ -3,7 +3,7 @@
body{
background: {{getSetting "backgroundCSS"}};
}
input[type="submit"], button, .button, .auth-buttons #login-buttons #login-buttons-password, .btn, .error, .mobile-menu-button, .login-link-text{
input[type="submit"], button, .button, .auth-buttons #login-buttons #login-buttons-password, .btn-primary, .error, .mobile-menu-button, .login-link-text{
background-color: {{getSetting "buttonColor"}} !important;
color: {{getSetting "buttonTextColor"}} !important;
}
@ -21,8 +21,8 @@
color: {{getSetting "headerTextColor"}};
}
.logo-image a{
height:{{getSetting "logoHeight"}}px;
width:{{getSetting "logoWidth"}}px;
height:{{getSetting "logoHeight"}}px;
width:{{getSetting "logoWidth"}}px;
}
</style>
</template>

View file

@ -1,5 +0,0 @@
<template name="error">
{{#each errors}}
{{> UI.dynamic template=error_item}}
{{/each}}
</template>

View file

@ -1,8 +0,0 @@
Template[getTemplate('error')].helpers({
error_item: function () {
return getTemplate('error_item');
},
errors: function(){
return Errors.find({show: true});
}
});

View file

@ -1,10 +0,0 @@
Template[getTemplate('error_item')].helpers({
});
Template[getTemplate('error_item')].created = function(){
var error_id=this.data._id;
Meteor.setTimeout(function(){
Errors.update(error_id, {$set: {seen:true}});
}, 100);
};

View file

@ -6,7 +6,7 @@
<div class="inner-wrapper template-{{pageName}}">
{{> UI.dynamic template=nav}}
<div class="content-wrapper">
{{> UI.dynamic template=error}}
{{> UI.dynamic template=messages}}
{{#each heroModules}}
{{> UI.dynamic template=getTemplate}}
{{/each}}

View file

@ -5,8 +5,8 @@ Template[getTemplate('layout')].helpers({
nav: function () {
return getTemplate('nav');
},
error: function () {
return getTemplate('error');
messages: function () {
return getTemplate('messages');
},
notifications: function () {
return getTemplate('notifications');

View file

@ -1,4 +1,4 @@
<template name="error_item">
<template name="message_item">
{{#if show}}
<div class="grid">
<div class="error {{type}}-message module">

View file

@ -0,0 +1,10 @@
Template[getTemplate('message_item')].helpers({
});
Template[getTemplate('message_item')].created = function(){
var messageId=this.data._id;
Meteor.setTimeout(function(){
Messages.update(messageId, {$set: {seen:true}});
}, 100);
};

View file

@ -0,0 +1,5 @@
<template name="messages">
{{#each messages}}
{{> UI.dynamic template=message_item}}
{{/each}}
</template>

View file

@ -0,0 +1,8 @@
Template[getTemplate('messages')].helpers({
message_item: function () {
return getTemplate('message_item');
},
messages: function(){
return Messages.find({show: true});
}
});

View file

@ -0,0 +1,112 @@
<template name="quickForm_telescope">
{{#autoForm qfAutoFormContext}}
<div class='no-fieldset'>
{{#each fieldsWithNoFieldset}}
{{> afQuickField name=this template="telescope" input-col-class=inputClass label-class=labelClass}}
{{/each}}
</div>
{{#each afFieldsets}}
<fieldset>
<h3 class="fieldset-heading">{{fieldsetName}}</h3>
{{> afQuickFields fields=fieldsForFieldset template="telescope" input-col-class=inputClass label-class=labelClass}}
</fieldset>
{{/each}}
{{#if qfShouldRenderButton}}
<div class="form-group">
<div class="{{labelClass}}"></div>
<div class="{{inputClass}}">
<button {{submitButtonAtts}}>
{{#with ../atts.buttonContent}}
{{this}}
{{else}}
Submit
{{/with}}
</button>
</div>
</div>
{{/if}}
{{/autoForm}}
</template>
<template name="afFormGroup_telescope">
{{#if showField}}
<div class="form-group {{#if afFieldIsInvalid name=this.atts.name}}has-error{{/if}}">
<label class="control-label">
{{_ this.atts.name}}
{{#if fieldIsPrivate}}
<span class="private-field" title="{{_ 'Private'}}">(p)</span>
{{/if}}
</label>
<div class="{{rightColumnClass}}">
{{> afFieldInput afFieldInputAtts}}
{{#if afFieldInstructions}}
<span class="instructions-block">{{afFieldInstructions}}</span>
{{/if}}
<span class="help-block">{{{afFieldMessage name=this.atts.name}}}</span>
</div>
</div>
{{/if}}
</template>
<template name="afObjectField_telescope">
{{#if showField}}
<div class="form-group {{#if afFieldIsInvalid name=this.atts.name}}has-error{{/if}}">
<label class="control-label">{{_ this.atts.name}}</label>
<div class="{{rightColumnClass}}">
<div class="panel panel-default autoform-padding-fix">
<div class="panel-body">
{{> afQuickFields name=this.atts.name fields=this.atts.fields omitFields=this.atts.omitFields template="bootstrap3"}}
</div>
</div>
</div>
</div>
{{/if}}
</template>
<template name="afArrayField_telescope">
{{#if showField}}
<div class="form-group {{#if afFieldIsInvalid name=this.atts.name}}has-error{{/if}}">
<label class="control-label">{{_ this.atts.name}}</label>
<div class="{{rightColumnClass}}">
<div class="panel panel-default autoform-padding-fix">
{{#if afFieldIsInvalid name=this.atts.name}}
<div class="panel-body has-error">
<span class="help-block">{{{afFieldMessage name=this.atts.name}}}</span>
</div>
{{/if}}
<ul class="list-group">
{{#afEachArrayItem name=this.atts.name minCount=this.atts.minCount maxCount=this.atts.maxCount}}
<li class="list-group-item autoform-array-item">
<div class="media">
{{#if afArrayFieldHasMoreThanMinimum name=../atts.name minCount=../atts.minCount maxCount=../atts.maxCount}}
<button class="btn btn-primary autoform-remove-item pull-left"><span class="glyphicon glyphicon-minus"></span></button>
{{/if}}
<div class="media-body">
{{> afQuickField name=this.name label=false}}
</div>
</div>
</li>
{{/afEachArrayItem}}
{{#if afArrayFieldHasLessThanMaximum name=this.atts.name minCount=this.atts.minCount maxCount=this.atts.maxCount}}
<li class="list-group-item">
<button class="btn btn-primary autoform-add-item" data-autoform-field="{{this.atts.name}}" data-autoform-minCount="{{this.atts.minCount}}" data-autoform-maxCount="{{this.atts.maxCount}}"><span class="glyphicon glyphicon-plus"></span></button>
</li>
{{/if}}
</ul>
</div>
</div>
</div>
{{/if}}
</template>
<template name="afCheckbox_telescope">
<div class="checkbox">
<label>
<input type="checkbox" value="{{this.value}}" {{atts}} />aaaa
</label>
</div>
</template>

View file

@ -1,4 +1,4 @@
function findAtts() {
var findAtts = function () {
var c, n = 0;
do {
c = UI._parentData(n++);
@ -6,28 +6,55 @@ function findAtts() {
return c && c.atts;
}
Template[getTemplate('quickForm_settings')].helpers({
var getSchema = function () {
var schema = AutoForm.find().ss._schema;
// decorate schema with key names
schema = _.map(schema, function (field, key) {
field.name = key;
return field;
});
return schema;
}
var canEditField = function (field) {
// show field only if user is admin or it's marked as editable
return isAdmin(Meteor.user()) || !!field.atts.editable || (!!field.afFieldInputAtts && !!field.afFieldInputAtts.editable)
}
Template[getTemplate('quickForm_telescope')].helpers({
fieldsWithNoFieldset: function () {
// get names of fields who don't have an autoform attribute or don't have a group, but are not omitted
var fields = _.pluck(_.filter(getSchema(), function (field, key) {
if (field.name.indexOf('$') !== -1) // filter out fields with "$" in their name
return false
if (field.autoform && field.autoform.omit) // filter out fields with omit = true
return false
if (field.autoform && field.autoform.group) // filter out fields with a group
return false
return true // return remaining fields
}), 'name');
return fields;
},
afFieldsets: function () {
var schema = this._af.ss._schema;
var groups = _.compact(_.uniq(_.pluckDeep(schema, 'autoform.group')));
groups = groups.map(function (group) {
return capitalise(group);
});
var groups = _.compact(_.uniq(_.pluckDeep(getSchema(), 'autoform.group')));
// if user is not admin, exclude "admin" group from fieldsets
if (!isAdmin(Meteor.user()))
groups = _.without(groups, 'admin')
return groups;
},
fieldsetName: function () {
return capitalise(i18n.t(this));
},
fieldsForFieldset: function () {
var fieldset = this.toLowerCase();
var schema = AutoForm.find().ss._schema;
// decorate schema with key names
schema = _.map(schema, function (field, key) {
field.name = key;
return field;
});
// get names of fields whose group match the current fieldset
var fields = _.pluck(_.filter(schema, function (field, key) {
return field.autoform && field.autoform.group == fieldset;
var fields = _.pluck(_.filter(getSchema(), function (field, key) {
return (field.name.indexOf('$') === -1) && field.autoform && field.autoform.group == fieldset;
}), 'name');
return fields;
@ -69,7 +96,7 @@ Template[getTemplate('quickForm_settings')].helpers({
}
});
Template["afFormGroup_settings"].helpers({
Template["afFormGroup_telescope"].helpers({
afFieldInputAtts: function () {
var atts = _.clone(this.afFieldInputAtts || {});
if ('input-col-class' in atts) {
@ -94,19 +121,22 @@ Template["afFormGroup_settings"].helpers({
}
return labelAtts;
},
fieldIsPrivate: function () {
return !!this.afFieldInputAtts.private;
},
rightColumnClass: function () {
var atts = this.afFieldInputAtts || {};
return atts['input-col-class'] || "";
},
showField: function () {
return "showField" in this.afFieldInputAtts ? this.afFieldInputAtts.showField : true;
return canEditField(this);
},
afFieldInstructions: function () {
return this.afFieldInputAtts.instructions;
}
});
Template["afObjectField_settings"].helpers({
Template["afObjectField_telescope"].helpers({
rightColumnClass: function () {
var atts = this.atts || {};
return atts['input-col-class'] || "";
@ -118,10 +148,13 @@ Template["afObjectField_settings"].helpers({
"class": atts["label-class"],
"name": atts.name
}
}
},
showField: function () {
return canEditField(this);
},
});
Template["afArrayField_settings"].helpers({
Template["afArrayField_telescope"].helpers({
rightColumnClass: function () {
var atts = this.atts || {};
return atts['input-col-class'] || "";
@ -133,5 +166,8 @@ Template["afArrayField_settings"].helpers({
"class": atts["label-class"],
"name": atts.name
};
}
},
showField: function () {
return canEditField(this);
},
});

View file

@ -0,0 +1,3 @@
<template name="afBootstrapUrl_bootstrap3">
<input type="text" value="{{this.value}}" {{this.atts}}/>
</template>

View file

@ -0,0 +1,19 @@
AutoForm.addInputType("bootstrap-url", {
template: "afBootstrapUrl",
valueOut: function () {
var url = this.val();
if(!!url)
return (url.substring(0, 7) == "http://" || url.substring(0, 8) == "https://") ? url : "http://"+url;
}
});
Template.afBootstrapUrl_bootstrap3.helpers({
atts: function addFormControlAtts() {
console.log(this)
var atts = _.clone(this.atts);
// Add bootstrap class
atts = AutoForm.Utility.addClass(atts, "form-control");
return atts;
}
});

View file

@ -28,7 +28,7 @@ Template[getTemplate('nav')].helpers({
},
logo_top: function(){
return Math.floor((70-getSetting('logoHeight'))/2);
},
},
logo_offset: function(){
return -Math.floor(getSetting('logoWidth')/2);
},
@ -43,30 +43,9 @@ Template[getTemplate('nav')].helpers({
}
});
Template[getTemplate('nav')].rendered = function(){
if(!Meteor.loggingIn() && !Meteor.user()){
$('.login-link-text').text("Sign Up/Sign In");
}
};
Template[getTemplate('nav')].events({
'click #logout': function(e){
e.preventDefault();
Meteor.logout();
},
'click .mobile-menu-button': function(e){
e.preventDefault();
$('body').toggleClass('mobile-nav-open');
},
'click .login-header': function(e){
e.preventDefault();
Router.go('/account');
},
'click #login-name-link': function(){
if(Meteor.user() && !$('account-link').exists()){
var $loginButtonsLogout = $('#login-buttons-logout');
$loginButtonsLogout.before('<a href="/users/'+Meteor.user().slug+'" class="account-link button">View Profile</a>');
$loginButtonsLogout.before('<a href="/account" class="account-link button">Edit Account</a>');
}
}
});

View file

@ -1,5 +1,5 @@
<template name="submitButton">
{{#if canPost}}
<a id="submit" class="submit button header-submodule" href="/submit">{{_ "post"}}</a>
<a id="submit" class="submit btn header-submodule btn-primary" href="/submit">{{_ "post"}}</a>
{{/if}}
</template>

View file

@ -5,13 +5,13 @@
<div class="dropdown-menu">
<ul role="menu" aria-labelledby="dLabel">
<li><a href="{{profileUrl}}">{{_ "profile"}}</a></li>
<li><a href="/account">{{_ "edit_account"}}</a></li>
<li><a href="/sign-out">{{_ "sign_out"}}</a></li>
<li><a href="{{userEditUrl}}">{{_ "edit_account"}}</a></li>
<li><a href="{{pathFor route='signOut'}}">{{_ "sign_out"}}</a></li>
</ul>
</div>
</div>
{{else}}
<a class="account-link sign-in" href="/sign-in">{{_ "sign_in"}}</a>
<a class="account-link sign-up" href="/sign-up">{{_ "sign_up"}}</a>
<a class="account-link sign-in" href="{{pathFor route='atSignIn'}}">{{_ "sign_in"}}</a>
<a class="account-link sign-up" href="{{pathFor route='atSignUp'}}">{{_ "sign_up"}}</a>
{{/if}}
</template>

View file

@ -7,5 +7,8 @@ Template[getTemplate('userMenu')].helpers({
},
profileUrl: function () {
return getProfileUrl(Meteor.user());
},
userEditUrl: function () {
return Router.routes['user_edit'].path(Meteor.user());
}
});

View file

@ -8,4 +8,15 @@ Template[getTemplate('postAdmin')].helpers({
shortScore: function(){
return Math.floor(this.score*1000)/1000;
}
});
Template[getTemplate('postAdmin')].events({
'click .approve-link': function(e, instance){
Meteor.call('approvePost', this);
e.preventDefault();
},
'click .unapprove-link': function(e, instance){
Meteor.call('unapprovePost', this);
e.preventDefault();
}
});

View file

@ -0,0 +1,3 @@
<template name="postAuthor">
By <a class="post-author" href="{{profileUrl}}">{{authorName}}</a>,
</template>

View file

@ -0,0 +1,11 @@
Template[getTemplate('postAuthor')].helpers({
authorName: function(){
return getAuthorName(this);
},
profileUrl: function(){
// note: we don't want the post to be re-rendered every time user properties change
var user = Meteor.users.findOne(this.userId, {reactive: false});
if(user)
return getProfileUrl(user);
}
})

View file

@ -30,15 +30,4 @@ Template[getTemplate('postContent')].helpers({
commentsDisplayText: function(){
return this.comments == 1 ? i18n.t('comment') : i18n.t('comments');
}
});
Template[getTemplate('postContent')].events({
'click .approve-link': function(e, instance){
Meteor.call('approvePost', this);
e.preventDefault();
},
'click .unapprove-link': function(e, instance){
Meteor.call('unapprovePost', this);
e.preventDefault();
}
});

View file

@ -1,8 +1,10 @@
<template name="postInfo">
<div class="post-meta-item">
<span class="points">{{baseScore}} <span class="unit">{{pointsUnitDisplayText}} </span></span>by <a class="post-author" href="{{profileUrl}}">{{authorName}}</a> <span class="post-time">{{timeAgo ago}}</span>
<span class="points">{{baseScore}}</span>
<span class="unit">{{pointsUnitDisplayText}}</span>
<span class="post-time">{{timeAgo ago}}</span>
{{#if can_edit}}
| <a href="/posts/{{_id}}/edit" class="edit-link goto-edit">{{_ "edit"}}</a>
| <a href="{{pathFor route='post_edit' _id=_id}}" class="edit-link goto-edit">{{_ "edit"}}</a>
{{/if}}
</div>
</template>
</template>

View file

@ -5,18 +5,12 @@ Template[getTemplate('postInfo')].helpers({
can_edit: function(){
return canEdit(Meteor.user(), this);
},
authorName: function(){
return getAuthorName(this);
},
ago: function(){
// if post is approved show submission time, else show creation time.
// if post is approved show submission time, else show creation time.
time = this.status == STATUS_APPROVED ? this.postedAt : this.createdAt;
return time;
},
profileUrl: function(){
// note: we don't want the post to be re-rendered every time user properties change
var user = Meteor.users.findOne(this.userId, {reactive: false});
if(user)
return getProfileUrl(user);
getTemplate: function() {
return getTemplate("postAuthor");
}
});

View file

@ -16,7 +16,7 @@ Template[getTemplate('postUpvote')].events({
e.preventDefault();
if(!Meteor.user()){
Router.go(getSigninUrl());
throwError(i18n.t("please_log_in_first"));
flashMessage(i18n.t("please_log_in_first"), "info");
}
Meteor.call('upvotePost', post, function(error, result){
trackEvent("post upvoted", {'_id': post._id});

View file

@ -1,94 +1,7 @@
<template name="post_edit">
<div class="grid submit">
{{#with post}}
<form class="grid-block form-horizontal">
<div class="control-group">
<label class="control-label post-form-headline">{{_ "created"}}</label>
<div class="controls"><p>{{created}}</p></div>
</div>
<div class="control-group">
<label class="control-label post-form-headline">{{_ "title"}}</label>
<div class="controls"><input id="title" type="text" value="{{title}}" /></div>
</div>
<div class="control-group">
<label class="control-label post-form-url">{{_ "url"}}</label>
<div class="controls"><input id="url" type="text" value="{{url}}" /></div>
</div>
{{#if shorten}}
<div class="control-group">
<label class="control-label post-form-url">{{_ "short_url"}}</label>
<div class="controls"><input id="short-url" type="text" value="{{shortUrl}}" /></div>
</div>
{{/if}}
<div class="control-group">
<label class="control-label post-form-body">{{_ "body"}}</label>
<div class="controls" id="editor"><textarea id="body" value="" class="input-xlarge">{{body}}</textarea></div>
</div>
{{#if categoriesEnabled}}
<div class="control-group post-form-category">
<label class="control-label">{{_ "categories"}}</label>
<div class="controls">
{{#each categories}}
<label class="radio inline">
<input id="category_{{_id}}" type="checkbox" value="{{_id}}" name="category" {{checked}} /> {{name}}
</label>
{{/each}}
</div>
</div>
{{/if}}
{{#if isAdmin}}
<div class="control-group post-form-sticky">
<label class="control-label">{{_ "inactive_"}}</label>
<div class="controls">
{{inactive}}
</div>
</div>
<div class="control-group post-form-sticky">
<label class="control-label">{{_ "sticky_"}}</label>
<div class="controls">
<input type="checkbox" name="sticky" id="sticky" {{isSticky}} />
</div>
</div>
<div class="control-group post-form-user">
<label class="control-label">{{_ "user"}}</label>
<div class="controls">
<select id="postUser">
{{#each users}}
<option {{isSelected ..}} value="{{_id}}">{{userName}}</option>
{{/each}}
</select>
</div>
</div>
<div class="control-group post-form-status">
<label class="control-label">{{_ "status_"}}</label>
<div class="controls">
<label class="radio inline">
<input id="status_pending" type="radio" value="1" name="status" {{hasStatusPending}} /> {{_ "pending"}}
</label>
<label class="radio inline">
<input id="status_approved" type="radio" value="2" name="status" {{hasStatusApproved}} /> {{_ "approved"}}
</label>
<label class="radio inline">
<input id="status_rejected" type="radio" value="3" name="status" {{hasStatusRejected}} /> {{_ "rejected"}}
</label>
</div>
</div>
<div id="postedAt" class={{showPostedAt}}>
<div class="control-group">
<label class="control-label post-form-date">{{_ "posted_date"}}</label>
<div class="controls"><input id="postedAtDate" type="text" value="{{postedAtDate}}" /></div>
</div>
<div class="control-group">
<label class="control-label post-form-time">{{_ "posted_time"}}</label>
<div class="controls"><input id="postedAtTime" type="text" value="{{postedAtTime}}" /></div>
</div>
</div>
{{/if}}
<div class="form-actions">
<a class="delete-link" href="/posts/deleted">{{_ "delete_post"}}</a>
<input type="submit" class="button" value="{{_ "submit"}}" />
</div>
</form>
{{/with}}
<div class="grid grid-module">
{{> quickForm collection="Posts" doc=post id="editPostForm" template="telescope" label-class="control-label" input-col-class="controls"}}
</div>
</template>

View file

@ -1,219 +1,60 @@
Template[getTemplate('post_edit')].helpers({
created: function(){
return moment(this.createdAt).format("MMMM Do, h:mm:ss a");
},
categories: function(){
var post = this;
return Categories.find({}, {sort: {order: 1, name: 1}}).map(function(category) {
category.checked = _.contains(_.pluck(post.categories, '_id'), category._id) ? 'checked' : '';
return category;
});
},
categoriesEnabled: function(){
return Categories.find().count();
},
showPostedAt: function () {
if((Session.get('currentPostStatus') || this.status) == STATUS_APPROVED){
return 'visible'
}else{
return 'hidden'
}
},
isSticky: function(){
return this.sticky ? 'checked' : '';
},
isSelected: function(parentPost){
return parentPost && this._id == parentPost.userId ? 'selected' : '';
},
postedAtDate: function(){
return !!this.postedAt ? moment(this.postedAt).format("MM/DD/YYYY") : null;
},
postedAtTime: function(){
return !!this.postedAt ? moment(this.postedAt).format("HH:mm") : null;
},
users: function(){
return Meteor.users.find({}, {sort: {'profile.name': 1}});
},
userName: function(){
return getDisplayName(this);
},
hasStatusPending: function(){
return this.status == STATUS_PENDING ? 'checked' : '';
},
hasStatusApproved: function(){
return this.status == STATUS_APPROVED ? 'checked' : '';
},
hasStatusRejected: function(){
return this.status == STATUS_REJECTED ? 'checked' : '';
},
shorten: function(){
return !!getSetting('bitlyToken');
}
});
AutoForm.hooks({
editPostForm: {
onSubmit: function(insertDoc, updateDoc, currentDoc) {
Template[getTemplate('post_edit')].rendered = function(){
// run all post edit rendered callbacks
var instance = this;
postEditRenderedCallbacks.forEach(function(callback) {
callback(instance);
});
var updateObject = updateDoc;
var submit = this;
Session.set('currentPostStatus', this.status);
// ------------------------------ Checks ------------------------------ //
var post = this.data.post;
if(post && !this.editor){
this.editor= new EpicEditor(EpicEditorOptions).load();
this.editor.importFile('editor', post.body);
$('#postedAtDate').datepicker();
}
// $("#postUser").selectToAutocomplete(); // XXX
};
Template[getTemplate('post_edit')].events({
'change input[name=status]': function (e, i) {
Session.set('currentPostStatus', e.currentTarget.value);
},
'click input[type=submit]': function(e, instance){
var post = this;
var updateObject = {};
e.preventDefault();
$(e.target).addClass('disabled');
if(!Meteor.user()){
throwError('You must be logged in.');
return false;
}
// ------------------------------ Properties ------------------------------ //
// Basic Properties
var body = instance.editor.exportFile();
var properties = {
title: $('#title').val(),
body: body,
categories: []
};
// URL
var url = $('#url').val();
if(!!url){
properties.url = (url.substring(0, 7) == "http://" || url.substring(0, 8) == "https://") ? url : "http://"+url;
} else {
// if URL is empty, unset it
updateObject.$unset = {url: ""};
}
// ShortURL
var shortUrl = $('#short-url').val();
if(!!shortUrl)
properties.shortUrl = shortUrl;
// ------------------------------ Admin Properties ------------------------------ //
if(isAdmin(Meteor.user())){
// Basic Properties
adminProperties = {
sticky: $('#sticky').is(':checked'),
userId: $('#postUser').val()
};
// Status
adminProperties.status = parseInt($('input[name=status]:checked').val());
properties = _.extend(properties, adminProperties);
// PostedAt
if(adminProperties.status == STATUS_APPROVED){
var $postedAtDate = $('#postedAtDate');
var $postedAtTime = $('#postedAtTime');
var setPostedAt = false;
var postedAt = new Date(); // default to current browser date and time
var postedAtDate = $postedAtDate.datepicker('getDate');
var postedAtTime = $postedAtTime.val();
if($postedAtDate.exists() && postedAtDate != "Invalid Date"){ // if custom date is set, use it
postedAt = postedAtDate;
setPostedAt = true;
}
if($postedAtTime.exists() && postedAtTime.split(':').length==2){ // if custom time is set, use it
var hours = postedAtTime.split(':')[0];
var minutes = postedAtTime.split(':')[1];
postedAt = moment(postedAt).hour(hours).minute(minutes).toDate();
setPostedAt = true;
}
if(setPostedAt){ // if either custom date or time has been set, pass result to method
Meteor.call('setPostedAt', post, postedAt); // use a method to guarantee timestamp integrity
}else{
Meteor.call('setPostedAt', post);
}
if (!Meteor.user()) {
flashMessage(i18n.t('you_must_be_logged_in'), "");
return false;
}
}
// ------------------------------ Callbacks ------------------------------ //
// ------------------------------ Callbacks ------------------------------ //
// run all post edit client callbacks on properties object successively
properties = postEditClientCallbacks.reduce(function(result, currentFunction) {
return currentFunction(result);
}, properties);
// run all post edit client callbacks on updateObject object successively
updateObject = postEditClientCallbacks.reduce(function(result, currentFunction) {
return currentFunction(result);
}, updateObject);
// console.log(properties)
// ------------------------------ Update ------------------------------ //
if (properties) {
updateObject.$set = properties;
Posts.update(post._id, updateObject, function(error){
// ------------------------------ Update ------------------------------ //
Meteor.call('editPost', currentDoc._id, updateObject, function(error, post) {
if(error){
console.log(error);
throwError(error.message);
clearSeenErrors();
$(e.target).removeClass('disabled');
submit.done(error);
}else{
// note: find a way to do this in onSuccess instead?
trackEvent("edit post", {'postId': post._id});
Router.go("/posts/"+post._id);
Router.go('post_page', {_id: post._id});
submit.done();
}
});
} else {
$(e.target).removeClass('disabled');
return false
},
onSuccess: function(operation, result, template) {
// not used right now because I can't find a way to pass the "post" object to this callback
// console.log(result)
// trackEvent("new post", {'postId': post._id});
// if(post.status === STATUS_PENDING)
// throwError('Thanks, your post is awaiting approval.');
// Router.go('/posts/'+post._id);
},
onError: function(operation, error, template) {
console.log(error)
flashMessage(error.reason.split('|')[0], "error"); // workaround because error.details returns undefined
clearSeenMessages();
}
},
'click .delete-link': function(e){
var post = this;
e.preventDefault();
if(confirm("Are you sure?")){
Router.go("/");
Meteor.call("deletePostById", post._id, function(error) {
if (error) {
console.log(error);
throwError(error.reason);
} else {
throwError('Your post has been deleted.');
}
});
}
// Called at the beginning and end of submission, respectively.
// This is the place to disable/enable buttons or the form,
// show/hide a "Please wait" message, etc. If these hooks are
// not defined, then by default the submit button is disabled
// during submission.
// beginSubmit: function(formId, template) {},
// endSubmit: function(formId, template) {}
}
});

View file

@ -1,76 +1,7 @@
<template name="post_submit">
<div class="grid submit">
<form class="grid-block form-horizontal">
<div class="control-group">
<label class="control-label">{{_ "url"}}</label>
<div class="controls"><input id="url" type="text" value="" /></div>
</div>
<div class="control-group">
<label class="control-label">{{_ "title"}}</label>
<div class="controls"><input id="title" type="text" value="" /><a href="#" class="get-title-link inline-link">{{_ "suggest_title"}}</a></div>
</div>
<div class="control-group">
<label class="control-label">{{_ "body"}}</label>
<div class="controls" id="editor"><textarea id="body" value="" class="input-xlarge"></textarea></div>
</div>
{{#if categoriesEnabled}}
<div class="control-group">
<label class="control-label">{{_ "category"}}</label>
<div class="controls">
{{#each categories}}
<label class="radio inline">
<input id="category_{{_id}}" type="checkbox" value="{{_id}}" name="category" /> {{name}}
</label>
{{/each}}
</div>
</div>
{{/if}}
{{#if isAdmin}}
<div class="control-group">
<label class="control-label">{{_ "sticky_"}}</label>
<div class="controls">
<input type="checkbox" name="sticky" id="sticky" />
</div>
</div>
<div class="control-group">
<label class="control-label">{{_ "user"}}</label>
<div class="controls">
<select id="postUser">
{{#each users}}
<option {{isSelected this}} value="{{_id}}">{{userName}}</option>
{{/each}}
</select>
</div>
</div>
<div class="control-group">
<label class="control-label">{{_ "status_"}}</label>
<div class="controls">
<label class="radio inline">
<input id="status_pending" type="radio" value="1" name="status"/> {{_ "pending"}}
</label>
<label class="radio inline">
<input id="status_approved" type="radio" value="2" name="status" checked /> {{_ "approved"}}
</label>
<label class="radio inline">
<input id="status_rejected" type="radio" value="3" name="status"/> {{_ "rejected"}}
</label>
</div>
</div>
<div id="postedAt" class={{showPostedAt}}>
<div class="control-group">
<label class="control-label post-form-date">{{_ "posted_date"}}</label>
<div class="controls"><input id="postedAtDate" type="text" value="{{postedAtDate}}" /></div>
</div>
<div class="control-group">
<label class="control-label post-form-time">{{_ "posted_time"}}</label>
<div class="controls"><input id="postedAtTime" type="text" value="{{postedAtTime}}" /></div>
</div>
<p class="help-block">{{_ "leave_blank_to_post_at_current_time"}}</p>
</div>
{{/if}}
<div class="form-actions">
<input type="submit" class="button" value="{{_ "submit"}}" />
</div>
</form>
<div class="grid grid-module">
{{> quickForm collection="Posts" id="submitPostForm" template="telescope" label-class="control-label" input-col-class="controls"}}
</div>
</template>

View file

@ -1,156 +1,69 @@
Template[getTemplate('post_submit')].helpers({
categoriesEnabled: function(){
return Categories.find().count();
},
categories: function(){
return Categories.find();
},
users: function(){
return Meteor.users.find({}, {sort: {'profile.name': 1}});
},
userName: function(){
return getDisplayName(this);
},
isSelected: function(user){
return user._id == Meteor.userId() ? "selected" : "";
},
showPostedAt: function () {
if(Session.get('currentPostStatus') == STATUS_APPROVED){
return 'visible'
}else{
return 'hidden'
}
// return (Session.get('currentPostStatus') || STATUS_APPROVED) == STATUS_APPROVED; // default to approved
}
});
AutoForm.hooks({
submitPostForm: {
onSubmit: function(insertDoc, updateDoc, currentDoc) {
Template[getTemplate('post_submit')].rendered = function(){
// run all post submit rendered callbacks
var instance = this;
postSubmitRenderedCallbacks.forEach(function(callback) {
callback(instance);
});
var properties = insertDoc;
var submit = this;
Session.set('currentPostStatus', STATUS_APPROVED);
Session.set('selectedPostId', null);
if(!this.editor && $('#editor').exists())
this.editor= new EpicEditor(EpicEditorOptions).load();
// ------------------------------ Checks ------------------------------ //
$('#postedAtDate').datepicker();
if (!Meteor.user()) {
flashMessage(i18n.t('you_must_be_logged_in'), 'error');
return false;
}
// $("#postUser").selectToAutocomplete(); // XXX
// ------------------------------ Callbacks ------------------------------ //
};
// run all post submit client callbacks on properties object successively
properties = postSubmitClientCallbacks.reduce(function(result, currentFunction) {
return currentFunction(result);
}, properties);
Template[getTemplate('post_submit')].events({
'change input[name=status]': function (e, i) {
Session.set('currentPostStatus', e.currentTarget.value);
},
'click input[type=submit]': function(e, instance){
e.preventDefault();
// console.log(properties)
$(e.target).addClass('disabled');
// ------------------------------ Checks ------------------------------ //
if(!Meteor.user()){
throwError(i18n.t('you_must_be_logged_in'));
return false;
}
// ------------------------------ Properties ------------------------------ //
// Basic Properties
var properties = {
title: $('#title').val(),
body: instance.editor.exportFile(),
sticky: $('#sticky').is(':checked'),
userId: $('#postUser').val(),
status: parseInt($('input[name=status]:checked').val())
};
// PostedAt
var $postedAtDate = $('#postedAtDate');
var $postedAtTime = $('#postedAtTime');
var setPostedAt = false;
var postedAt = new Date(); // default to current browser date and time
var postedAtDate = $postedAtDate.datepicker('getDate');
var postedAtTime = $postedAtTime.val();
if ($postedAtDate.exists() && postedAtDate != "Invalid Date"){ // if custom date is set, use it
postedAt = postedAtDate;
setPostedAt = true;
}
if ($postedAtTime.exists() && postedAtTime.split(':').length==2){ // if custom time is set, use it
var hours = postedAtTime.split(':')[0];
var minutes = postedAtTime.split(':')[1];
postedAt = moment(postedAt).hour(hours).minute(minutes).toDate();
setPostedAt = true;
}
if(setPostedAt) // if either custom date or time has been set, pass result to properties
properties.postedAt = postedAt;
// URL
var url = $('#url').val();
if(!!url){
var cleanUrl = (url.substring(0, 7) == "http://" || url.substring(0, 8) == "https://") ? url : "http://"+url;
properties.url = cleanUrl;
}
// ------------------------------ Callbacks ------------------------------ //
// run all post submit client callbacks on properties object successively
properties = postSubmitClientCallbacks.reduce(function(result, currentFunction) {
return currentFunction(result);
}, properties);
// console.log(properties)
// ------------------------------ Insert ------------------------------ //
if (properties) {
Meteor.call('post', properties, function(error, post) {
// ------------------------------ Insert ------------------------------ //
Meteor.call('submitPost', properties, function(error, post) {
if(error){
throwError(error.reason);
clearSeenErrors();
$(e.target).removeClass('disabled');
if(error.error == 603)
Router.go('/posts/'+error.details);
submit.done(error);
}else{
// note: find a way to do this in onSuccess instead?
trackEvent("new post", {'postId': post._id});
if(post.status === STATUS_PENDING)
throwError('Thanks, your post is awaiting approval.');
Router.go('/posts/'+post._id);
if (post.status === STATUS_PENDING) {
flashMessage(i18n.t('thanks_your_post_is_awaiting_approval'), 'success');
}
Router.go('post_page', {_id: post._id});
submit.done();
}
});
} else {
$(e.target).removeClass('disabled');
return false
},
onSuccess: function(operation, result, template) {
// not used right now because I can't find a way to pass the "post" object to this callback
// console.log(post)
// trackEvent("new post", {'postId': post._id});
// if(post.status === STATUS_PENDING)
// throwError('Thanks, your post is awaiting approval.');
// Router.go('/posts/'+post._id);
},
onError: function(operation, error, template) {
flashMessage(error.message.split('|')[0], 'error'); // workaround because error.details returns undefined
clearSeenMessages();
// $(e.target).removeClass('disabled');
if (error.error == 603) {
var dupePostId = error.reason.split('|')[1];
Router.go('/posts/'+dupePostId);
}
}
},
'click .get-title-link': function(e){
e.preventDefault();
var url=$("#url").val();
var $getTitleLink = $(".get-title-link");
$getTitleLink.addClass("loading");
if(url){
$.get(url, function(response){
if ((suggestedTitle=((/<title>(.*?)<\/title>/m).exec(response.responseText))) != null){
$("#title").val(suggestedTitle[1]);
}else{
alert("Sorry, couldn't find a title...");
}
$getTitleLink.removeClass("loading");
});
}else{
alert("Please fill in an URL first!");
$getTitleLink.removeClass("loading");
}
// Called at the beginning and end of submission, respectively.
// This is the place to disable/enable buttons or the form,
// show/hide a "Please wait" message, etc. If these hooks are
// not defined, then by default the submit button is disabled
// during submission.
// beginSubmit: function(formId, template) {},
// endSubmit: function(formId, template) {}
}
});

View file

@ -11,7 +11,7 @@
{{/if}}
</div>
</div>
{{> UI.dynamic template=postsListIncoming data=incoming}}
{{#if hasPosts}}

View file

@ -1,7 +1,7 @@
Template[getTemplate('posts_digest')].created = function(){
$(document).unbind('keyup'); //remove any potential existing bindings to avoid duplicates
var currentDate=moment(Session.get('currentDate')).startOf('day');
var today=moment(new Date()).startOf('daysy');
var today=moment(new Date()).startOf('day');
$(document).bind('keyup', 'left', function(){
Router.go($('.prev-link').attr('href'));
});

View file

@ -1,97 +0,0 @@
<template name="quickForm_settings">
{{#autoForm qfAutoFormContext}}
{{#each afFieldsets}}
<fieldset>
<h3 class="fieldset-heading">{{this}}</h3>
{{> afQuickFields fields=fieldsForFieldset omitFields=../atts.omitFields template="settings" input-col-class=inputClass label-class=labelClass}}
</fieldset>
{{/each}}
{{#if qfShouldRenderButton}}
<div class="form-group">
<div class="{{labelClass}}"></div>
<div class="{{inputClass}}">
<button {{submitButtonAtts}}>
{{#with ../atts.buttonContent}}
{{this}}
{{else}}
Submit
{{/with}}
</button>
</div>
</div>
{{/if}}
{{/autoForm}}
</template>
<template name="afFormGroup_settings">
{{#if showField}}
<div class="form-group {{#if afFieldIsInvalid name=this.atts.name}}has-error{{/if}}">
<label class="control-label">{{afFieldLabelText name=this.atts.name}}</label>
<div class="{{rightColumnClass}}">
{{> afFieldInput afFieldInputAtts}}
{{#if afFieldInstructions}}
<span class="instructions-block">{{afFieldInstructions}}</span>
{{/if}}
<span class="help-block">{{{afFieldMessage name=this.atts.name}}}</span>
</div>
</div>
{{/if}}
</template>
<template name="afObjectField_settings">
<div class="form-group {{#if afFieldIsInvalid name=this.atts.name}}has-error{{/if}}">
<label class="control-label">{{afFieldLabelText name=this.atts.name}}</label>
<div class="{{rightColumnClass}}">
<div class="panel panel-default autoform-padding-fix">
<div class="panel-body">
{{> afQuickFields name=this.atts.name fields=this.atts.fields omitFields=this.atts.omitFields template="bootstrap3"}}
</div>
</div>
</div>
</div>
</template>
<template name="afArrayField_settings">
<div class="form-group {{#if afFieldIsInvalid name=this.atts.name}}has-error{{/if}}">
<label class="control-label">{{afFieldLabelText name=this.atts.name}}</label>
<div class="{{rightColumnClass}}">
<div class="panel panel-default autoform-padding-fix">
{{#if afFieldIsInvalid name=this.atts.name}}
<div class="panel-body has-error">
<span class="help-block">{{{afFieldMessage name=this.atts.name}}}</span>
</div>
{{/if}}
<ul class="list-group">
{{#afEachArrayItem name=this.atts.name minCount=this.atts.minCount maxCount=this.atts.maxCount}}
<li class="list-group-item autoform-array-item">
<div class="media">
{{#if afArrayFieldHasMoreThanMinimum name=../atts.name minCount=../atts.minCount maxCount=../atts.maxCount}}
<button class="btn btn-primary autoform-remove-item pull-left"><span class="glyphicon glyphicon-minus"></span></button>
{{/if}}
<div class="media-body">
{{> afQuickField name=this.name label=false}}
</div>
</div>
</li>
{{/afEachArrayItem}}
{{#if afArrayFieldHasLessThanMaximum name=this.atts.name minCount=this.atts.minCount maxCount=this.atts.maxCount}}
<li class="list-group-item">
<button class="btn btn-primary autoform-add-item" data-autoform-field="{{this.atts.name}}" data-autoform-minCount="{{this.atts.minCount}}" data-autoform-maxCount="{{this.atts.maxCount}}"><span class="glyphicon glyphicon-plus"></span></button>
</li>
{{/if}}
</ul>
</div>
</div>
</div>
</template>
<template name="afCheckbox_settings">
<div class="checkbox">
<label>
<input type="checkbox" value="{{this.value}}" {{atts}} />aaaa
</label>
</div>
</template>

View file

@ -25,23 +25,23 @@ var scrollUp = function(){
AutoForm.hooks({
inviteForm: {
onSuccess: function(operation, result, template) {
clearSeenErrors();
clearSeenMessages();
if(result && result.newUser){
throwError('An invite has been sent out. Thank you!');
flashMessage('An invite has been sent out. Thank you!', "success");
} else {
throwError('Thank you!');
flashMessage('Thank you!', "info");
}
scrollUp();
},
onError: function(operation, error, template) {
clearSeenErrors();
clearSeenMessages();
if(error && error.reason){
throwError(error.reason);
flashMessage(error.reason, "error");
scrollUp();
}
}
}
});
});

View file

@ -15,7 +15,7 @@ Template[getTemplate('user_edit')].helpers({
return getGitHubName(this) || "";
},
profileUrl: function(){
return Meteor.absoluteUrl()+"users/"+this.slug;
return getProfileUrlBySlugOrId(this.slug);
},
hasNotificationsUsers : function(){
return getUserSetting('notifications.users', '', this) ? 'checked' : '';
@ -38,9 +38,9 @@ Template[getTemplate('user_edit')].events({
'submit #account-form': function(e){
e.preventDefault();
clearSeenErrors();
clearSeenMessages();
if(!Meteor.user())
throwError(i18n.t('you_must_be_logged_in'));
flashMessage(i18n.t('you_must_be_logged_in'), 'error');
var $target=$(e.target);
var name = $target.find('[name=name]').val();
@ -67,17 +67,21 @@ Template[getTemplate('user_edit')].events({
Accounts.changePassword(old_password, new_password, function(error){
// TODO: interrupt update if there's an error at this point
if(error)
throwError(error.reason);
flashMessage(error.reason, "error");
});
}
update = userEditClientCallbacks.reduce(function(result, currentFunction) {
return currentFunction(user, result);
}, update);
Meteor.users.update(user._id, {
$set: update
}, function(error){
if(error){
throwError(error.reason);
flashMessage(error.reason, "error");
} else {
throwError(i18n.t('profile_updated'));
flashMessage(i18n.t('profile_updated'), 'success');
}
Deps.afterFlush(function() {
var element = $('.grid > .error');
@ -89,4 +93,4 @@ Template[getTemplate('user_edit')].events({
}
});
});

View file

@ -17,7 +17,7 @@
</div>
</div>
<div class="form-actions">
<input type="submit" class="button" value="{{_ "submit"}}" />
<input type="submit" class="button btn btn-primary" value="{{_ "submit"}}" />
</div>
</form>
{{/with}}

View file

@ -10,7 +10,8 @@ Template[getTemplate('user_email')].helpers({
Template[getTemplate('user_email')].events({
'submit form': function(e){
e.preventDefault();
if(!Meteor.user()) throwError(i18n.t('you_must_be_logged_in'));
if(!Meteor.user())
flashMessage(i18n.t('you_must_be_logged_in'), 'error');
var $target=$(e.target);
var user=Session.get('selectedUserId')? Meteor.users.findOne(Session.get('selectedUserId')) : Meteor.user();
var update = {
@ -26,9 +27,9 @@ Template[getTemplate('user_email')].events({
$set: update
}, function(error){
if(error){
throwError(error.reason);
flashMessage(error.reason, "error");
} else {
throwError(i18n.t('thanks_for_signing_up'));
flashMessage(i18n.t('thanks_for_signing_up'), "success");
// Meteor.call('addCurrentUserToMailChimpList');
trackEvent("new sign-up", {'userId': user._id, 'auth':'twitter'});
Router.go('/');

View file

@ -43,7 +43,7 @@
{{/if}}
</table>
{{#if canEditProfile}}
<a class="button inline" href="/users/{{slug}}/edit">{{_ "edit_profile"}}</a>
<a class="button inline" href="{{pathFor route='user_edit' slug=slug}}">{{_ "edit_profile"}}</a>
{{/if}}
{{#if canInvite}}
{{#if inviteCount}}
@ -65,7 +65,7 @@
</thead>
{{#each posts}}
<tr>
<td><a href="/posts/{{_id}}/">{{title}}</a></td>
<td><a href="{{pathFor route='post_page' _id=_id}}">{{title}}</a></td>
<td>{{formatDate createdAt "MM/DD/YYYY, HH:mm"}}</td>
</tr>
{{/each}}
@ -90,7 +90,7 @@
</thead>
{{#each upvotedPosts}}
<tr>
<td><a href="/posts/{{_id}}/">{{title}}</a></td>
<td><a href="{{pathFor route='post_page' _id=_id}}">{{title}}</a></td>
<td>{{formatDate votedAt "MM/DD/YYYY, HH:mm"}}</td>
</tr>
{{/each}}
@ -115,7 +115,7 @@
</thead>
{{#each downvoted}}
<tr>
<td><a href="/posts/{{_id}}/">{{title}}</a></td>
<td><a href="{{pathFor route='post_page' _id=_id}}">{{title}}</a></td>
<td>{{formatDate votedAt "MM/DD/YYYY, HH:mm"}}</td>
</tr>
{{/each}}
@ -141,7 +141,7 @@
</thead>
{{#each comments}}
<tr>
<td><a href="/posts/{{postId}}/">{{postTitle}}</a></td>
<td><a href="{{pathFor route='post_page' _id=postId}}/">{{postTitle}}</a></td>
<td>{{{htmlBody}}}</td>
<td>{{formatDate createdAt "MM/DD/YYYY, HH:mm"}}</td>
</tr>

View file

@ -79,7 +79,7 @@ Template[getTemplate('user_profile')].helpers({
Template[getTemplate('user_profile')].events({
'click .invite-link': function(e, instance){
Meteor.call('inviteUser', instance.data.user._id);
throwError('Thanks, user has been invited.');
flashMessage('Thanks, user has been invited.', "success");
},
'click .posts-more': function (e) {
e.preventDefault();
@ -101,4 +101,4 @@ Template[getTemplate('user_profile')].events({
var commentsShown = Session.get('commentsShown');
Session.set('commentsShown', commentsShown + 10);
}
});
});

View file

@ -194,7 +194,7 @@ Meteor.methods({
isDeleted: true
}});
}else{
throwError("You don't have permission to delete this comment.");
flashMessage("You don't have permission to delete this comment.", "error");
}
}
});

View file

@ -1,4 +1,4 @@
if(Meteor.isClient){
// Local (client-only) collection
Errors = new Meteor.Collection(null);
Messages = new Meteor.Collection(null);
}

View file

@ -1,94 +1,200 @@
// ------------------------------------------------------------------------------------------- //
// ----------------------------------------- Schema ----------------------------------------- //
// ------------------------------------------------------------------------------------------- //
SimpleSchema.extendOptions({
editable: Match.Optional(Boolean) // editable: true means the field can be edited by the document's owner
});
postSchemaObject = {
_id: {
type: String,
optional: true
optional: true,
autoform: {
omit: true
}
},
createdAt: {
type: Date,
optional: true
optional: true,
autoform: {
omit: true
}
},
postedAt: {
type: Date,
optional: true
},
title: {
type: String,
label: "Title"
optional: true,
autoform: {
group: 'admin',
type: "bootstrap-datetimepicker"
}
},
url: {
type: String,
label: "URL",
optional: true
optional: true,
autoform: {
editable: true,
type: "bootstrap-url"
}
},
title: {
type: String,
optional: false,
label: "Title",
editable: true,
autoform: {
editable: true
}
},
body: {
type: String,
optional: true
optional: true,
editable: true,
autoform: {
editable: true,
rows: 5
}
},
htmlBody: {
type: String,
optional: true
optional: true,
autoform: {
omit: true
}
},
viewCount: {
type: Number,
optional: false
optional: true,
autoform: {
omit: true
}
},
commentCount: {
type: Number,
optional: false
optional: true,
autoform: {
omit: true
}
},
commenters: {
type: [String],
optional: true
optional: true,
autoform: {
omit: true
}
},
lastCommentedAt: {
type: Date,
optional: true
optional: true,
autoform: {
omit: true
}
},
clickCount: {
type: Number,
optional: false
optional: true,
autoform: {
omit: true
}
},
baseScore: {
type: Number,
decimal: true,
optional: true
optional: true,
autoform: {
omit: true
}
},
upvotes: {
type: Number,
optional: true
optional: true,
autoform: {
omit: true
}
},
upvoters: {
type: [String], // XXX
optional: true
optional: true,
autoform: {
omit: true
}
},
downvotes: {
type: Number,
optional: true
optional: true,
autoform: {
omit: true
}
},
downvoters: {
type: [String], // XXX
optional: true
optional: true,
autoform: {
omit: true
}
},
score: {
type: Number,
decimal: true,
optional: true
optional: true,
autoform: {
omit: true
}
},
status: {
type: Number,
optional: true
optional: true,
autoValue: function () {
// only provide a default value
// 1) this is an insert operation
// 2) status field is not set in the document being inserted
if(this.isInsert && !this.isSet)
return getSetting('requirePostsApproval', false) ? STATUS_PENDING: STATUS_APPROVED
},
autoform: {
noselect: true,
options: postStatuses,
group: 'admin'
}
},
sticky: {
type: Boolean,
optional: true
optional: true,
autoform: {
group: 'admin',
leftLabel: "Sticky"
}
},
inactive: {
type: Boolean,
optional: true
optional: true,
autoform: {
omit: true
}
},
author: {
type: String,
optional: true,
autoform: {
omit: true
}
},
userId: {
type: String, // XXX
optional: true
optional: true,
autoform: {
group: 'admin',
options: function () {
return Meteor.users.find().map(function (user) {
return {
value: user._id,
label: getDisplayName(user)
}
});
}
}
}
};
@ -102,39 +208,36 @@ Posts = new Meteor.Collection("posts");
PostSchema = new SimpleSchema(postSchemaObject);
Posts.attachSchema(PostSchema);
STATUS_PENDING=1;
STATUS_APPROVED=2;
STATUS_REJECTED=3;
// Posts.deny({
// update: function(userId, post, fieldNames) {
// if(isAdminById(userId))
// return false;
// // deny the update if it contains something other than the following fields
// return (_.without(fieldNames, 'title', 'url', 'body', 'shortUrl', 'shortTitle', 'categories').length > 0);
// }
// });
Posts.deny({
update: function(userId, post, fieldNames) {
if(isAdminById(userId))
return false;
// deny the update if it contains something other than the following fields
return (_.without(fieldNames, 'title', 'url', 'body', 'shortUrl', 'shortTitle', 'categories').length > 0);
}
});
// Posts.allow({
// update: canEditById,
// remove: canEditById
// });
Posts.allow({
update: canEditById,
remove: canEditById
});
// ------------------------------------------------------------------------------------------- //
// ----------------------------------------- Helpers ----------------------------------------- //
// ------------------------------------------------------------------------------------------- //
postClicks = [];
postViews = [];
getPostProperties = function(post) {
getPostProperties = function (post) {
var postAuthor = Meteor.users.findOne(post.userId);
var p = {
postAuthorName : getDisplayName(postAuthor),
postTitle : cleanUp(post.title),
profileUrl: getProfileUrlById(post.userId),
profileUrl: getProfileUrlBySlugOrId(post.userId),
postUrl: getPostPageUrl(post),
thumbnailUrl: post.thumbnailUrl,
linkUrl: !!post.url ? getOutgoingUrl(post.url) : getPostPageUrl(post._id)
};
if(post.url)
p.url = post.url;
@ -157,6 +260,23 @@ getPostLink = function (post) {
return !!post.url ? getOutgoingUrl(post.url) : getPostPageUrl(post);
};
checkForPostsWithSameUrl = function (url) {
// check that there are no previous posts with the same link in the past 6 months
var sixMonthsAgo = moment().subtract(6, 'months').toDate();
var postWithSameLink = Posts.findOne({url: url, postedAt: {$gte: sixMonthsAgo}});
if(typeof postWithSameLink !== 'undefined'){
Meteor.call('upvotePost', postWithSameLink);
// note: error.details returns undefined on the client, so add post ID to reason
throw new Meteor.Error('603', i18n.t('this_link_has_already_been_posted') + '|' + postWithSameLink._id, postWithSameLink._id);
}
}
// ------------------------------------------------------------------------------------------- //
// ------------------------------------------ Hooks ------------------------------------------ //
// ------------------------------------------------------------------------------------------- //
Posts.before.insert(function (userId, doc) {
if(Meteor.isServer && !!doc.body)
doc.htmlBody = sanitize(marked(doc.body));
@ -170,8 +290,16 @@ Posts.before.update(function (userId, doc, fieldNames, modifier, options) {
}
});
// ------------------------------------------------------------------------------------------- //
// ----------------------------------------- Methods ----------------------------------------- //
// ------------------------------------------------------------------------------------------- //
postClicks = [];
postViews = [];
Meteor.methods({
post: function(post){
submitPost: function(post){
var title = cleanUp(post.title),
body = post.body,
userId = this.userId,
@ -181,7 +309,6 @@ Meteor.methods({
postInterval = Math.abs(parseInt(getSetting('postInterval', 30))),
maxPostsPer24Hours = Math.abs(parseInt(getSetting('maxPostsPerDay', 30))),
postId = '';
// ------------------------------ Checks ------------------------------ //
@ -193,17 +320,11 @@ Meteor.methods({
if(!post.title)
throw new Meteor.Error(602, i18n.t('please_fill_in_a_title'));
// check that there are no posts with the same URL
if(!!post.url)
checkForPostsWithSameUrl(post.url);
if(!!post.url){
// check that there are no previous posts with the same link in the past 6 months
var sixMonthsAgo = moment().subtract(6, 'months').toDate();
var postWithSameLink = Posts.findOne({url: post.url, postedAt: {$gte: sixMonthsAgo}});
if(typeof postWithSameLink !== 'undefined'){
Meteor.call('upvotePost', postWithSameLink);
throw new Meteor.Error(603, i18n.t('this_link_has_already_been_posted'), postWithSameLink._id);
}
}
// --------------------------- Rate Limiting -------------------------- //
if(!isAdmin(Meteor.user())){
// check that user waits more than X seconds between posts
@ -233,9 +354,9 @@ Meteor.methods({
inactive: false
};
// UserId
// UserId
if(isAdmin(Meteor.user()) && !!post.userId){ // only let admins post as other users
properties.userId = post.userId;
properties.userId = post.userId;
}
// Status
@ -243,7 +364,7 @@ Meteor.methods({
if(isAdmin(Meteor.user()) && !!post.status){ // if user is admin and a custom status has been set
properties.status = post.status;
}else{ // else use default status
properties.status = defaultPostStatus;
properties.status = defaultPostStatus;
}
// CreatedAt
@ -279,7 +400,7 @@ Meteor.methods({
return currentFunction(result);
}, post);
// ------------------------------ Post-Insert ------------------------------ //
// ------------------------------ After Insert ------------------------------ //
// increment posts count
Meteor.users.update({_id: userId}, {$inc: {postCount: 1}});
@ -290,33 +411,72 @@ Meteor.methods({
return post;
},
editPost: function (postId, updateObject) {
var user = Meteor.user();
// console.log(updateObject)
// ------------------------------ Checks ------------------------------ //
// check that user can edit
if (!user || !canEdit(user, Posts.findOne(postId)))
throw new Meteor.Error(601, i18n.t('sorry_you_cannot_edit_this_post'));
// ------------------------------ Callbacks ------------------------------ //
// run all post submit server callbacks on updateObject successively
updateObject = postEditMethodCallbacks.reduce(function(result, currentFunction) {
return currentFunction(result);
}, updateObject);
console.log(updateObject)
// ------------------------------ Update ------------------------------ //
Posts.update(postId, updateObject);
// ------------------------------ Callbacks ------------------------------ //
// run all post submit server callbacks on updateObject successively
updateObject = postAfterEditMethodCallbacks.reduce(function(result, currentFunction) {
return currentFunction(result);
}, updateObject);
// ------------------------------ After Update ------------------------------ //
return Posts.findOne(postId);
},
setPostedAt: function(post, customPostedAt){
var postedAt = new Date(); // default to current date and time
if(isAdmin(Meteor.user()) && typeof customPostedAt !== 'undefined') // if user is admin and a custom datetime has been set
postedAt = customPostedAt;
Posts.update(post._id, {$set: {postedAt: postedAt}});
},
post_edit: function(post){
// TODO: make post_edit server-side?
},
approvePost: function(post){
if(isAdmin(Meteor.user())){
var now = new Date();
Posts.update(post._id, {$set: {status: 2, postedAt: now}});
var result = Posts.update(post._id, {$set: {status: 2, postedAt: now}}, {validate: false});
}else{
throwError('You need to be an admin to do that.');
flashMessage('You need to be an admin to do that.', "error");
}
},
unapprovePost: function(post){
if(isAdmin(Meteor.user())){
Posts.update(post._id, {$set: {status: 1}});
}else{
throwError('You need to be an admin to do that.');
flashMessage('You need to be an admin to do that.', "error");
}
},
increasePostViews: function(postId, sessionId){
this.unblock();
@ -328,7 +488,8 @@ Meteor.methods({
Posts.update(postId, { $inc: { viewCount: 1 }});
}
},
increasePostClicks: function(postId, sessionId){
increasePostClicks: function(postId, sessionId){
this.unblock();
// only let clients increment a post's click counter once per session
@ -339,6 +500,7 @@ Meteor.methods({
Posts.update(postId, { $inc: { clickCount: 1 }});
}
},
deletePostById: function(postId) {
// remove post comments
// if(!this.isSimulation) {
@ -349,8 +511,9 @@ Meteor.methods({
// decrement post count
var post = Posts.findOne({_id: postId});
if(!Meteor.userId() || !canEditById(Meteor.userId(), post)) throw new Meteor.Error(606, 'You need permission to edit or delete a post');
Meteor.users.update({_id: post.userId}, {$inc: {postCount: -1}});
Posts.remove(postId);
}
});

View file

@ -29,7 +29,8 @@ settingsSchemaObject = {
label: "Require invite to view",
optional: true,
autoform: {
group: 'invites'
group: 'invites',
leftLabel: 'Require View Invite'
}
},
requirePostInvite: {
@ -37,7 +38,8 @@ settingsSchemaObject = {
label: "Require invite to post",
optional: true,
autoform: {
group: 'invites'
group: 'invites',
leftLabel: 'Require Post Invite'
}
},
requirePostsApproval: {
@ -45,7 +47,8 @@ settingsSchemaObject = {
optional: true,
autoform: {
group: 'general',
instructions: "Posts must be approved by admin"
instructions: "Posts must be approved by admin",
leftLabel: "Require Posts Approval"
}
},
// nestedComments: {
@ -69,7 +72,8 @@ settingsSchemaObject = {
optional: true,
autoform: {
group: 'email',
instructions: 'The address all outgoing emails will be sent from.'
instructions: 'The address all outgoing emails will be sent from.',
private: true
}
},
scoreUpdateInterval: {
@ -78,7 +82,8 @@ settingsSchemaObject = {
defaultValue: 30,
autoform: {
group: 'scoring',
instructions: 'How often to recalculate scores, in seconds (default to 30)'
instructions: 'How often to recalculate scores, in seconds (default to 30)',
private: true
}
},
defaultView: {
@ -195,28 +200,32 @@ settingsSchemaObject = {
type: String,
optional: true,
autoform: {
group: 'colors'
group: 'colors',
// type: 'color'
}
},
buttonTextColor: {
type: String,
optional: true,
autoform: {
group: 'colors'
group: 'colors',
// type: 'color'
}
},
headerColor: {
type: String,
optional: true,
autoform: {
group: 'colors'
group: 'colors',
// type: 'color'
}
},
headerTextColor: {
type: String,
optional: true,
autoform: {
group: 'colors'
group: 'colors',
// type: 'color'
}
},
twitterAccount: {
@ -271,7 +280,8 @@ settingsSchemaObject = {
autoform: {
group: 'email',
instructions: 'Content that will appear at the bottom of outgoing emails (accepts HTML).',
rows: 5
rows: 5,
private: true
}
},
notes: {
@ -280,7 +290,8 @@ settingsSchemaObject = {
autoform: {
group: 'extras',
instructions: 'You can store any notes or extra information here.',
rows: 5
rows: 5,
private: true
}
},
};

View file

@ -1,6 +1,5 @@
var Schema = {};
Schema.User = new SimpleSchema({
var userSchemaObj = {
_id: {
type: String,
optional: true
@ -42,7 +41,14 @@ Schema.User = new SimpleSchema({
optional: true,
blackbox: true
}
};
// add any extra properties to postSchemaObject (provided by packages for example)
_.each(addToUserSchema, function(item){
userSchemaObj[item.propertyName] = item.propertySchema;
});
Schema.User = new SimpleSchema(userSchemaObj);
// Meteor.users.attachSchema(Schema.User);

View file

@ -19,6 +19,77 @@
//Main
"new_posts": "New Posts",
// Settings Schema
"title": "Title",
"siteUrl": "Site URL",
"tagline": "Tagline",
"requireViewInvite": "Require Invite to View",
"requirePostInvite": "Require Invite to Post",
"requirePostsApproval": "Require Posts to be Approved",
"defaultEmail": "Default Email",
"scoreUpdateInterval": "Score Update Interval",
"defaultView": "Default View",
"postInterval": "Post Interval",
"commentInterval": "Comment Interval",
"maxPostsPerDay": "Max Posts Per Day",
"startInvitesCount": "Invites Start Count",
"postsPerPage": "Posts Per Page",
"logoUrl": "Logo URL",
"logoHeight": "Logo Height",
"logoWidth": "Logo Width",
"language": "Language",
"backgroundCSS": "Background CSS",
"buttonColor": "Button Color",
"buttonTextColor": "Button Text Color",
"headerColor": "Header Color",
"headerTextColor": "Header Text Color",
"twitterAccount": "Twitter Account",
"googleAnalyticsId": "Google Analytics ID",
"mixpanelId": "Mixpanel ID",
"clickyId": "Clicky ID",
"footerCode": "Footer Code",
"extraCode": "Extra Code",
"emailFooter": "Email Footer",
"notes": "Notes",
// Settings Fieldsets
"general": "General",
"invites": "Invites",
"email": "Email",
"scoring": "Scoring",
"posts": "Posts",
"comments": "Comments",
"logo": "Logo",
"extras": "Extras",
"colors": "Colors",
"integrations": "Integrations",
// Settings Help Text
// Post Schema
"createdAt": "Created At",
"postedAt": "Posted At",
"url": "URL",
"title": "Title",
"body": "Body",
"htmlBody": "HTML Body",
"viewCount": "View Count",
"commentCount": "Comment Count",
"commenters": "Commenters",
"lastCommentedAt": "Last Commented At",
"clickCount": "Click Count",
"baseScore": "Base Score",
"upvotes": "Upvotes",
"upvoters": "Upvoters",
"downvotes": "Downvotes",
"downvoters": "Downvoters",
"score": "Score",
"status": "Status",
"sticky": "Sticky",
"inactive": "Inactive",
"author": "Author",
"userId": "User",
//Commments
"your_comment_has_been_deleted": "Your comment has been deleted.",
"comment_": "Comment",

View file

@ -19,6 +19,77 @@
//Main
"new_posts": "Nouveaux Posts",
// Settings Schema
"title": "Titre",
"siteUrl": "URL du Site",
"tagline": "Slogan",
"requireViewInvite": "Consultation Restrainte",
"requirePostInvite": "Participation Restrainte",
"requirePostsApproval": "Modération Obligatoire",
"defaultEmail": "Email par Défaut",
"scoreUpdateInterval": "Mise à Jour du Score",
"defaultView": "Vue Par Défaut",
"postInterval": "Interval des Posts",
"commentInterval": "Interval des Commentaires",
"maxPostsPerDay": "Posts par Jour Max",
"startInvitesCount": "Invitations au Départ",
"postsPerPage": "Posts par Page",
"logoUrl": "URL du Logo",
"logoHeight": "Hauteur du Logo",
"logoWidth": "Largeur du Logo",
"language": "Language",
"backgroundCSS": "CSS de Fond",
"buttonColor": "Couleur des Boutons",
"buttonTextColor": "Couleur du Texte des Boutons",
"headerColor": "Couleur du Header",
"headerTextColor": "Couleur du Texte du Header",
"twitterAccount": "Compte Twitter",
"googleAnalyticsId": "ID Google Analytics",
"mixpanelId": "ID Mixpanel",
"clickyId": "ID Clicky",
"footerCode": "Code du Footer",
"extraCode": "Code en Plus",
"emailFooter": "Footer des Emails",
"notes": "Notes",
// Settings Fieldsets
"general": "Général",
"invites": "Invitations",
"email": "Email",
"scoring": "Score",
"posts": "Posts",
"comments": "Commentaires",
"logo": "Logo",
"extras": "Extras",
"colors": "Couleurs",
"integrations": "Intégrations",
// Settings Help Text
//Post Schema
"createdAt": "Créé à",
"postedAt": "Posté à",
"url": "Lien",
"title": "Titre",
"body": "Texte",
"htmlBody": "Texte HTML",
"viewCount": "Vues",
"commentCount": "Commentaires",
"commenters": "Commentateurs",
"lastCommentedAt": "Dernier Commentaire à",
"clickCount": "Clics",
"baseScore": "Score de Base",
"upvotes": "Upvotes",
"upvoters": "Upvoteurs",
"downvotes": "Downvotes",
"downvoters": "Downvoteurs",
"score": "Score",
"status": "Statut",
"sticky": "Mis en Avant",
"inactive": "Inactif",
"author": "Auteur",
"userId": "Utilisateur",
//Commments
"your_comment_has_been_deleted": "Votre commentaire a été supprimé.",
"comment_": "Commentaire",

1
lib/config/debug.js Normal file
View file

@ -0,0 +1 @@
// SimpleSchema.debug = true;

View file

@ -1,13 +0,0 @@
if(Meteor.isClient){
throwError = function(message, type){
type = (typeof type === 'undefined') ? 'error': type;
// Store errors in the 'Errors' local collection
Errors.insert({message:message, type:type, seen: false, show:true});
};
clearSeenErrors = function(){
Errors.update({seen:true}, {$set: {show:false}}, {multi:true});
};
}

View file

@ -12,19 +12,23 @@ t=function(message){
var d=new Date();
console.log("### "+message+" rendered at "+d.getHours()+":"+d.getMinutes()+":"+d.getSeconds());
};
nl2br= function(str) {
var breakTag = '<br />';
return (str + '').replace(/([^>\r\n]?)(\r\n|\n\r|\r|\n)/g, '$1'+ breakTag +'$2');
nl2br= function(str) {
var breakTag = '<br />';
return (str + '').replace(/([^>\r\n]?)(\r\n|\n\r|\r|\n)/g, '$1'+ breakTag +'$2');
};
getAuthorName = function(item){
var user = Meteor.users.findOne(item.userId);
return typeof user === 'undefined' ? '' : getDisplayName(user);
};
scrollPageTo = function(selector){
$('body').scrollTop($(selector).offset().top);
$('body').scrollTop($(selector).offset().top);
};
getDigestURL = function(moment){
return '/digest/'+moment.year()+'/'+(moment.month()+1)+'/'+moment.date();
return Router.routes['posts_digest'].path({
year: moment.year(),
month: moment.month() + 1,
day: moment.date()
});
};
getDateRange= function(pageNumber){
var now = moment(new Date());
@ -50,24 +54,27 @@ goTo = function(url){
};
getSignupUrl = function(){
return Meteor.absoluteUrl()+'sign-up';
return Router.routes['atSignUp'].url();
};
getSigninUrl = function(){
return Meteor.absoluteUrl()+'sign-in';
return Router.routes['atSignIn'].url();
};
getPostUrl = function(id){
return Meteor.absoluteUrl()+'posts/'+id;
return Router.routes['post_page'].url({_id: id});
};
getPostEditUrl = function(id){
return Meteor.absoluteUrl()+'posts/'+id+'/edit';
return Router.routes['post_edit'].url({_id: id});
};
getCommentUrl = function(id){
return Meteor.absoluteUrl()+'comments/'+id;
return Router.routes['comment_reply'].url({_id: id});
};
getPostCommentUrl = function(postId, commentId){
// get link to a comment on a post page
return Meteor.absoluteUrl()+'posts/'+postId+'/comment/'+commentId;
return Router.routes['post_page_comment'].url({
_id: postId,
commentId: commentId
});
};
slugify = function(text) {
if(text){
@ -78,7 +85,7 @@ slugify = function(text) {
return text;
};
getShortUrl = function(post){
return post.shortUrl ? post.shortUrl : post.url;
return post.shortUrl || post.url;
};
getDomain = function(url){
urlObject = Npm.require('url');
@ -88,7 +95,7 @@ invitesEnabled = function () {
return getSetting("requireViewInvite") || getSetting("requirePostInvite");
};
getOutgoingUrl = function(url){
return getSiteUrl() + 'out?url=' + encodeURIComponent(url);
return Router.routes['out'].url({}, {query: {url: url}});
};
// ---------------------------------- String Helper Functions ----------------------------------- //
cleanUp = function(s){
@ -99,11 +106,11 @@ sanitize = function (s) {
// console.log(s)
if(Meteor.isServer){
var s = sanitizeHtml(s, {
allowedTags: [
'h3', 'h4', 'h5', 'h6', 'blockquote', 'p', 'a', 'ul',
'ol', 'nl', 'li', 'b', 'i', 'strong', 'em', 'strike',
'code', 'hr', 'br', 'div', 'table', 'thead', 'caption',
'tbody', 'tr', 'th', 'td', 'pre', 'img'
allowedTags: [
'h3', 'h4', 'h5', 'h6', 'blockquote', 'p', 'a', 'ul',
'ol', 'nl', 'li', 'b', 'i', 'strong', 'em', 'strike',
'code', 'hr', 'br', 'div', 'table', 'thead', 'caption',
'tbody', 'tr', 'th', 'td', 'pre', 'img'
]
});
// console.log('// after sanitization:')

13
lib/messages.js Normal file
View file

@ -0,0 +1,13 @@
if(Meteor.isClient){
flashMessage = function(message, type){
type = (typeof type === 'undefined') ? 'error': type;
// Store errors in the 'Messages' local collection
Messages.insert({message:message, type:type, seen: false, show:true});
};
clearSeenMessages = function(){
Messages.update({seen:true}, {$set: {show:false}}, {multi:true});
};
}

View file

@ -10,7 +10,10 @@ getPostsParameters = function (terms) {
var baseParameters = {
find: {
status: STATUS_APPROVED,
postedAt: {$lte: new Date()}
$or: [
{postedAt: {$lte: new Date()}}, // only show posts if they're in the past
{postedAt: {$exists: false}} // or if they don't have a "postedAt" date (i.e. pending posts)
]
},
options: {
limit: 10

View file

@ -14,8 +14,8 @@ Router._filters = {
}
},
clearSeenErrors: function () {
clearSeenErrors();
clearSeenMessages: function () {
clearSeenMessages();
this.next();
},
@ -73,7 +73,7 @@ Router._filters = {
if(!this.ready() || Meteor.loggingIn()){
this.render(getTemplate('loading'));
} else if(!canPost()) {
throwError(i18n.t("sorry_you_dont_have_permissions_to_add_new_items"));
flashMessage(i18n.t("sorry_you_dont_have_permissions_to_add_new_items"), "error");
this.render(getTemplate('no_rights'));
} else {
this.next();
@ -85,7 +85,7 @@ Router._filters = {
// Already subscribed to this post by route({waitOn: ...})
var post = Posts.findOne(this.params._id);
if(!currentUserCanEdit(post)){
throwError(i18n.t("sorry_you_cannot_edit_this_post"));
flashMessage(i18n.t("sorry_you_cannot_edit_this_post"), "error");
this.render(getTemplate('no_rights'));
} else {
this.next();
@ -97,7 +97,7 @@ Router._filters = {
// Already subscribed to this comment by CommentPageController
var comment = Comments.findOne(this.params._id);
if(!currentUserCanEdit(comment)){
throwError(i18n.t("sorry_you_cannot_edit_this_comment"));
flashMessage(i18n.t("sorry_you_cannot_edit_this_comment"), "error");
this.render(getTemplate('no_rights'));
} else {
this.next();
@ -141,7 +141,7 @@ Meteor.startup( function (){
Session.set('categorySlug', null);
// if we're not on the search page itself, clear search query and field
if(getCurrentRoute().indexOf('search') == -1){
if(Router.current().route.getName() !== 'search'){
Session.set('searchQuery', '');
$('.search-field').val('').blur();
}
@ -153,7 +153,7 @@ Meteor.startup( function (){
// Before Hooks
Router.onBeforeAction(filters.isReady);
Router.onBeforeAction(filters.clearSeenErrors);
Router.onBeforeAction(filters.clearSeenMessages);
Router.onBeforeAction(filters.canView, {except: ['atSignIn', 'atSignUp', 'atForgotPwd', 'atResetPwd', 'signOut']});
Router.onBeforeAction(filters.hasCompletedProfile);
Router.onBeforeAction(filters.isLoggedIn, {only: ['post_submit']});

View file

@ -9,12 +9,12 @@ UserPageController = RouteController.extend({
data: function() {
var findById = Meteor.users.findOne(this.params._idOrSlug);
var findBySlug = Meteor.users.findOne({slug: this.params._idOrSlug});
if(typeof findById !== "undefined"){
if (typeof findById !== 'undefined') {
// redirect to slug-based URL
Router.go(getProfileUrl(findById), {replaceState: true});
}else{
} else {
return {
user: (typeof findById == "undefined") ? findBySlug : findById
user: (typeof findById == 'undefined') ? findBySlug : findById
};
}
},
@ -23,15 +23,22 @@ UserPageController = RouteController.extend({
// Controller for user account editing
AccountController = RouteController.extend({
UserEditController = RouteController.extend({
waitOn: function() {
return coreSubscriptions.subscribe('invites');
return [
coreSubscriptions.subscribe('userProfile', this.params.slug),
coreSubscriptions.subscribe('invites', this.params.slug)
]
},
data: function() {
return {
user : Meteor.user(),
invites: Invites.find({invitingUserId:Meteor.userId()})
};
var user = Meteor.users.findOne({slug: this.params.slug});
var data = {
user: user
}
if (user) {
data.invites = Invites.find({invitingUserId: user._id});
}
return data;
},
fastRender: true
});
@ -59,18 +66,21 @@ Meteor.startup(function () {
// User Edit
Router.route('/users/:_idOrSlug/edit', {
Router.route('/users/:slug/edit', {
name: 'user_edit',
template: getTemplate('user_edit'),
controller: UserPageController
});
// Account
Router.route('/account', {
name: 'account',
template: getTemplate('user_edit'),
controller: AccountController
controller: UserEditController,
onBeforeAction: function () {
// Only allow users with permissions to see the user edit page.
if (Meteor.user() && (
isAdmin(Meteor.user()) ||
this.params.slug === Meteor.user().slug
)) {
this.next();
} else {
this.render(getTemplate('no_rights'));
}
}
});
// All Users

View file

@ -21,8 +21,10 @@ isInvited=function(user){
adminUsers = function(){
return Meteor.users.find({isAdmin : true}).fetch();
};
adminMongoQuery = {isAdmin: true};
notAdminMongoQuery = {isAdmin: false};
getUserName = function(user){
try{
if (user.username)
@ -42,13 +44,10 @@ getDisplayNameById = function(userId){
return getDisplayName(Meteor.users.findOne(userId));
};
getProfileUrl = function(user) {
return Meteor.absoluteUrl()+'users/' + slugify(getUserName(user));
return getProfileUrlBySlugOrId(user.slug);
};
getProfileUrlById = function(id){
return Meteor.absoluteUrl()+'users/'+ id;
};
getProfileUrlBySlug = function(slug) {
return Meteor.absoluteUrl()+'users/' + slug;
getProfileUrlBySlugOrId = function(slugOrId) {
return Router.routes['user_profile'].url({_idOrSlug: slugOrId});
};
hasPassword = function(user) {
return !!user.services.password;
@ -77,8 +76,8 @@ getTwitterNameById = function(userId){
getEmail = function(user){
if(user.profile && user.profile.email){
return user.profile.email;
}else{
return null;
}else{
return null;
}
};
getEmailHash = function(user){
@ -93,7 +92,12 @@ getCurrentUserEmail = function(){
return Meteor.user() ? getEmail(Meteor.user()) : '';
};
userProfileComplete = function(user) {
return !!getEmail(user);
for (var i = 0; i < userProfileCompleteChecks.length; i++) {
if (!userProfileCompleteChecks[i](user)) {
return false;
}
}
return true;
};
findLast = function(user, collection){

View file

@ -1 +1 @@
telescopeVersion = "0.9.10";
telescopeVersion = "0.9.11";

View file

@ -4,6 +4,32 @@
addToPostSchema = [];
addToCommentsSchema = [];
addToSettingsSchema = [];
addToUserSchema = [];
SimpleSchema.extendOptions({
editable: Match.Optional(Boolean), // editable: true means the field can be edited by the document's owner
hidden: Match.Optional(Boolean) // hidden: true means the field is never shown in a form no matter what
});
// ----------------------------------- Posts Statuses ------------------------------ //
postStatuses = [
{
value: 1,
label: 'Pending'
},
{
value: 2,
label: 'Approved'
},
{
value: 3,
label: 'Rejected'
}
]
STATUS_PENDING=1;
STATUS_APPROVED=2;
STATUS_REJECTED=3;
// ------------------------------------- Navigation -------------------------------- //
@ -64,7 +90,10 @@ viewParameters.pending = function (terms) {
return {
find: {
status: 1,
postedAt: {$lte: null}
$or: [
{postedAt: {$lte: new Date("December 31, 2999")}},
{postedAt: {$exists: false}}
] // for pending view, show future posts too
},
options: {sort: {createdAt: -1}}
};
@ -130,13 +159,17 @@ postHeading = [
template: 'postDomain',
order: 5
}
]
];
postMeta = [
{
template: 'postInfo',
template: 'postAuthor',
order: 1
},
{
template: 'postInfo',
order: 2
},
{
template: 'postCommentsLink',
order: 3
@ -156,7 +189,7 @@ postAfterSubmitMethodCallbacks = [];
postEditRenderedCallbacks = [];
postEditClientCallbacks = [];
postEditMethodCallbacks = []; // not used yet
postAfterMethodCallbacks = []; // not used yet
postAfterEditMethodCallbacks = []; // not used yet
commentSubmitRenderedCallbacks = [];
commentSubmitClientCallbacks = [];
@ -168,6 +201,16 @@ commentEditClientCallbacks = [];
commentEditMethodCallbacks = []; // not used yet
commentAfterEditMethodCallbacks = []; // not used yet
userEditRenderedCallbacks = [];
userEditClientCallbacks = [];
userProfileCompleteChecks = [
function(user) {
return !!getEmail(user) && !!getUserName(user);
}
];
// ------------------------------ Dynamic Templates ------------------------------ //

View file

@ -2,18 +2,25 @@ Package.describe({summary: "Telescope base package"});
Package.onUse(function (api) {
api.use(['telescope-i18n', 'telescope-lib'], ['client', 'server']);
api.use(['telescope-i18n', 'telescope-lib', 'aldeed:simple-schema', 'check']);
api.imply(['aldeed:simple-schema']);
api.add_files(['lib/base.js'], ['client', 'server']);
api.add_files(['lib/base_client.js'], ['client']);
api.add_files(['lib/base_server.js'], ['server']);
api.export([
'postStatuses',
'STATUS_PENDING',
'STATUS_APPROVED',
'STATUS_REJECTED',
'adminNav',
'viewNav',
'addToPostSchema',
'addToCommentsSchema',
'addToSettingsSchema',
'addToUserSchema',
'preloadSubscriptions',
'primaryNav',
'secondaryNav',
@ -44,10 +51,14 @@ Package.onUse(function (api) {
'commentEditClientCallbacks',
'commentEditMethodCallbacks',
'commentAfterEditMethodCallbacks',
'userEditRenderedCallbacks',
'userEditClientCallbacks',
'userProfileCompleteChecks',
'getTemplate',
'templates',
'themeSettings'
]);
});
});

View file

@ -1,5 +1,9 @@
{
"dependencies": [
[
"aldeed:simple-schema",
"1.1.0"
],
[
"application-configuration",
"1.0.3"

View file

@ -1 +1,14 @@
both.js
var customProperty = {
propertyName: 'customProperty',
propertySchema: {
type: String, // property type
label: 'customLabel', // key string used for internationalization
optional: true, // make this property optional
autoform: {
editable: true, // make this property editable by users
group: 'xyz', // assign a custom form group
type: "bootstrap-datetimepicker" // assign a custom input type
}
}
}
addToPostSchema.push(customProperty);

View file

@ -4,6 +4,10 @@
"accounts-base",
"1.1.2"
],
[
"aldeed:simple-schema",
"1.1.0"
],
[
"application-configuration",
"1.0.3"

View file

@ -0,0 +1,3 @@
<template name="afBootstrapDateTimePicker">
<input type="text" value="" {{atts}}/>
</template>

View file

@ -0,0 +1,102 @@
AutoForm.addInputType("bootstrap-datetimepicker", {
template: "afBootstrapDateTimePicker",
valueOut: function () {
// var val = this.datepicker('getUTCDate');
if (!!this.data("DateTimePicker").getDate()) {
var val = this.data("DateTimePicker").getDate().toDate();
// console.log(val)
return (val instanceof Date) ? val : this.val();
}
},
valueConverters: {
"string": function (val) {
return (val instanceof Date) ? AutoForm.Utility.dateToDateStringUTC(val) : val;
},
"stringArray": function (val) {
if (val instanceof Date) {
return [AutoForm.Utility.dateToDateStringUTC(val)];
}
return val;
},
"number": function (val) {
return (val instanceof Date) ? val.getTime() : val;
},
"numberArray": function (val) {
if (val instanceof Date) {
return [val.getTime()];
}
return val;
},
"dateArray": function (val) {
if (val instanceof Date) {
return [val];
}
return val;
}
}
});
Template.afBootstrapDateTimePicker.helpers({
atts: function addFormControlAtts() {
var atts = _.clone(this.atts);
// Add bootstrap class
atts = AutoForm.Utility.addClass(atts, "form-control");
return atts;
}
});
Template.afBootstrapDateTimePicker.rendered = function () {
var $input = this.$('input');
var data = this.data;
// instanciate datepicker
$input.datetimepicker(data.atts.datePickerOptions);
// set and reactively update values
this.autorun(function () {
var data = Template.currentData();
// set field value
if (data.value instanceof Date) {
// $input.datepicker('setUTCDate', data.value);
$input.data("DateTimePicker").setDate(data.value);
} else if (typeof data.value === "string") {
// $input.datepicker('update', data.value);
$input.data("DateTimePicker").setDate(moment(data.value).toDate());
}
// set start date if there's a min in the schema
if (data.min instanceof Date) {
// datepicker plugin expects local Date object, so convert UTC Date object to local
var startDate = utcToLocal(data.min);
// $input.datepicker('setStartDate', startDate);
$input.data("DateTimePicker").setMinDate(startDate);
}
// set end date if there's a max in the schema
if (data.max instanceof Date) {
// datepicker plugin expects local Date object, so convert UTC Date object to local
var endDate = utcToLocal(data.max);
// $input.datepicker('setEndDate', endDate);
$input.data("DateTimePicker").setMinDate(endDate);
}
});
};
Template.afBootstrapDateTimePicker.destroyed = function () {
// this.$('input').datepicker('remove');
this.$('input').data('DateTimePicker').destroy();
};
function utcToLocal(utcDate) {
var localDateObj = new Date();
localDateObj.setDate(utcDate.getUTCDate());
localDateObj.setMonth(utcDate.getUTCMonth());
localDateObj.setFullYear(utcDate.getUTCFullYear());
localDateObj.setHours(0);
localDateObj.setMinutes(0);
localDateObj.setSeconds(0);
localDateObj.setMilliseconds(0);
return localDateObj;
}

View file

@ -0,0 +1,291 @@
/*!
* Bootstrap v3.3.1 (http://getbootstrap.com)
* Copyright 2011-2014 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
*/
/*!
* Generated using the Bootstrap Customizer (http://getbootstrap.com/customize/?id=81631ba40c2459b10edf)
* Config saved to config.json and https://gist.github.com/81631ba40c2459b10edf
*/
if (typeof jQuery === 'undefined') {
throw new Error('Bootstrap\'s JavaScript requires jQuery')
}
+function ($) {
var version = $.fn.jquery.split(' ')[0].split('.')
if ((version[0] < 2 && version[1] < 9) || (version[0] == 1 && version[1] == 9 && version[2] < 1)) {
throw new Error('Bootstrap\'s JavaScript requires jQuery version 1.9.1 or higher')
}
}(jQuery);
/* ========================================================================
* Bootstrap: collapse.js v3.3.1
* http://getbootstrap.com/javascript/#collapse
* ========================================================================
* Copyright 2011-2014 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
* ======================================================================== */
+function ($) {
'use strict';
// COLLAPSE PUBLIC CLASS DEFINITION
// ================================
var Collapse = function (element, options) {
this.$element = $(element)
this.options = $.extend({}, Collapse.DEFAULTS, options)
this.$trigger = $(this.options.trigger).filter('[href="#' + element.id + '"], [data-target="#' + element.id + '"]')
this.transitioning = null
if (this.options.parent) {
this.$parent = this.getParent()
} else {
this.addAriaAndCollapsedClass(this.$element, this.$trigger)
}
if (this.options.toggle) this.toggle()
}
Collapse.VERSION = '3.3.1'
Collapse.TRANSITION_DURATION = 350
Collapse.DEFAULTS = {
toggle: true,
trigger: '[data-toggle="collapse"]'
}
Collapse.prototype.dimension = function () {
var hasWidth = this.$element.hasClass('width')
return hasWidth ? 'width' : 'height'
}
Collapse.prototype.show = function () {
if (this.transitioning || this.$element.hasClass('in')) return
var activesData
var actives = this.$parent && this.$parent.find('> .panel').children('.in, .collapsing')
if (actives && actives.length) {
activesData = actives.data('bs.collapse')
if (activesData && activesData.transitioning) return
}
var startEvent = $.Event('show.bs.collapse')
this.$element.trigger(startEvent)
if (startEvent.isDefaultPrevented()) return
if (actives && actives.length) {
Plugin.call(actives, 'hide')
activesData || actives.data('bs.collapse', null)
}
var dimension = this.dimension()
this.$element
.removeClass('collapse')
.addClass('collapsing')[dimension](0)
.attr('aria-expanded', true)
this.$trigger
.removeClass('collapsed')
.attr('aria-expanded', true)
this.transitioning = 1
var complete = function () {
this.$element
.removeClass('collapsing')
.addClass('collapse in')[dimension]('')
this.transitioning = 0
this.$element
.trigger('shown.bs.collapse')
}
if (!$.support.transition) return complete.call(this)
var scrollSize = $.camelCase(['scroll', dimension].join('-'))
this.$element
.one('bsTransitionEnd', $.proxy(complete, this))
.emulateTransitionEnd(Collapse.TRANSITION_DURATION)[dimension](this.$element[0][scrollSize])
}
Collapse.prototype.hide = function () {
if (this.transitioning || !this.$element.hasClass('in')) return
var startEvent = $.Event('hide.bs.collapse')
this.$element.trigger(startEvent)
if (startEvent.isDefaultPrevented()) return
var dimension = this.dimension()
this.$element[dimension](this.$element[dimension]())[0].offsetHeight
this.$element
.addClass('collapsing')
.removeClass('collapse in')
.attr('aria-expanded', false)
this.$trigger
.addClass('collapsed')
.attr('aria-expanded', false)
this.transitioning = 1
var complete = function () {
this.transitioning = 0
this.$element
.removeClass('collapsing')
.addClass('collapse')
.trigger('hidden.bs.collapse')
}
if (!$.support.transition) return complete.call(this)
this.$element
[dimension](0)
.one('bsTransitionEnd', $.proxy(complete, this))
.emulateTransitionEnd(Collapse.TRANSITION_DURATION)
}
Collapse.prototype.toggle = function () {
this[this.$element.hasClass('in') ? 'hide' : 'show']()
}
Collapse.prototype.getParent = function () {
return $(this.options.parent)
.find('[data-toggle="collapse"][data-parent="' + this.options.parent + '"]')
.each($.proxy(function (i, element) {
var $element = $(element)
this.addAriaAndCollapsedClass(getTargetFromTrigger($element), $element)
}, this))
.end()
}
Collapse.prototype.addAriaAndCollapsedClass = function ($element, $trigger) {
var isOpen = $element.hasClass('in')
$element.attr('aria-expanded', isOpen)
$trigger
.toggleClass('collapsed', !isOpen)
.attr('aria-expanded', isOpen)
}
function getTargetFromTrigger($trigger) {
var href
var target = $trigger.attr('data-target')
|| (href = $trigger.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') // strip for ie7
return $(target)
}
// COLLAPSE PLUGIN DEFINITION
// ==========================
function Plugin(option) {
return this.each(function () {
var $this = $(this)
var data = $this.data('bs.collapse')
var options = $.extend({}, Collapse.DEFAULTS, $this.data(), typeof option == 'object' && option)
if (!data && options.toggle && option == 'show') options.toggle = false
if (!data) $this.data('bs.collapse', (data = new Collapse(this, options)))
if (typeof option == 'string') data[option]()
})
}
var old = $.fn.collapse
$.fn.collapse = Plugin
$.fn.collapse.Constructor = Collapse
// COLLAPSE NO CONFLICT
// ====================
$.fn.collapse.noConflict = function () {
$.fn.collapse = old
return this
}
// COLLAPSE DATA-API
// =================
$(document).on('click.bs.collapse.data-api', '[data-toggle="collapse"]', function (e) {
var $this = $(this)
if (!$this.attr('data-target')) e.preventDefault()
var $target = getTargetFromTrigger($this)
var data = $target.data('bs.collapse')
var option = data ? 'toggle' : $.extend({}, $this.data(), { trigger: this })
Plugin.call($target, option)
})
}(jQuery);
/* ========================================================================
* Bootstrap: transition.js v3.3.1
* http://getbootstrap.com/javascript/#transitions
* ========================================================================
* Copyright 2011-2014 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
* ======================================================================== */
+function ($) {
'use strict';
// CSS TRANSITION SUPPORT (Shoutout: http://www.modernizr.com/)
// ============================================================
function transitionEnd() {
var el = document.createElement('bootstrap')
var transEndEventNames = {
WebkitTransition : 'webkitTransitionEnd',
MozTransition : 'transitionend',
OTransition : 'oTransitionEnd otransitionend',
transition : 'transitionend'
}
for (var name in transEndEventNames) {
if (el.style[name] !== undefined) {
return { end: transEndEventNames[name] }
}
}
return false // explicit for ie8 ( ._.)
}
// http://blog.alexmaccaw.com/css-transitions
$.fn.emulateTransitionEnd = function (duration) {
var called = false
var $el = this
$(this).one('bsTransitionEnd', function () { called = true })
var callback = function () { if (!called) $($el).trigger($.support.transition.end) }
setTimeout(callback, duration)
return this
}
$(function () {
$.support.transition = transitionEnd()
if (!$.support.transition) return
$.event.special.bsTransitionEnd = {
bindType: $.support.transition.end,
delegateType: $.support.transition.end,
handle: function (e) {
if ($(e.target).is(this)) return e.handleObj.handler.apply(this, arguments)
}
}
})
}(jQuery);

View file

@ -0,0 +1,52 @@
body{
.bootstrap-datetimepicker-widget{
background: white;
width: auto;
box-shadow: 0 6px 12px rgba(0, 0, 0, 0.175);
border: 1px solid rgba(0, 0, 0, 0.15);
font-size: 13px;
display: none;
.btn{
}
.collapse{
display: none;
&.in{
display: block;
}
}
}
}
@font-face {
font-family: 'Glyphicons Halflings';
src: url('/packages/telescope-datetimepicker/fonts/glyphicons-halflings-regular.eot');
src: url('/packages/telescope-datetimepicker/fonts/glyphicons-halflings-regular.eot?#iefix') format('embedded-opentype'), url('/packages/telescope-datetimepicker/fonts/glyphicons-halflings-regular.woff') format('woff'), url('/packages/telescope-datetimepicker/fonts/glyphicons-halflings-regular.ttf') format('truetype'), url('/packages/telescope-datetimepicker/fonts/glyphicons-halflings-regular.svg#glyphicons_halflingsregular') format('svg');
}
.glyphicon {
position: relative;
top: 1px;
display: inline-block;
font-family: 'Glyphicons Halflings';
font-style: normal;
font-weight: normal;
line-height: 1;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
.glyphicon-chevron-left:before {
content: "\e079";
}
.glyphicon-chevron-right:before {
content: "\e080";
}
.glyphicon-chevron-up:before {
content: "\e113";
}
.glyphicon-chevron-down:before {
content: "\e114";
}
.glyphicon-time:before {
content: "\e023";
}
.glyphicon-calendar:before {
content: "\e109";
}

View file

@ -0,0 +1,229 @@
<?xml version="1.0" standalone="no"?>
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" >
<svg xmlns="http://www.w3.org/2000/svg">
<metadata></metadata>
<defs>
<font id="glyphicons_halflingsregular" horiz-adv-x="1200" >
<font-face units-per-em="1200" ascent="960" descent="-240" />
<missing-glyph horiz-adv-x="500" />
<glyph />
<glyph />
<glyph unicode="&#xd;" />
<glyph unicode=" " />
<glyph unicode="*" d="M100 500v200h259l-183 183l141 141l183 -183v259h200v-259l183 183l141 -141l-183 -183h259v-200h-259l183 -183l-141 -141l-183 183v-259h-200v259l-183 -183l-141 141l183 183h-259z" />
<glyph unicode="+" d="M0 400v300h400v400h300v-400h400v-300h-400v-400h-300v400h-400z" />
<glyph unicode="&#xa0;" />
<glyph unicode="&#x2000;" horiz-adv-x="652" />
<glyph unicode="&#x2001;" horiz-adv-x="1304" />
<glyph unicode="&#x2002;" horiz-adv-x="652" />
<glyph unicode="&#x2003;" horiz-adv-x="1304" />
<glyph unicode="&#x2004;" horiz-adv-x="434" />
<glyph unicode="&#x2005;" horiz-adv-x="326" />
<glyph unicode="&#x2006;" horiz-adv-x="217" />
<glyph unicode="&#x2007;" horiz-adv-x="217" />
<glyph unicode="&#x2008;" horiz-adv-x="163" />
<glyph unicode="&#x2009;" horiz-adv-x="260" />
<glyph unicode="&#x200a;" horiz-adv-x="72" />
<glyph unicode="&#x202f;" horiz-adv-x="260" />
<glyph unicode="&#x205f;" horiz-adv-x="326" />
<glyph unicode="&#x20ac;" d="M100 500l100 100h113q0 47 5 100h-218l100 100h135q37 167 112 257q117 141 297 141q242 0 354 -189q60 -103 66 -209h-181q0 55 -25.5 99t-63.5 68t-75 36.5t-67 12.5q-24 0 -52.5 -10t-62.5 -32t-65.5 -67t-50.5 -107h379l-100 -100h-300q-6 -46 -6 -100h406l-100 -100 h-300q9 -74 33 -132t52.5 -91t62 -54.5t59 -29t46.5 -7.5q29 0 66 13t75 37t63.5 67.5t25.5 96.5h174q-31 -172 -128 -278q-107 -117 -274 -117q-205 0 -324 158q-36 46 -69 131.5t-45 205.5h-217z" />
<glyph unicode="&#x2212;" d="M200 400h900v300h-900v-300z" />
<glyph unicode="&#x25fc;" horiz-adv-x="500" d="M0 0z" />
<glyph unicode="&#x2601;" d="M-14 494q0 -80 56.5 -137t135.5 -57h750q120 0 205 86.5t85 207.5t-85 207t-205 86q-46 0 -90 -14q-44 97 -134.5 156.5t-200.5 59.5q-152 0 -260 -107.5t-108 -260.5q0 -25 2 -37q-66 -14 -108.5 -67.5t-42.5 -122.5z" />
<glyph unicode="&#x2709;" d="M0 100l400 400l200 -200l200 200l400 -400h-1200zM0 300v600l300 -300zM0 1100l600 -603l600 603h-1200zM900 600l300 300v-600z" />
<glyph unicode="&#x270f;" d="M-13 -13l333 112l-223 223zM187 403l214 -214l614 614l-214 214zM887 1103l214 -214l99 92q13 13 13 32.5t-13 33.5l-153 153q-15 13 -33 13t-33 -13z" />
<glyph unicode="&#xe001;" d="M0 1200h1200l-500 -550v-550h300v-100h-800v100h300v550z" />
<glyph unicode="&#xe002;" d="M14 84q18 -55 86 -75.5t147 5.5q65 21 109 69t44 90v606l600 155v-521q-64 16 -138 -7q-79 -26 -122.5 -83t-25.5 -111q18 -55 86 -75.5t147 4.5q70 23 111.5 63.5t41.5 95.5v881q0 10 -7 15.5t-17 2.5l-752 -193q-10 -3 -17 -12.5t-7 -19.5v-689q-64 17 -138 -7 q-79 -25 -122.5 -82t-25.5 -112z" />
<glyph unicode="&#xe003;" d="M23 693q0 200 142 342t342 142t342 -142t142 -342q0 -142 -78 -261l300 -300q7 -8 7 -18t-7 -18l-109 -109q-8 -7 -18 -7t-18 7l-300 300q-119 -78 -261 -78q-200 0 -342 142t-142 342zM176 693q0 -136 97 -233t234 -97t233.5 96.5t96.5 233.5t-96.5 233.5t-233.5 96.5 t-234 -97t-97 -233z" />
<glyph unicode="&#xe005;" d="M100 784q0 64 28 123t73 100.5t104.5 64t119 20.5t120 -38.5t104.5 -104.5q48 69 109.5 105t121.5 38t118.5 -20.5t102.5 -64t71 -100.5t27 -123q0 -57 -33.5 -117.5t-94 -124.5t-126.5 -127.5t-150 -152.5t-146 -174q-62 85 -145.5 174t-149.5 152.5t-126.5 127.5 t-94 124.5t-33.5 117.5z" />
<glyph unicode="&#xe006;" d="M-72 800h479l146 400h2l146 -400h472l-382 -278l145 -449l-384 275l-382 -275l146 447zM168 71l2 1z" />
<glyph unicode="&#xe007;" d="M-72 800h479l146 400h2l146 -400h472l-382 -278l145 -449l-384 275l-382 -275l146 447zM168 71l2 1zM237 700l196 -142l-73 -226l192 140l195 -141l-74 229l193 140h-235l-77 211l-78 -211h-239z" />
<glyph unicode="&#xe008;" d="M0 0v143l400 257v100q-37 0 -68.5 74.5t-31.5 125.5v200q0 124 88 212t212 88t212 -88t88 -212v-200q0 -51 -31.5 -125.5t-68.5 -74.5v-100l400 -257v-143h-1200z" />
<glyph unicode="&#xe009;" d="M0 0v1100h1200v-1100h-1200zM100 100h100v100h-100v-100zM100 300h100v100h-100v-100zM100 500h100v100h-100v-100zM100 700h100v100h-100v-100zM100 900h100v100h-100v-100zM300 100h600v400h-600v-400zM300 600h600v400h-600v-400zM1000 100h100v100h-100v-100z M1000 300h100v100h-100v-100zM1000 500h100v100h-100v-100zM1000 700h100v100h-100v-100zM1000 900h100v100h-100v-100z" />
<glyph unicode="&#xe010;" d="M0 50v400q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5zM0 650v400q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400 q-21 0 -35.5 14.5t-14.5 35.5zM600 50v400q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5v-400q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5zM600 650v400q0 21 14.5 35.5t35.5 14.5h400q21 0 35.5 -14.5t14.5 -35.5v-400 q0 -21 -14.5 -35.5t-35.5 -14.5h-400q-21 0 -35.5 14.5t-14.5 35.5z" />
<glyph unicode="&#xe011;" d="M0 50v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM0 450v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200 q-21 0 -35.5 14.5t-14.5 35.5zM0 850v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM400 50v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5 t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM400 450v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM400 850v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5 v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM800 50v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM800 450v200q0 21 14.5 35.5t35.5 14.5h200 q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM800 850v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5z" />
<glyph unicode="&#xe012;" d="M0 50v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM0 450q0 -21 14.5 -35.5t35.5 -14.5h200q21 0 35.5 14.5t14.5 35.5v200q0 21 -14.5 35.5t-35.5 14.5h-200q-21 0 -35.5 -14.5 t-14.5 -35.5v-200zM0 850v200q0 21 14.5 35.5t35.5 14.5h200q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5zM400 50v200q0 21 14.5 35.5t35.5 14.5h700q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5 t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5zM400 450v200q0 21 14.5 35.5t35.5 14.5h700q21 0 35.5 -14.5t14.5 -35.5v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5zM400 850v200q0 21 14.5 35.5t35.5 14.5h700q21 0 35.5 -14.5t14.5 -35.5 v-200q0 -21 -14.5 -35.5t-35.5 -14.5h-700q-21 0 -35.5 14.5t-14.5 35.5z" />
<glyph unicode="&#xe013;" d="M29 454l419 -420l818 820l-212 212l-607 -607l-206 207z" />
<glyph unicode="&#xe014;" d="M106 318l282 282l-282 282l212 212l282 -282l282 282l212 -212l-282 -282l282 -282l-212 -212l-282 282l-282 -282z" />
<glyph unicode="&#xe015;" d="M23 693q0 200 142 342t342 142t342 -142t142 -342q0 -142 -78 -261l300 -300q7 -8 7 -18t-7 -18l-109 -109q-8 -7 -18 -7t-18 7l-300 300q-119 -78 -261 -78q-200 0 -342 142t-142 342zM176 693q0 -136 97 -233t234 -97t233.5 96.5t96.5 233.5t-96.5 233.5t-233.5 96.5 t-234 -97t-97 -233zM300 600v200h100v100h200v-100h100v-200h-100v-100h-200v100h-100z" />
<glyph unicode="&#xe016;" d="M23 694q0 200 142 342t342 142t342 -142t142 -342q0 -141 -78 -262l300 -299q7 -7 7 -18t-7 -18l-109 -109q-8 -8 -18 -8t-18 8l-300 300q-119 -78 -261 -78q-200 0 -342 142t-142 342zM176 694q0 -136 97 -233t234 -97t233.5 97t96.5 233t-96.5 233t-233.5 97t-234 -97 t-97 -233zM300 601h400v200h-400v-200z" />
<glyph unicode="&#xe017;" d="M23 600q0 183 105 331t272 210v-166q-103 -55 -165 -155t-62 -220q0 -177 125 -302t302 -125t302 125t125 302q0 120 -62 220t-165 155v166q167 -62 272 -210t105 -331q0 -118 -45.5 -224.5t-123 -184t-184 -123t-224.5 -45.5t-224.5 45.5t-184 123t-123 184t-45.5 224.5 zM500 750q0 -21 14.5 -35.5t35.5 -14.5h100q21 0 35.5 14.5t14.5 35.5v400q0 21 -14.5 35.5t-35.5 14.5h-100q-21 0 -35.5 -14.5t-14.5 -35.5v-400z" />
<glyph unicode="&#xe018;" d="M100 1h200v300h-200v-300zM400 1v500h200v-500h-200zM700 1v800h200v-800h-200zM1000 1v1200h200v-1200h-200z" />
<glyph unicode="&#xe019;" d="M26 601q0 -33 6 -74l151 -38l2 -6q14 -49 38 -93l3 -5l-80 -134q45 -59 105 -105l133 81l5 -3q45 -26 94 -39l5 -2l38 -151q40 -5 74 -5q27 0 74 5l38 151l6 2q46 13 93 39l5 3l134 -81q56 44 104 105l-80 134l3 5q24 44 39 93l1 6l152 38q5 40 5 74q0 28 -5 73l-152 38 l-1 6q-16 51 -39 93l-3 5l80 134q-44 58 -104 105l-134 -81l-5 3q-45 25 -93 39l-6 1l-38 152q-40 5 -74 5q-27 0 -74 -5l-38 -152l-5 -1q-50 -14 -94 -39l-5 -3l-133 81q-59 -47 -105 -105l80 -134l-3 -5q-25 -47 -38 -93l-2 -6l-151 -38q-6 -48 -6 -73zM385 601 q0 88 63 151t152 63t152 -63t63 -151q0 -89 -63 -152t-152 -63t-152 63t-63 152z" />
<glyph unicode="&#xe020;" d="M100 1025v50q0 10 7.5 17.5t17.5 7.5h275v100q0 41 29.5 70.5t70.5 29.5h300q41 0 70.5 -29.5t29.5 -70.5v-100h275q10 0 17.5 -7.5t7.5 -17.5v-50q0 -11 -7 -18t-18 -7h-1050q-11 0 -18 7t-7 18zM200 100v800h900v-800q0 -41 -29.5 -71t-70.5 -30h-700q-41 0 -70.5 30 t-29.5 71zM300 100h100v700h-100v-700zM500 100h100v700h-100v-700zM500 1100h300v100h-300v-100zM700 100h100v700h-100v-700zM900 100h100v700h-100v-700z" />
<glyph unicode="&#xe021;" d="M1 601l656 644l644 -644h-200v-600h-300v400h-300v-400h-300v600h-200z" />
<glyph unicode="&#xe022;" d="M100 25v1150q0 11 7 18t18 7h475v-500h400v-675q0 -11 -7 -18t-18 -7h-850q-11 0 -18 7t-7 18zM700 800v300l300 -300h-300z" />
<glyph unicode="&#xe023;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -171 121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM500 500v400h100 v-300h200v-100h-300z" />
<glyph unicode="&#xe024;" d="M-100 0l431 1200h209l-21 -300h162l-20 300h208l431 -1200h-538l-41 400h-242l-40 -400h-539zM488 500h224l-27 300h-170z" />
<glyph unicode="&#xe025;" d="M0 0v400h490l-290 300h200v500h300v-500h200l-290 -300h490v-400h-1100zM813 200h175v100h-175v-100z" />
<glyph unicode="&#xe026;" d="M1 600q0 122 47.5 233t127.5 191t191 127.5t233 47.5t233 -47.5t191 -127.5t127.5 -191t47.5 -233t-47.5 -233t-127.5 -191t-191 -127.5t-233 -47.5t-233 47.5t-191 127.5t-127.5 191t-47.5 233zM188 600q0 -170 121 -291t291 -121t291 121t121 291t-121 291t-291 121 t-291 -121t-121 -291zM350 600h150v300h200v-300h150l-250 -300z" />
<glyph unicode="&#xe027;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -171 121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM350 600l250 300 l250 -300h-150v-300h-200v300h-150z" />
<glyph unicode="&#xe028;" d="M0 25v475l200 700h800l199 -700l1 -475q0 -11 -7 -18t-18 -7h-1150q-11 0 -18 7t-7 18zM200 500h200l50 -200h300l50 200h200l-97 500h-606z" />
<glyph unicode="&#xe029;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -172 121.5 -293t292.5 -121t292.5 121t121.5 293q0 171 -121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM500 397v401 l297 -200z" />
<glyph unicode="&#xe030;" d="M23 600q0 -118 45.5 -224.5t123 -184t184 -123t224.5 -45.5t224.5 45.5t184 123t123 184t45.5 224.5h-150q0 -177 -125 -302t-302 -125t-302 125t-125 302t125 302t302 125q136 0 246 -81l-146 -146h400v400l-145 -145q-157 122 -355 122q-118 0 -224.5 -45.5t-184 -123 t-123 -184t-45.5 -224.5z" />
<glyph unicode="&#xe031;" d="M23 600q0 118 45.5 224.5t123 184t184 123t224.5 45.5q198 0 355 -122l145 145v-400h-400l147 147q-112 80 -247 80q-177 0 -302 -125t-125 -302h-150zM100 0v400h400l-147 -147q112 -80 247 -80q177 0 302 125t125 302h150q0 -118 -45.5 -224.5t-123 -184t-184 -123 t-224.5 -45.5q-198 0 -355 122z" />
<glyph unicode="&#xe032;" d="M100 0h1100v1200h-1100v-1200zM200 100v900h900v-900h-900zM300 200v100h100v-100h-100zM300 400v100h100v-100h-100zM300 600v100h100v-100h-100zM300 800v100h100v-100h-100zM500 200h500v100h-500v-100zM500 400v100h500v-100h-500zM500 600v100h500v-100h-500z M500 800v100h500v-100h-500z" />
<glyph unicode="&#xe033;" d="M0 100v600q0 41 29.5 70.5t70.5 29.5h100v200q0 82 59 141t141 59h300q82 0 141 -59t59 -141v-200h100q41 0 70.5 -29.5t29.5 -70.5v-600q0 -41 -29.5 -70.5t-70.5 -29.5h-900q-41 0 -70.5 29.5t-29.5 70.5zM400 800h300v150q0 21 -14.5 35.5t-35.5 14.5h-200 q-21 0 -35.5 -14.5t-14.5 -35.5v-150z" />
<glyph unicode="&#xe034;" d="M100 0v1100h100v-1100h-100zM300 400q60 60 127.5 84t127.5 17.5t122 -23t119 -30t110 -11t103 42t91 120.5v500q-40 -81 -101.5 -115.5t-127.5 -29.5t-138 25t-139.5 40t-125.5 25t-103 -29.5t-65 -115.5v-500z" />
<glyph unicode="&#xe035;" d="M0 275q0 -11 7 -18t18 -7h50q11 0 18 7t7 18v300q0 127 70.5 231.5t184.5 161.5t245 57t245 -57t184.5 -161.5t70.5 -231.5v-300q0 -11 7 -18t18 -7h50q11 0 18 7t7 18v300q0 116 -49.5 227t-131 192.5t-192.5 131t-227 49.5t-227 -49.5t-192.5 -131t-131 -192.5 t-49.5 -227v-300zM200 20v460q0 8 6 14t14 6h160q8 0 14 -6t6 -14v-460q0 -8 -6 -14t-14 -6h-160q-8 0 -14 6t-6 14zM800 20v460q0 8 6 14t14 6h160q8 0 14 -6t6 -14v-460q0 -8 -6 -14t-14 -6h-160q-8 0 -14 6t-6 14z" />
<glyph unicode="&#xe036;" d="M0 400h300l300 -200v800l-300 -200h-300v-400zM688 459l141 141l-141 141l71 71l141 -141l141 141l71 -71l-141 -141l141 -141l-71 -71l-141 141l-141 -141z" />
<glyph unicode="&#xe037;" d="M0 400h300l300 -200v800l-300 -200h-300v-400zM700 857l69 53q111 -135 111 -310q0 -169 -106 -302l-67 54q86 110 86 248q0 146 -93 257z" />
<glyph unicode="&#xe038;" d="M0 401v400h300l300 200v-800l-300 200h-300zM702 858l69 53q111 -135 111 -310q0 -170 -106 -303l-67 55q86 110 86 248q0 145 -93 257zM889 951l7 -8q123 -151 123 -344q0 -189 -119 -339l-7 -8l81 -66l6 8q142 178 142 405q0 230 -144 408l-6 8z" />
<glyph unicode="&#xe039;" d="M0 0h500v500h-200v100h-100v-100h-200v-500zM0 600h100v100h400v100h100v100h-100v300h-500v-600zM100 100v300h300v-300h-300zM100 800v300h300v-300h-300zM200 200v100h100v-100h-100zM200 900h100v100h-100v-100zM500 500v100h300v-300h200v-100h-100v-100h-200v100 h-100v100h100v200h-200zM600 0v100h100v-100h-100zM600 1000h100v-300h200v-300h300v200h-200v100h200v500h-600v-200zM800 800v300h300v-300h-300zM900 0v100h300v-100h-300zM900 900v100h100v-100h-100zM1100 200v100h100v-100h-100z" />
<glyph unicode="&#xe040;" d="M0 200h100v1000h-100v-1000zM100 0v100h300v-100h-300zM200 200v1000h100v-1000h-100zM500 0v91h100v-91h-100zM500 200v1000h200v-1000h-200zM700 0v91h100v-91h-100zM800 200v1000h100v-1000h-100zM900 0v91h200v-91h-200zM1000 200v1000h200v-1000h-200z" />
<glyph unicode="&#xe041;" d="M0 700l1 475q0 10 7.5 17.5t17.5 7.5h474l700 -700l-500 -500zM148 953q0 -42 29 -71q30 -30 71.5 -30t71.5 30q29 29 29 71t-29 71q-30 30 -71.5 30t-71.5 -30q-29 -29 -29 -71z" />
<glyph unicode="&#xe042;" d="M1 700l1 475q0 11 7 18t18 7h474l700 -700l-500 -500zM148 953q0 -42 30 -71q29 -30 71 -30t71 30q30 29 30 71t-30 71q-29 30 -71 30t-71 -30q-30 -29 -30 -71zM701 1200h100l700 -700l-500 -500l-50 50l450 450z" />
<glyph unicode="&#xe043;" d="M100 0v1025l175 175h925v-1000l-100 -100v1000h-750l-100 -100h750v-1000h-900z" />
<glyph unicode="&#xe044;" d="M200 0l450 444l450 -443v1150q0 20 -14.5 35t-35.5 15h-800q-21 0 -35.5 -15t-14.5 -35v-1151z" />
<glyph unicode="&#xe045;" d="M0 100v700h200l100 -200h600l100 200h200v-700h-200v200h-800v-200h-200zM253 829l40 -124h592l62 124l-94 346q-2 11 -10 18t-18 7h-450q-10 0 -18 -7t-10 -18zM281 24l38 152q2 10 11.5 17t19.5 7h500q10 0 19.5 -7t11.5 -17l38 -152q2 -10 -3.5 -17t-15.5 -7h-600 q-10 0 -15.5 7t-3.5 17z" />
<glyph unicode="&#xe046;" d="M0 200q0 -41 29.5 -70.5t70.5 -29.5h1000q41 0 70.5 29.5t29.5 70.5v600q0 41 -29.5 70.5t-70.5 29.5h-150q-4 8 -11.5 21.5t-33 48t-53 61t-69 48t-83.5 21.5h-200q-41 0 -82 -20.5t-70 -50t-52 -59t-34 -50.5l-12 -20h-150q-41 0 -70.5 -29.5t-29.5 -70.5v-600z M356 500q0 100 72 172t172 72t172 -72t72 -172t-72 -172t-172 -72t-172 72t-72 172zM494 500q0 -44 31 -75t75 -31t75 31t31 75t-31 75t-75 31t-75 -31t-31 -75zM900 700v100h100v-100h-100z" />
<glyph unicode="&#xe047;" d="M53 0h365v66q-41 0 -72 11t-49 38t1 71l92 234h391l82 -222q16 -45 -5.5 -88.5t-74.5 -43.5v-66h417v66q-34 1 -74 43q-18 19 -33 42t-21 37l-6 13l-385 998h-93l-399 -1006q-24 -48 -52 -75q-12 -12 -33 -25t-36 -20l-15 -7v-66zM416 521l178 457l46 -140l116 -317h-340 z" />
<glyph unicode="&#xe048;" d="M100 0v89q41 7 70.5 32.5t29.5 65.5v827q0 28 -1 39.5t-5.5 26t-15.5 21t-29 14t-49 14.5v71l471 -1q120 0 213 -88t93 -228q0 -55 -11.5 -101.5t-28 -74t-33.5 -47.5t-28 -28l-12 -7q8 -3 21.5 -9t48 -31.5t60.5 -58t47.5 -91.5t21.5 -129q0 -84 -59 -156.5t-142 -111 t-162 -38.5h-500zM400 200h161q89 0 153 48.5t64 132.5q0 90 -62.5 154.5t-156.5 64.5h-159v-400zM400 700h139q76 0 130 61.5t54 138.5q0 82 -84 130.5t-239 48.5v-379z" />
<glyph unicode="&#xe049;" d="M200 0v57q77 7 134.5 40.5t65.5 80.5l173 849q10 56 -10 74t-91 37q-6 1 -10.5 2.5t-9.5 2.5v57h425l2 -57q-33 -8 -62 -25.5t-46 -37t-29.5 -38t-17.5 -30.5l-5 -12l-128 -825q-10 -52 14 -82t95 -36v-57h-500z" />
<glyph unicode="&#xe050;" d="M-75 200h75v800h-75l125 167l125 -167h-75v-800h75l-125 -167zM300 900v300h150h700h150v-300h-50q0 29 -8 48.5t-18.5 30t-33.5 15t-39.5 5.5t-50.5 1h-200v-850l100 -50v-100h-400v100l100 50v850h-200q-34 0 -50.5 -1t-40 -5.5t-33.5 -15t-18.5 -30t-8.5 -48.5h-49z " />
<glyph unicode="&#xe051;" d="M33 51l167 125v-75h800v75l167 -125l-167 -125v75h-800v-75zM100 901v300h150h700h150v-300h-50q0 29 -8 48.5t-18 30t-33.5 15t-40 5.5t-50.5 1h-200v-650l100 -50v-100h-400v100l100 50v650h-200q-34 0 -50.5 -1t-39.5 -5.5t-33.5 -15t-18.5 -30t-8 -48.5h-50z" />
<glyph unicode="&#xe052;" d="M0 50q0 -20 14.5 -35t35.5 -15h1100q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM0 350q0 -20 14.5 -35t35.5 -15h800q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-800q-21 0 -35.5 -14.5t-14.5 -35.5 v-100zM0 650q0 -20 14.5 -35t35.5 -15h1000q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-1000q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM0 950q0 -20 14.5 -35t35.5 -15h600q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-600q-21 0 -35.5 -14.5 t-14.5 -35.5v-100z" />
<glyph unicode="&#xe053;" d="M0 50q0 -20 14.5 -35t35.5 -15h1100q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM0 650q0 -20 14.5 -35t35.5 -15h1100q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5 v-100zM200 350q0 -20 14.5 -35t35.5 -15h700q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-700q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM200 950q0 -20 14.5 -35t35.5 -15h700q21 0 35.5 15t14.5 35v100q0 21 -14.5 35.5t-35.5 14.5h-700q-21 0 -35.5 -14.5 t-14.5 -35.5v-100z" />
<glyph unicode="&#xe054;" d="M0 50v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100q-21 0 -35.5 15t-14.5 35zM100 650v100q0 21 14.5 35.5t35.5 14.5h1000q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1000q-21 0 -35.5 15 t-14.5 35zM300 350v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-800q-21 0 -35.5 15t-14.5 35zM500 950v100q0 21 14.5 35.5t35.5 14.5h600q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-600 q-21 0 -35.5 15t-14.5 35z" />
<glyph unicode="&#xe055;" d="M0 50v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100q-21 0 -35.5 15t-14.5 35zM0 350v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100q-21 0 -35.5 15 t-14.5 35zM0 650v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100q-21 0 -35.5 15t-14.5 35zM0 950v100q0 21 14.5 35.5t35.5 14.5h1100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-1100 q-21 0 -35.5 15t-14.5 35z" />
<glyph unicode="&#xe056;" d="M0 50v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15t-14.5 35zM0 350v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15 t-14.5 35zM0 650v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15t-14.5 35zM0 950v100q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-100q-21 0 -35.5 15 t-14.5 35zM300 50v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-800q-21 0 -35.5 15t-14.5 35zM300 350v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-800 q-21 0 -35.5 15t-14.5 35zM300 650v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15h-800q-21 0 -35.5 15t-14.5 35zM300 950v100q0 21 14.5 35.5t35.5 14.5h800q21 0 35.5 -14.5t14.5 -35.5v-100q0 -20 -14.5 -35t-35.5 -15 h-800q-21 0 -35.5 15t-14.5 35z" />
<glyph unicode="&#xe057;" d="M-101 500v100h201v75l166 -125l-166 -125v75h-201zM300 0h100v1100h-100v-1100zM500 50q0 -20 14.5 -35t35.5 -15h600q20 0 35 15t15 35v100q0 21 -15 35.5t-35 14.5h-600q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM500 350q0 -20 14.5 -35t35.5 -15h300q20 0 35 15t15 35 v100q0 21 -15 35.5t-35 14.5h-300q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM500 650q0 -20 14.5 -35t35.5 -15h500q20 0 35 15t15 35v100q0 21 -15 35.5t-35 14.5h-500q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM500 950q0 -20 14.5 -35t35.5 -15h100q20 0 35 15t15 35v100 q0 21 -15 35.5t-35 14.5h-100q-21 0 -35.5 -14.5t-14.5 -35.5v-100z" />
<glyph unicode="&#xe058;" d="M1 50q0 -20 14.5 -35t35.5 -15h600q20 0 35 15t15 35v100q0 21 -15 35.5t-35 14.5h-600q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM1 350q0 -20 14.5 -35t35.5 -15h300q20 0 35 15t15 35v100q0 21 -15 35.5t-35 14.5h-300q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM1 650 q0 -20 14.5 -35t35.5 -15h500q20 0 35 15t15 35v100q0 21 -15 35.5t-35 14.5h-500q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM1 950q0 -20 14.5 -35t35.5 -15h100q20 0 35 15t15 35v100q0 21 -15 35.5t-35 14.5h-100q-21 0 -35.5 -14.5t-14.5 -35.5v-100zM801 0v1100h100v-1100 h-100zM934 550l167 -125v75h200v100h-200v75z" />
<glyph unicode="&#xe059;" d="M0 275v650q0 31 22 53t53 22h750q31 0 53 -22t22 -53v-650q0 -31 -22 -53t-53 -22h-750q-31 0 -53 22t-22 53zM900 600l300 300v-600z" />
<glyph unicode="&#xe060;" d="M0 44v1012q0 18 13 31t31 13h1112q19 0 31.5 -13t12.5 -31v-1012q0 -18 -12.5 -31t-31.5 -13h-1112q-18 0 -31 13t-13 31zM100 263l247 182l298 -131l-74 156l293 318l236 -288v500h-1000v-737zM208 750q0 56 39 95t95 39t95 -39t39 -95t-39 -95t-95 -39t-95 39t-39 95z " />
<glyph unicode="&#xe062;" d="M148 745q0 124 60.5 231.5t165 172t226.5 64.5q123 0 227 -63t164.5 -169.5t60.5 -229.5t-73 -272q-73 -114 -166.5 -237t-150.5 -189l-57 -66q-10 9 -27 26t-66.5 70.5t-96 109t-104 135.5t-100.5 155q-63 139 -63 262zM342 772q0 -107 75.5 -182.5t181.5 -75.5 q107 0 182.5 75.5t75.5 182.5t-75.5 182t-182.5 75t-182 -75.5t-75 -181.5z" />
<glyph unicode="&#xe063;" d="M1 600q0 122 47.5 233t127.5 191t191 127.5t233 47.5t233 -47.5t191 -127.5t127.5 -191t47.5 -233t-47.5 -233t-127.5 -191t-191 -127.5t-233 -47.5t-233 47.5t-191 127.5t-127.5 191t-47.5 233zM173 600q0 -177 125.5 -302t301.5 -125v854q-176 0 -301.5 -125 t-125.5 -302z" />
<glyph unicode="&#xe064;" d="M117 406q0 94 34 186t88.5 172.5t112 159t115 177t87.5 194.5q21 -71 57.5 -142.5t76 -130.5t83 -118.5t82 -117t70 -116t50 -125.5t18.5 -136q0 -89 -39 -165.5t-102 -126.5t-140 -79.5t-156 -33.5q-114 6 -211.5 53t-161.5 139t-64 210zM243 414q14 -82 59.5 -136 t136.5 -80l16 98q-7 6 -18 17t-34 48t-33 77q-15 73 -14 143.5t10 122.5l9 51q-92 -110 -119.5 -185t-12.5 -156z" />
<glyph unicode="&#xe065;" d="M0 400v300q0 165 117.5 282.5t282.5 117.5q366 -6 397 -14l-186 -186h-311q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v125l200 200v-225q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5 t-117.5 282.5zM436 341l161 50l412 412l-114 113l-405 -405zM995 1015l113 -113l113 113l-21 85l-92 28z" />
<glyph unicode="&#xe066;" d="M0 400v300q0 165 117.5 282.5t282.5 117.5h261l2 -80q-133 -32 -218 -120h-145q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5l200 153v-53q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5 zM423 524q30 38 81.5 64t103 35.5t99 14t77.5 3.5l29 -1v-209l360 324l-359 318v-216q-7 0 -19 -1t-48 -8t-69.5 -18.5t-76.5 -37t-76.5 -59t-62 -88t-39.5 -121.5z" />
<glyph unicode="&#xe067;" d="M0 400v300q0 165 117.5 282.5t282.5 117.5h300q61 0 127 -23l-178 -177h-349q-41 0 -70.5 -29.5t-29.5 -70.5v-500q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v69l200 200v-169q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5 t-117.5 282.5zM342 632l283 -284l567 567l-137 137l-430 -431l-146 147z" />
<glyph unicode="&#xe068;" d="M0 603l300 296v-198h200v200h-200l300 300l295 -300h-195v-200h200v198l300 -296l-300 -300v198h-200v-200h195l-295 -300l-300 300h200v200h-200v-198z" />
<glyph unicode="&#xe069;" d="M200 50v1000q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-437l500 487v-1100l-500 488v-438q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5z" />
<glyph unicode="&#xe070;" d="M0 50v1000q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-437l500 487v-487l500 487v-1100l-500 488v-488l-500 488v-438q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5z" />
<glyph unicode="&#xe071;" d="M136 550l564 550v-487l500 487v-1100l-500 488v-488z" />
<glyph unicode="&#xe072;" d="M200 0l900 550l-900 550v-1100z" />
<glyph unicode="&#xe073;" d="M200 150q0 -21 14.5 -35.5t35.5 -14.5h200q21 0 35.5 14.5t14.5 35.5v800q0 21 -14.5 35.5t-35.5 14.5h-200q-21 0 -35.5 -14.5t-14.5 -35.5v-800zM600 150q0 -21 14.5 -35.5t35.5 -14.5h200q21 0 35.5 14.5t14.5 35.5v800q0 21 -14.5 35.5t-35.5 14.5h-200 q-21 0 -35.5 -14.5t-14.5 -35.5v-800z" />
<glyph unicode="&#xe074;" d="M200 150q0 -20 14.5 -35t35.5 -15h800q21 0 35.5 15t14.5 35v800q0 21 -14.5 35.5t-35.5 14.5h-800q-21 0 -35.5 -14.5t-14.5 -35.5v-800z" />
<glyph unicode="&#xe075;" d="M0 0v1100l500 -487v487l564 -550l-564 -550v488z" />
<glyph unicode="&#xe076;" d="M0 0v1100l500 -487v487l500 -487v437q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-1000q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v438l-500 -488v488z" />
<glyph unicode="&#xe077;" d="M300 0v1100l500 -487v437q0 21 14.5 35.5t35.5 14.5h100q21 0 35.5 -14.5t14.5 -35.5v-1000q0 -21 -14.5 -35.5t-35.5 -14.5h-100q-21 0 -35.5 14.5t-14.5 35.5v438z" />
<glyph unicode="&#xe078;" d="M100 250v100q0 21 14.5 35.5t35.5 14.5h1000q21 0 35.5 -14.5t14.5 -35.5v-100q0 -21 -14.5 -35.5t-35.5 -14.5h-1000q-21 0 -35.5 14.5t-14.5 35.5zM100 500h1100l-550 564z" />
<glyph unicode="&#xe079;" d="M185 599l592 -592l240 240l-353 353l353 353l-240 240z" />
<glyph unicode="&#xe080;" d="M272 194l353 353l-353 353l241 240l572 -571l21 -22l-1 -1v-1l-592 -591z" />
<glyph unicode="&#xe081;" d="M3 600q0 162 80 299.5t217.5 217.5t299.5 80t299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5zM300 500h200v-200h200v200h200v200h-200v200h-200v-200h-200v-200z" />
<glyph unicode="&#xe082;" d="M3 600q0 162 80 299.5t217.5 217.5t299.5 80t299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5zM300 500h600v200h-600v-200z" />
<glyph unicode="&#xe083;" d="M3 600q0 162 80 299.5t217.5 217.5t299.5 80t299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5zM246 459l213 -213l141 142l141 -142l213 213l-142 141l142 141l-213 212l-141 -141l-141 142l-212 -213l141 -141 z" />
<glyph unicode="&#xe084;" d="M3 600q0 162 80 299.5t217.5 217.5t299.5 80t299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5zM270 551l276 -277l411 411l-175 174l-236 -236l-102 102z" />
<glyph unicode="&#xe085;" d="M3 600q0 162 80 299.5t217.5 217.5t299.5 80t299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5zM364 700h143q4 0 11.5 -1t11 -1t6.5 3t3 9t1 11t3.5 8.5t3.5 6t5.5 4t6.5 2.5t9 1.5t9 0.5h11.5h12.5 q19 0 30 -10t11 -26q0 -22 -4 -28t-27 -22q-5 -1 -12.5 -3t-27 -13.5t-34 -27t-26.5 -46t-11 -68.5h200q5 3 14 8t31.5 25.5t39.5 45.5t31 69t14 94q0 51 -17.5 89t-42 58t-58.5 32t-58.5 15t-51.5 3q-50 0 -90.5 -12t-75 -38.5t-53.5 -74.5t-19 -114zM500 300h200v100h-200 v-100z" />
<glyph unicode="&#xe086;" d="M3 600q0 162 80 299.5t217.5 217.5t299.5 80t299.5 -80t217.5 -217.5t80 -299.5t-80 -299.5t-217.5 -217.5t-299.5 -80t-299.5 80t-217.5 217.5t-80 299.5zM400 300h400v100h-100v300h-300v-100h100v-200h-100v-100zM500 800h200v100h-200v-100z" />
<glyph unicode="&#xe087;" d="M0 500v200h195q31 125 98.5 199.5t206.5 100.5v200h200v-200q54 -20 113 -60t112.5 -105.5t71.5 -134.5h203v-200h-203q-25 -102 -116.5 -186t-180.5 -117v-197h-200v197q-140 27 -208 102.5t-98 200.5h-194zM290 500q24 -73 79.5 -127.5t130.5 -78.5v206h200v-206 q149 48 201 206h-201v200h200q-25 74 -75.5 127t-124.5 77v-204h-200v203q-75 -23 -130 -77t-79 -126h209v-200h-210z" />
<glyph unicode="&#xe088;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -171 121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM356 465l135 135 l-135 135l109 109l135 -135l135 135l109 -109l-135 -135l135 -135l-109 -109l-135 135l-135 -135z" />
<glyph unicode="&#xe089;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -171 121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM322 537l141 141 l87 -87l204 205l142 -142l-346 -345z" />
<glyph unicode="&#xe090;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -115 62 -215l568 567q-100 62 -216 62q-171 0 -292.5 -121.5t-121.5 -292.5zM391 245q97 -59 209 -59q171 0 292.5 121.5t121.5 292.5 q0 112 -59 209z" />
<glyph unicode="&#xe091;" d="M0 547l600 453v-300h600v-300h-600v-301z" />
<glyph unicode="&#xe092;" d="M0 400v300h600v300l600 -453l-600 -448v301h-600z" />
<glyph unicode="&#xe093;" d="M204 600l450 600l444 -600h-298v-600h-300v600h-296z" />
<glyph unicode="&#xe094;" d="M104 600h296v600h300v-600h298l-449 -600z" />
<glyph unicode="&#xe095;" d="M0 200q6 132 41 238.5t103.5 193t184 138t271.5 59.5v271l600 -453l-600 -448v301q-95 -2 -183 -20t-170 -52t-147 -92.5t-100 -135.5z" />
<glyph unicode="&#xe096;" d="M0 0v400l129 -129l294 294l142 -142l-294 -294l129 -129h-400zM635 777l142 -142l294 294l129 -129v400h-400l129 -129z" />
<glyph unicode="&#xe097;" d="M34 176l295 295l-129 129h400v-400l-129 130l-295 -295zM600 600v400l129 -129l295 295l142 -141l-295 -295l129 -130h-400z" />
<glyph unicode="&#xe101;" d="M23 600q0 118 45.5 224.5t123 184t184 123t224.5 45.5t224.5 -45.5t184 -123t123 -184t45.5 -224.5t-45.5 -224.5t-123 -184t-184 -123t-224.5 -45.5t-224.5 45.5t-184 123t-123 184t-45.5 224.5zM456 851l58 -302q4 -20 21.5 -34.5t37.5 -14.5h54q20 0 37.5 14.5 t21.5 34.5l58 302q4 20 -8 34.5t-32 14.5h-207q-21 0 -33 -14.5t-8 -34.5zM500 300h200v100h-200v-100z" />
<glyph unicode="&#xe102;" d="M0 800h100v-200h400v300h200v-300h400v200h100v100h-111q1 1 1 6.5t-1.5 15t-3.5 17.5l-34 172q-11 39 -41.5 63t-69.5 24q-32 0 -61 -17l-239 -144q-22 -13 -40 -35q-19 24 -40 36l-238 144q-33 18 -62 18q-39 0 -69.5 -23t-40.5 -61l-35 -177q-2 -8 -3 -18t-1 -15v-6 h-111v-100zM100 0h400v400h-400v-400zM200 900q-3 0 14 48t36 96l18 47l213 -191h-281zM700 0v400h400v-400h-400zM731 900l202 197q5 -12 12 -32.5t23 -64t25 -72t7 -28.5h-269z" />
<glyph unicode="&#xe103;" d="M0 -22v143l216 193q-9 53 -13 83t-5.5 94t9 113t38.5 114t74 124q47 60 99.5 102.5t103 68t127.5 48t145.5 37.5t184.5 43.5t220 58.5q0 -189 -22 -343t-59 -258t-89 -181.5t-108.5 -120t-122 -68t-125.5 -30t-121.5 -1.5t-107.5 12.5t-87.5 17t-56.5 7.5l-99 -55z M238.5 300.5q19.5 -6.5 86.5 76.5q55 66 367 234q70 38 118.5 69.5t102 79t99 111.5t86.5 148q22 50 24 60t-6 19q-7 5 -17 5t-26.5 -14.5t-33.5 -39.5q-35 -51 -113.5 -108.5t-139.5 -89.5l-61 -32q-369 -197 -458 -401q-48 -111 -28.5 -117.5z" />
<glyph unicode="&#xe104;" d="M111 408q0 -33 5 -63q9 -56 44 -119.5t105 -108.5q31 -21 64 -16t62 23.5t57 49.5t48 61.5t35 60.5q32 66 39 184.5t-13 157.5q79 -80 122 -164t26 -184q-5 -33 -20.5 -69.5t-37.5 -80.5q-10 -19 -14.5 -29t-12 -26t-9 -23.5t-3 -19t2.5 -15.5t11 -9.5t19.5 -5t30.5 2.5 t42 8q57 20 91 34t87.5 44.5t87 64t65.5 88.5t47 122q38 172 -44.5 341.5t-246.5 278.5q22 -44 43 -129q39 -159 -32 -154q-15 2 -33 9q-79 33 -120.5 100t-44 175.5t48.5 257.5q-13 -8 -34 -23.5t-72.5 -66.5t-88.5 -105.5t-60 -138t-8 -166.5q2 -12 8 -41.5t8 -43t6 -39.5 t3.5 -39.5t-1 -33.5t-6 -31.5t-13.5 -24t-21 -20.5t-31 -12q-38 -10 -67 13t-40.5 61.5t-15 81.5t10.5 75q-52 -46 -83.5 -101t-39 -107t-7.5 -85z" />
<glyph unicode="&#xe105;" d="M-61 600l26 40q6 10 20 30t49 63.5t74.5 85.5t97 90t116.5 83.5t132.5 59t145.5 23.5t145.5 -23.5t132.5 -59t116.5 -83.5t97 -90t74.5 -85.5t49 -63.5t20 -30l26 -40l-26 -40q-6 -10 -20 -30t-49 -63.5t-74.5 -85.5t-97 -90t-116.5 -83.5t-132.5 -59t-145.5 -23.5 t-145.5 23.5t-132.5 59t-116.5 83.5t-97 90t-74.5 85.5t-49 63.5t-20 30zM120 600q7 -10 40.5 -58t56 -78.5t68 -77.5t87.5 -75t103 -49.5t125 -21.5t123.5 20t100.5 45.5t85.5 71.5t66.5 75.5t58 81.5t47 66q-1 1 -28.5 37.5t-42 55t-43.5 53t-57.5 63.5t-58.5 54 q49 -74 49 -163q0 -124 -88 -212t-212 -88t-212 88t-88 212q0 85 46 158q-102 -87 -226 -258zM377 656q49 -124 154 -191l105 105q-37 24 -75 72t-57 84l-20 36z" />
<glyph unicode="&#xe106;" d="M-61 600l26 40q6 10 20 30t49 63.5t74.5 85.5t97 90t116.5 83.5t132.5 59t145.5 23.5q61 0 121 -17l37 142h148l-314 -1200h-148l37 143q-82 21 -165 71.5t-140 102t-109.5 112t-72 88.5t-29.5 43zM120 600q210 -282 393 -336l37 141q-107 18 -178.5 101.5t-71.5 193.5 q0 85 46 158q-102 -87 -226 -258zM377 656q49 -124 154 -191l47 47l23 87q-30 28 -59 69t-44 68l-14 26zM780 161l38 145q22 15 44.5 34t46 44t40.5 44t41 50.5t33.5 43.5t33 44t24.5 34q-97 127 -140 175l39 146q67 -54 131.5 -125.5t87.5 -103.5t36 -52l26 -40l-26 -40 q-7 -12 -25.5 -38t-63.5 -79.5t-95.5 -102.5t-124 -100t-146.5 -79z" />
<glyph unicode="&#xe107;" d="M-97.5 34q13.5 -34 50.5 -34h1294q37 0 50.5 35.5t-7.5 67.5l-642 1056q-20 34 -48 36.5t-48 -29.5l-642 -1066q-21 -32 -7.5 -66zM155 200l445 723l445 -723h-345v100h-200v-100h-345zM500 600l100 -300l100 300v100h-200v-100z" />
<glyph unicode="&#xe108;" d="M100 262v41q0 20 11 44.5t26 38.5l363 325v339q0 62 44 106t106 44t106 -44t44 -106v-339l363 -325q15 -14 26 -38.5t11 -44.5v-41q0 -20 -12 -26.5t-29 5.5l-359 249v-263q100 -91 100 -113v-64q0 -20 -13 -28.5t-32 0.5l-94 78h-222l-94 -78q-19 -9 -32 -0.5t-13 28.5 v64q0 22 100 113v263l-359 -249q-17 -12 -29 -5.5t-12 26.5z" />
<glyph unicode="&#xe109;" d="M0 50q0 -20 14.5 -35t35.5 -15h1000q21 0 35.5 15t14.5 35v750h-1100v-750zM0 900h1100v150q0 21 -14.5 35.5t-35.5 14.5h-150v100h-100v-100h-500v100h-100v-100h-150q-21 0 -35.5 -14.5t-14.5 -35.5v-150zM100 100v100h100v-100h-100zM100 300v100h100v-100h-100z M100 500v100h100v-100h-100zM300 100v100h100v-100h-100zM300 300v100h100v-100h-100zM300 500v100h100v-100h-100zM500 100v100h100v-100h-100zM500 300v100h100v-100h-100zM500 500v100h100v-100h-100zM700 100v100h100v-100h-100zM700 300v100h100v-100h-100zM700 500 v100h100v-100h-100zM900 100v100h100v-100h-100zM900 300v100h100v-100h-100zM900 500v100h100v-100h-100z" />
<glyph unicode="&#xe110;" d="M0 200v200h259l600 600h241v198l300 -295l-300 -300v197h-159l-600 -600h-341zM0 800h259l122 -122l141 142l-181 180h-341v-200zM678 381l141 142l122 -123h159v198l300 -295l-300 -300v197h-241z" />
<glyph unicode="&#xe111;" d="M0 400v600q0 41 29.5 70.5t70.5 29.5h1000q41 0 70.5 -29.5t29.5 -70.5v-600q0 -41 -29.5 -70.5t-70.5 -29.5h-596l-304 -300v300h-100q-41 0 -70.5 29.5t-29.5 70.5z" />
<glyph unicode="&#xe112;" d="M100 600v200h300v-250q0 -113 6 -145q17 -92 102 -117q39 -11 92 -11q37 0 66.5 5.5t50 15.5t36 24t24 31.5t14 37.5t7 42t2.5 45t0 47v25v250h300v-200q0 -42 -3 -83t-15 -104t-31.5 -116t-58 -109.5t-89 -96.5t-129 -65.5t-174.5 -25.5t-174.5 25.5t-129 65.5t-89 96.5 t-58 109.5t-31.5 116t-15 104t-3 83zM100 900v300h300v-300h-300zM800 900v300h300v-300h-300z" />
<glyph unicode="&#xe113;" d="M-30 411l227 -227l352 353l353 -353l226 227l-578 579z" />
<glyph unicode="&#xe114;" d="M70 797l580 -579l578 579l-226 227l-353 -353l-352 353z" />
<glyph unicode="&#xe115;" d="M-198 700l299 283l300 -283h-203v-400h385l215 -200h-800v600h-196zM402 1000l215 -200h381v-400h-198l299 -283l299 283h-200v600h-796z" />
<glyph unicode="&#xe116;" d="M18 939q-5 24 10 42q14 19 39 19h896l38 162q5 17 18.5 27.5t30.5 10.5h94q20 0 35 -14.5t15 -35.5t-15 -35.5t-35 -14.5h-54l-201 -961q-2 -4 -6 -10.5t-19 -17.5t-33 -11h-31v-50q0 -20 -14.5 -35t-35.5 -15t-35.5 15t-14.5 35v50h-300v-50q0 -20 -14.5 -35t-35.5 -15 t-35.5 15t-14.5 35v50h-50q-21 0 -35.5 15t-14.5 35q0 21 14.5 35.5t35.5 14.5h535l48 200h-633q-32 0 -54.5 21t-27.5 43z" />
<glyph unicode="&#xe117;" d="M0 0v800h1200v-800h-1200zM0 900v100h200q0 41 29.5 70.5t70.5 29.5h300q41 0 70.5 -29.5t29.5 -70.5h500v-100h-1200z" />
<glyph unicode="&#xe118;" d="M1 0l300 700h1200l-300 -700h-1200zM1 400v600h200q0 41 29.5 70.5t70.5 29.5h300q41 0 70.5 -29.5t29.5 -70.5h500v-200h-1000z" />
<glyph unicode="&#xe119;" d="M302 300h198v600h-198l298 300l298 -300h-198v-600h198l-298 -300z" />
<glyph unicode="&#xe120;" d="M0 600l300 298v-198h600v198l300 -298l-300 -297v197h-600v-197z" />
<glyph unicode="&#xe121;" d="M0 100v100q0 41 29.5 70.5t70.5 29.5h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5zM31 400l172 739q5 22 23 41.5t38 19.5h672q19 0 37.5 -22.5t23.5 -45.5l172 -732h-1138zM800 100h100v100h-100v-100z M1000 100h100v100h-100v-100z" />
<glyph unicode="&#xe122;" d="M-101 600v50q0 24 25 49t50 38l25 13v-250l-11 5.5t-24 14t-30 21.5t-24 27.5t-11 31.5zM100 500v250v8v8v7t0.5 7t1.5 5.5t2 5t3 4t4.5 3.5t6 1.5t7.5 0.5h200l675 250v-850l-675 200h-38l47 -276q2 -12 -3 -17.5t-11 -6t-21 -0.5h-8h-83q-20 0 -34.5 14t-18.5 35 q-55 337 -55 351zM1100 200v850q0 21 14.5 35.5t35.5 14.5q20 0 35 -14.5t15 -35.5v-850q0 -20 -15 -35t-35 -15q-21 0 -35.5 15t-14.5 35z" />
<glyph unicode="&#xe123;" d="M74 350q0 21 13.5 35.5t33.5 14.5h18l117 173l63 327q15 77 76 140t144 83l-18 32q-6 19 3 32t29 13h94q20 0 29 -10.5t3 -29.5q-18 -36 -18 -37q83 -19 144 -82.5t76 -140.5l63 -327l118 -173h17q20 0 33.5 -14.5t13.5 -35.5q0 -20 -13 -40t-31 -27q-8 -3 -23 -8.5 t-65 -20t-103 -25t-132.5 -19.5t-158.5 -9q-125 0 -245.5 20.5t-178.5 40.5l-58 20q-18 7 -31 27.5t-13 40.5zM497 110q12 -49 40 -79.5t63 -30.5t63 30.5t39 79.5q-48 -6 -102 -6t-103 6z" />
<glyph unicode="&#xe124;" d="M21 445l233 -45l-78 -224l224 78l45 -233l155 179l155 -179l45 233l224 -78l-78 224l234 45l-180 155l180 156l-234 44l78 225l-224 -78l-45 233l-155 -180l-155 180l-45 -233l-224 78l78 -225l-233 -44l179 -156z" />
<glyph unicode="&#xe125;" d="M0 200h200v600h-200v-600zM300 275q0 -75 100 -75h61q124 -100 139 -100h250q46 0 83 57l238 344q29 31 29 74v100q0 44 -30.5 84.5t-69.5 40.5h-328q28 118 28 125v150q0 44 -30.5 84.5t-69.5 40.5h-50q-27 0 -51 -20t-38 -48l-96 -198l-145 -196q-20 -26 -20 -63v-400z M400 300v375l150 213l100 212h50v-175l-50 -225h450v-125l-250 -375h-214l-136 100h-100z" />
<glyph unicode="&#xe126;" d="M0 400v600h200v-600h-200zM300 525v400q0 75 100 75h61q124 100 139 100h250q46 0 83 -57l238 -344q29 -31 29 -74v-100q0 -44 -30.5 -84.5t-69.5 -40.5h-328q28 -118 28 -125v-150q0 -44 -30.5 -84.5t-69.5 -40.5h-50q-27 0 -51 20t-38 48l-96 198l-145 196 q-20 26 -20 63zM400 525l150 -212l100 -213h50v175l-50 225h450v125l-250 375h-214l-136 -100h-100v-375z" />
<glyph unicode="&#xe127;" d="M8 200v600h200v-600h-200zM308 275v525q0 17 14 35.5t28 28.5l14 9l362 230q14 6 25 6q17 0 29 -12l109 -112q14 -14 14 -34q0 -18 -11 -32l-85 -121h302q85 0 138.5 -38t53.5 -110t-54.5 -111t-138.5 -39h-107l-130 -339q-7 -22 -20.5 -41.5t-28.5 -19.5h-341 q-7 0 -90 81t-83 94zM408 289l100 -89h293l131 339q6 21 19.5 41t28.5 20h203q16 0 25 15t9 36q0 20 -9 34.5t-25 14.5h-457h-6.5h-7.5t-6.5 0.5t-6 1t-5 1.5t-5.5 2.5t-4 4t-4 5.5q-5 12 -5 20q0 14 10 27l147 183l-86 83l-339 -236v-503z" />
<glyph unicode="&#xe128;" d="M-101 651q0 72 54 110t139 38l302 -1l-85 121q-11 16 -11 32q0 21 14 34l109 113q13 12 29 12q11 0 25 -6l365 -230q7 -4 17 -10.5t26.5 -26t16.5 -36.5v-526q0 -13 -86 -93.5t-94 -80.5h-341q-16 0 -29.5 20t-19.5 41l-130 339h-107q-84 0 -139 39t-55 111zM-1 601h222 q15 0 28.5 -20.5t19.5 -40.5l131 -339h293l107 89v502l-343 237l-87 -83l145 -184q10 -11 10 -26q0 -11 -5 -20q-1 -3 -3.5 -5.5l-4 -4t-5 -2.5t-5.5 -1.5t-6.5 -1t-6.5 -0.5h-7.5h-6.5h-476v-100zM1000 201v600h200v-600h-200z" />
<glyph unicode="&#xe129;" d="M97 719l230 -363q4 -6 10.5 -15.5t26 -25t36.5 -15.5h525q13 0 94 83t81 90v342q0 15 -20 28.5t-41 19.5l-339 131v106q0 84 -39 139t-111 55t-110 -53.5t-38 -138.5v-302l-121 84q-15 12 -33.5 11.5t-32.5 -13.5l-112 -110q-22 -22 -6 -53zM172 739l83 86l183 -146 q22 -18 47 -5q3 1 5.5 3.5l4 4t2.5 5t1.5 5.5t1 6.5t0.5 6.5v7.5v6.5v456q0 22 25 31t50 -0.5t25 -30.5v-202q0 -16 20 -29.5t41 -19.5l339 -130v-294l-89 -100h-503zM400 0v200h600v-200h-600z" />
<glyph unicode="&#xe130;" d="M2 585q-16 -31 6 -53l112 -110q13 -13 32 -13.5t34 10.5l121 85q0 -51 -0.5 -153.5t-0.5 -148.5q0 -84 38.5 -138t110.5 -54t111 55t39 139v106l339 131q20 6 40.5 19.5t20.5 28.5v342q0 7 -81 90t-94 83h-525q-17 0 -35.5 -14t-28.5 -28l-10 -15zM77 565l236 339h503 l89 -100v-294l-340 -130q-20 -6 -40 -20t-20 -29v-202q0 -22 -25 -31t-50 0t-25 31v456v14.5t-1.5 11.5t-5 12t-9.5 7q-24 13 -46 -5l-184 -146zM305 1104v200h600v-200h-600z" />
<glyph unicode="&#xe131;" d="M5 597q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5q162 0 299.5 -80t217.5 -218t80 -300t-80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5zM298 701l2 -201h300l-2 -194l402 294l-402 298v-197h-300z" />
<glyph unicode="&#xe132;" d="M0 597q0 122 47.5 232.5t127.5 190.5t190.5 127.5t231.5 47.5q122 0 232.5 -47.5t190.5 -127.5t127.5 -190.5t47.5 -232.5q0 -162 -80 -299.5t-218 -217.5t-300 -80t-299.5 80t-217.5 217.5t-80 299.5zM200 600l402 -294l-2 194h300l2 201h-300v197z" />
<glyph unicode="&#xe133;" d="M5 597q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5q162 0 299.5 -80t217.5 -218t80 -300t-80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5zM300 600h200v-300h200v300h200l-300 400z" />
<glyph unicode="&#xe134;" d="M5 597q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5q162 0 299.5 -80t217.5 -218t80 -300t-80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5zM300 600l300 -400l300 400h-200v300h-200v-300h-200z" />
<glyph unicode="&#xe135;" d="M5 597q0 122 47.5 232.5t127.5 190.5t190.5 127.5t232.5 47.5q121 0 231.5 -47.5t190.5 -127.5t127.5 -190.5t47.5 -232.5q0 -162 -80 -299.5t-217.5 -217.5t-299.5 -80t-300 80t-218 217.5t-80 299.5zM254 780q-8 -33 5.5 -92.5t7.5 -87.5q0 -9 17 -44t16 -60 q12 0 23 -5.5t23 -15t20 -13.5q24 -12 108 -42q22 -8 53 -31.5t59.5 -38.5t57.5 -11q8 -18 -15 -55t-20 -57q42 -71 87 -80q0 -6 -3 -15.5t-3.5 -14.5t4.5 -17q104 -3 221 112q30 29 47 47t34.5 49t20.5 62q-14 9 -37 9.5t-36 7.5q-14 7 -49 15t-52 19q-9 0 -39.5 -0.5 t-46.5 -1.5t-39 -6.5t-39 -16.5q-50 -35 -66 -12q-4 2 -3.5 25.5t0.5 25.5q-6 13 -26.5 17t-24.5 7q2 22 -2 41t-16.5 28t-38.5 -20q-23 -25 -42 4q-19 28 -8 58q6 16 22 22q6 -1 26 -1.5t33.5 -4t19.5 -13.5q12 -19 32 -37.5t34 -27.5l14 -8q0 3 9.5 39.5t5.5 57.5 q-4 23 14.5 44.5t22.5 31.5q5 14 10 35t8.5 31t15.5 22.5t34 21.5q-6 18 10 37q8 0 23.5 -1.5t24.5 -1.5t20.5 4.5t20.5 15.5q-10 23 -30.5 42.5t-38 30t-49 26.5t-43.5 23q11 39 2 44q31 -13 58 -14.5t39 3.5l11 4q7 36 -16.5 53.5t-64.5 28.5t-56 23q-19 -3 -37 0 q-15 -12 -36.5 -21t-34.5 -12t-44 -8t-39 -6q-15 -3 -45.5 0.5t-45.5 -2.5q-21 -7 -52 -26.5t-34 -34.5q-3 -11 6.5 -22.5t8.5 -18.5q-3 -34 -27.5 -90.5t-29.5 -79.5zM518 916q3 12 16 30t16 25q10 -10 18.5 -10t14 6t14.5 14.5t16 12.5q0 -24 17 -66.5t17 -43.5 q-9 2 -31 5t-36 5t-32 8t-30 14zM692 1003h1h-1z" />
<glyph unicode="&#xe136;" d="M0 164.5q0 21.5 15 37.5l600 599q-33 101 6 201.5t135 154.5q164 92 306 -9l-259 -138l145 -232l251 126q13 -175 -151 -267q-123 -70 -253 -23l-596 -596q-15 -16 -36.5 -16t-36.5 16l-111 110q-15 15 -15 36.5z" />
<glyph unicode="&#xe137;" horiz-adv-x="1220" d="M0 196v100q0 41 29.5 70.5t70.5 29.5h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5zM0 596v100q0 41 29.5 70.5t70.5 29.5h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000 q-41 0 -70.5 29.5t-29.5 70.5zM0 996v100q0 41 29.5 70.5t70.5 29.5h1000q41 0 70.5 -29.5t29.5 -70.5v-100q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5zM600 596h500v100h-500v-100zM800 196h300v100h-300v-100zM900 996h200v100h-200v-100z" />
<glyph unicode="&#xe138;" d="M100 1100v100h1000v-100h-1000zM150 1000h900l-350 -500v-300l-200 -200v500z" />
<glyph unicode="&#xe139;" d="M0 200v200h1200v-200q0 -41 -29.5 -70.5t-70.5 -29.5h-1000q-41 0 -70.5 29.5t-29.5 70.5zM0 500v400q0 41 29.5 70.5t70.5 29.5h300v100q0 41 29.5 70.5t70.5 29.5h200q41 0 70.5 -29.5t29.5 -70.5v-100h300q41 0 70.5 -29.5t29.5 -70.5v-400h-500v100h-200v-100h-500z M500 1000h200v100h-200v-100z" />
<glyph unicode="&#xe140;" d="M0 0v400l129 -129l200 200l142 -142l-200 -200l129 -129h-400zM0 800l129 129l200 -200l142 142l-200 200l129 129h-400v-400zM729 329l142 142l200 -200l129 129v-400h-400l129 129zM729 871l200 200l-129 129h400v-400l-129 129l-200 -200z" />
<glyph unicode="&#xe141;" d="M0 596q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM182 596q0 -172 121.5 -293t292.5 -121t292.5 121t121.5 293q0 171 -121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM291 655 q0 23 15.5 38.5t38.5 15.5t39 -16t16 -38q0 -23 -16 -39t-39 -16q-22 0 -38 16t-16 39zM400 850q0 22 16 38.5t39 16.5q22 0 38 -16t16 -39t-16 -39t-38 -16q-23 0 -39 16.5t-16 38.5zM514 609q0 32 20.5 56.5t51.5 29.5l122 126l1 1q-9 14 -9 28q0 22 16 38.5t39 16.5 q22 0 38 -16t16 -39t-16 -39t-38 -16q-14 0 -29 10l-55 -145q17 -22 17 -51q0 -36 -25.5 -61.5t-61.5 -25.5t-61.5 25.5t-25.5 61.5zM800 655q0 22 16 38t39 16t38.5 -15.5t15.5 -38.5t-16 -39t-38 -16q-23 0 -39 16t-16 39z" />
<glyph unicode="&#xe142;" d="M-40 375q-13 -95 35 -173q35 -57 94 -89t129 -32q63 0 119 28q33 16 65 40.5t52.5 45.5t59.5 64q40 44 57 61l394 394q35 35 47 84t-3 96q-27 87 -117 104q-20 2 -29 2q-46 0 -78.5 -16.5t-67.5 -51.5l-389 -396l-7 -7l69 -67l377 373q20 22 39 38q23 23 50 23 q38 0 53 -36q16 -39 -20 -75l-547 -547q-52 -52 -125 -52q-55 0 -100 33t-54 96q-5 35 2.5 66t31.5 63t42 50t56 54q24 21 44 41l348 348q52 52 82.5 79.5t84 54t107.5 26.5q25 0 48 -4q95 -17 154 -94.5t51 -175.5q-7 -101 -98 -192l-252 -249l-253 -256l7 -7l69 -60 l517 511q67 67 95 157t11 183q-16 87 -67 154t-130 103q-69 33 -152 33q-107 0 -197 -55q-40 -24 -111 -95l-512 -512q-68 -68 -81 -163z" />
<glyph unicode="&#xe143;" d="M80 784q0 131 98.5 229.5t230.5 98.5q143 0 241 -129q103 129 246 129q129 0 226 -98.5t97 -229.5q0 -46 -17.5 -91t-61 -99t-77 -89.5t-104.5 -105.5q-197 -191 -293 -322l-17 -23l-16 23q-43 58 -100 122.5t-92 99.5t-101 100q-71 70 -104.5 105.5t-77 89.5t-61 99 t-17.5 91zM250 784q0 -27 30.5 -70t61.5 -75.5t95 -94.5l22 -22q93 -90 190 -201q82 92 195 203l12 12q64 62 97.5 97t64.5 79t31 72q0 71 -48 119.5t-105 48.5q-74 0 -132 -83l-118 -171l-114 174q-51 80 -123 80q-60 0 -109.5 -49.5t-49.5 -118.5z" />
<glyph unicode="&#xe144;" d="M57 353q0 -95 66 -159l141 -142q68 -66 159 -66q93 0 159 66l283 283q66 66 66 159t-66 159l-141 141q-8 9 -19 17l-105 -105l212 -212l-389 -389l-247 248l95 95l-18 18q-46 45 -75 101l-55 -55q-66 -66 -66 -159zM269 706q0 -93 66 -159l141 -141q7 -7 19 -17l105 105 l-212 212l389 389l247 -247l-95 -96l18 -17q47 -49 77 -100l29 29q35 35 62.5 88t27.5 96q0 93 -66 159l-141 141q-66 66 -159 66q-95 0 -159 -66l-283 -283q-66 -64 -66 -159z" />
<glyph unicode="&#xe145;" d="M200 100v953q0 21 30 46t81 48t129 38t163 15t162 -15t127 -38t79 -48t29 -46v-953q0 -41 -29.5 -70.5t-70.5 -29.5h-600q-41 0 -70.5 29.5t-29.5 70.5zM300 300h600v700h-600v-700zM496 150q0 -43 30.5 -73.5t73.5 -30.5t73.5 30.5t30.5 73.5t-30.5 73.5t-73.5 30.5 t-73.5 -30.5t-30.5 -73.5z" />
<glyph unicode="&#xe146;" d="M0 0l303 380l207 208l-210 212h300l267 279l-35 36q-15 14 -15 35t15 35q14 15 35 15t35 -15l283 -282q15 -15 15 -36t-15 -35q-14 -15 -35 -15t-35 15l-36 35l-279 -267v-300l-212 210l-208 -207z" />
<glyph unicode="&#xe148;" d="M295 433h139q5 -77 48.5 -126.5t117.5 -64.5v335q-6 1 -15.5 4t-11.5 3q-46 14 -79 26.5t-72 36t-62.5 52t-40 72.5t-16.5 99q0 92 44 159.5t109 101t144 40.5v78h100v-79q38 -4 72.5 -13.5t75.5 -31.5t71 -53.5t51.5 -84t24.5 -118.5h-159q-8 72 -35 109.5t-101 50.5 v-307l64 -14q34 -7 64 -16.5t70 -31.5t67.5 -52t47.5 -80.5t20 -112.5q0 -139 -89 -224t-244 -96v-77h-100v78q-152 17 -237 104q-40 40 -52.5 93.5t-15.5 139.5zM466 889q0 -29 8 -51t16.5 -34t29.5 -22.5t31 -13.5t38 -10q7 -2 11 -3v274q-61 -8 -97.5 -37.5t-36.5 -102.5 zM700 237q170 18 170 151q0 64 -44 99.5t-126 60.5v-311z" />
<glyph unicode="&#xe149;" d="M100 600v100h166q-24 49 -44 104q-10 26 -14.5 55.5t-3 72.5t25 90t68.5 87q97 88 263 88q129 0 230 -89t101 -208h-153q0 52 -34 89.5t-74 51.5t-76 14q-37 0 -79 -14.5t-62 -35.5q-41 -44 -41 -101q0 -28 16.5 -69.5t28 -62.5t41.5 -72h241v-100h-197q8 -50 -2.5 -115 t-31.5 -94q-41 -59 -99 -113q35 11 84 18t70 7q33 1 103 -16t103 -17q76 0 136 30l50 -147q-41 -25 -80.5 -36.5t-59 -13t-61.5 -1.5q-23 0 -128 33t-155 29q-39 -4 -82 -17t-66 -25l-24 -11l-55 145l16.5 11t15.5 10t13.5 9.5t14.5 12t14.5 14t17.5 18.5q48 55 54 126.5 t-30 142.5h-221z" />
<glyph unicode="&#xe150;" d="M2 300l298 -300l298 300h-198v900h-200v-900h-198zM602 900l298 300l298 -300h-198v-900h-200v900h-198z" />
<glyph unicode="&#xe151;" d="M2 300h198v900h200v-900h198l-298 -300zM700 0v200h100v-100h200v-100h-300zM700 400v100h300v-200h-99v-100h-100v100h99v100h-200zM700 700v500h300v-500h-100v100h-100v-100h-100zM801 900h100v200h-100v-200z" />
<glyph unicode="&#xe152;" d="M2 300h198v900h200v-900h198l-298 -300zM700 0v500h300v-500h-100v100h-100v-100h-100zM700 700v200h100v-100h200v-100h-300zM700 1100v100h300v-200h-99v-100h-100v100h99v100h-200zM801 200h100v200h-100v-200z" />
<glyph unicode="&#xe153;" d="M2 300l298 -300l298 300h-198v900h-200v-900h-198zM800 100v400h300v-500h-100v100h-200zM800 1100v100h200v-500h-100v400h-100zM901 200h100v200h-100v-200z" />
<glyph unicode="&#xe154;" d="M2 300l298 -300l298 300h-198v900h-200v-900h-198zM800 400v100h200v-500h-100v400h-100zM800 800v400h300v-500h-100v100h-200zM901 900h100v200h-100v-200z" />
<glyph unicode="&#xe155;" d="M2 300l298 -300l298 300h-198v900h-200v-900h-198zM700 100v200h500v-200h-500zM700 400v200h400v-200h-400zM700 700v200h300v-200h-300zM700 1000v200h200v-200h-200z" />
<glyph unicode="&#xe156;" d="M2 300l298 -300l298 300h-198v900h-200v-900h-198zM700 100v200h200v-200h-200zM700 400v200h300v-200h-300zM700 700v200h400v-200h-400zM700 1000v200h500v-200h-500z" />
<glyph unicode="&#xe157;" d="M0 400v300q0 165 117.5 282.5t282.5 117.5h300q162 0 281 -118.5t119 -281.5v-300q0 -165 -118.5 -282.5t-281.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5zM200 300q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5 h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500z" />
<glyph unicode="&#xe158;" d="M0 400v300q0 163 119 281.5t281 118.5h300q165 0 282.5 -117.5t117.5 -282.5v-300q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-163 0 -281.5 117.5t-118.5 282.5zM200 300q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5 h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500zM400 300l333 250l-333 250v-500z" />
<glyph unicode="&#xe159;" d="M0 400v300q0 163 117.5 281.5t282.5 118.5h300q163 0 281.5 -119t118.5 -281v-300q0 -165 -117.5 -282.5t-282.5 -117.5h-300q-165 0 -282.5 117.5t-117.5 282.5zM200 300q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5 h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500zM300 700l250 -333l250 333h-500z" />
<glyph unicode="&#xe160;" d="M0 400v300q0 165 117.5 282.5t282.5 117.5h300q165 0 282.5 -117.5t117.5 -282.5v-300q0 -162 -118.5 -281t-281.5 -119h-300q-165 0 -282.5 118.5t-117.5 281.5zM200 300q0 -41 29.5 -70.5t70.5 -29.5h500q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5 h-500q-41 0 -70.5 -29.5t-29.5 -70.5v-500zM300 400h500l-250 333z" />
<glyph unicode="&#xe161;" d="M0 400v300h300v200l400 -350l-400 -350v200h-300zM500 0v200h500q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5h-500v200h400q165 0 282.5 -117.5t117.5 -282.5v-300q0 -165 -117.5 -282.5t-282.5 -117.5h-400z" />
<glyph unicode="&#xe162;" d="M217 519q8 -19 31 -19h302q-155 -438 -160 -458q-5 -21 4 -32l9 -8h9q14 0 26 15q11 13 274.5 321.5t264.5 308.5q14 19 5 36q-8 17 -31 17l-301 -1q1 4 78 219.5t79 227.5q2 15 -5 27l-9 9h-9q-15 0 -25 -16q-4 -6 -98 -111.5t-228.5 -257t-209.5 -237.5q-16 -19 -6 -41 z" />
<glyph unicode="&#xe163;" d="M0 400q0 -165 117.5 -282.5t282.5 -117.5h300q47 0 100 15v185h-500q-41 0 -70.5 29.5t-29.5 70.5v500q0 41 29.5 70.5t70.5 29.5h500v185q-14 4 -114 7.5t-193 5.5l-93 2q-165 0 -282.5 -117.5t-117.5 -282.5v-300zM600 400v300h300v200l400 -350l-400 -350v200h-300z " />
<glyph unicode="&#xe164;" d="M0 400q0 -165 117.5 -282.5t282.5 -117.5h300q163 0 281.5 117.5t118.5 282.5v98l-78 73l-122 -123v-148q0 -41 -29.5 -70.5t-70.5 -29.5h-500q-41 0 -70.5 29.5t-29.5 70.5v500q0 41 29.5 70.5t70.5 29.5h156l118 122l-74 78h-100q-165 0 -282.5 -117.5t-117.5 -282.5 v-300zM496 709l353 342l-149 149h500v-500l-149 149l-342 -353z" />
<glyph unicode="&#xe165;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -171 121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM406 600 q0 80 57 137t137 57t137 -57t57 -137t-57 -137t-137 -57t-137 57t-57 137z" />
<glyph unicode="&#xe166;" d="M0 0v275q0 11 7 18t18 7h1048q11 0 19 -7.5t8 -17.5v-275h-1100zM100 800l445 -500l450 500h-295v400h-300v-400h-300zM900 150h100v50h-100v-50z" />
<glyph unicode="&#xe167;" d="M0 0v275q0 11 7 18t18 7h1048q11 0 19 -7.5t8 -17.5v-275h-1100zM100 700h300v-300h300v300h295l-445 500zM900 150h100v50h-100v-50z" />
<glyph unicode="&#xe168;" d="M0 0v275q0 11 7 18t18 7h1048q11 0 19 -7.5t8 -17.5v-275h-1100zM100 705l305 -305l596 596l-154 155l-442 -442l-150 151zM900 150h100v50h-100v-50z" />
<glyph unicode="&#xe169;" d="M0 0v275q0 11 7 18t18 7h1048q11 0 19 -7.5t8 -17.5v-275h-1100zM100 988l97 -98l212 213l-97 97zM200 400l697 1l3 699l-250 -239l-149 149l-212 -212l149 -149zM900 150h100v50h-100v-50z" />
<glyph unicode="&#xe170;" d="M0 0v275q0 11 7 18t18 7h1048q11 0 19 -7.5t8 -17.5v-275h-1100zM200 612l212 -212l98 97l-213 212zM300 1200l239 -250l-149 -149l212 -212l149 148l249 -237l-1 697zM900 150h100v50h-100v-50z" />
<glyph unicode="&#xe171;" d="M23 415l1177 784v-1079l-475 272l-310 -393v416h-392zM494 210l672 938l-672 -712v-226z" />
<glyph unicode="&#xe172;" d="M0 150v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-850q0 -21 -15 -35.5t-35 -14.5h-150v400h-700v-400h-150q-21 0 -35.5 14.5t-14.5 35.5zM600 1000h100v200h-100v-200z" />
<glyph unicode="&#xe173;" d="M0 150v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-218l-276 -275l-120 120l-126 -127h-378v-400h-150q-21 0 -35.5 14.5t-14.5 35.5zM581 306l123 123l120 -120l353 352l123 -123l-475 -476zM600 1000h100v200h-100v-200z" />
<glyph unicode="&#xe174;" d="M0 150v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-269l-103 -103l-170 170l-298 -298h-329v-400h-150q-21 0 -35.5 14.5t-14.5 35.5zM600 1000h100v200h-100v-200zM700 133l170 170l-170 170l127 127l170 -170l170 170l127 -128l-170 -169l170 -170 l-127 -127l-170 170l-170 -170z" />
<glyph unicode="&#xe175;" d="M0 150v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-300h-400v-200h-500v-400h-150q-21 0 -35.5 14.5t-14.5 35.5zM600 300l300 -300l300 300h-200v300h-200v-300h-200zM600 1000v200h100v-200h-100z" />
<glyph unicode="&#xe176;" d="M0 150v1000q0 20 14.5 35t35.5 15h250v-300h500v300h100l200 -200v-402l-200 200l-298 -298h-402v-400h-150q-21 0 -35.5 14.5t-14.5 35.5zM600 300h200v-300h200v300h200l-300 300zM600 1000v200h100v-200h-100z" />
<glyph unicode="&#xe177;" d="M0 250q0 -21 14.5 -35.5t35.5 -14.5h1100q21 0 35.5 14.5t14.5 35.5v550h-1200v-550zM0 900h1200v150q0 21 -14.5 35.5t-35.5 14.5h-1100q-21 0 -35.5 -14.5t-14.5 -35.5v-150zM100 300v200h400v-200h-400z" />
<glyph unicode="&#xe178;" d="M0 400l300 298v-198h400v-200h-400v-198zM100 800v200h100v-200h-100zM300 800v200h100v-200h-100zM500 800v200h400v198l300 -298l-300 -298v198h-400zM800 300v200h100v-200h-100zM1000 300h100v200h-100v-200z" />
<glyph unicode="&#xe179;" d="M100 700v400l50 100l50 -100v-300h100v300l50 100l50 -100v-300h100v300l50 100l50 -100v-400l-100 -203v-447q0 -21 -14.5 -35.5t-35.5 -14.5h-200q-21 0 -35.5 14.5t-14.5 35.5v447zM800 597q0 -29 10.5 -55.5t25 -43t29 -28.5t25.5 -18l10 -5v-397q0 -21 14.5 -35.5 t35.5 -14.5h200q21 0 35.5 14.5t14.5 35.5v1106q0 31 -18 40.5t-44 -7.5l-276 -116q-25 -17 -43.5 -51.5t-18.5 -65.5v-359z" />
<glyph unicode="&#xe180;" d="M100 0h400v56q-75 0 -87.5 6t-12.5 44v394h500v-394q0 -38 -12.5 -44t-87.5 -6v-56h400v56q-4 0 -11 0.5t-24 3t-30 7t-24 15t-11 24.5v888q0 22 25 34.5t50 13.5l25 2v56h-400v-56q75 0 87.5 -6t12.5 -44v-394h-500v394q0 38 12.5 44t87.5 6v56h-400v-56q4 0 11 -0.5 t24 -3t30 -7t24 -15t11 -24.5v-888q0 -22 -25 -34.5t-50 -13.5l-25 -2v-56z" />
<glyph unicode="&#xe181;" d="M0 300q0 -41 29.5 -70.5t70.5 -29.5h300q41 0 70.5 29.5t29.5 70.5v500q0 41 -29.5 70.5t-70.5 29.5h-300q-41 0 -70.5 -29.5t-29.5 -70.5v-500zM100 100h400l200 200h105l295 98v-298h-425l-100 -100h-375zM100 300v200h300v-200h-300zM100 600v200h300v-200h-300z M100 1000h400l200 -200v-98l295 98h105v200h-425l-100 100h-375zM700 402v163l400 133v-163z" />
<glyph unicode="&#xe182;" d="M16.5 974.5q0.5 -21.5 16 -90t46.5 -140t104 -177.5t175 -208q103 -103 207.5 -176t180 -103.5t137 -47t92.5 -16.5l31 1l163 162q17 18 13.5 41t-22.5 37l-192 136q-19 14 -45 12t-42 -19l-118 -118q-142 101 -268 227t-227 268l118 118q17 17 20 41.5t-11 44.5 l-139 194q-14 19 -36.5 22t-40.5 -14l-162 -162q-1 -11 -0.5 -32.5z" />
<glyph unicode="&#xe183;" d="M0 50v212q0 20 10.5 45.5t24.5 39.5l365 303v50q0 4 1 10.5t12 22.5t30 28.5t60 23t97 10.5t97 -10t60 -23.5t30 -27.5t12 -24l1 -10v-50l365 -303q14 -14 24.5 -39.5t10.5 -45.5v-212q0 -21 -14.5 -35.5t-35.5 -14.5h-1100q-20 0 -35 14.5t-15 35.5zM0 712 q0 -21 14.5 -33.5t34.5 -8.5l202 33q20 4 34.5 21t14.5 38v146q141 24 300 24t300 -24v-146q0 -21 14.5 -38t34.5 -21l202 -33q20 -4 34.5 8.5t14.5 33.5v200q-6 8 -19 20.5t-63 45t-112 57t-171 45t-235 20.5q-92 0 -175 -10.5t-141.5 -27t-108.5 -36.5t-81.5 -40 t-53.5 -36.5t-31 -27.5l-9 -10v-200z" />
<glyph unicode="&#xe184;" d="M100 0v100h1100v-100h-1100zM175 200h950l-125 150v250l100 100v400h-100v-200h-100v200h-200v-200h-100v200h-200v-200h-100v200h-100v-400l100 -100v-250z" />
<glyph unicode="&#xe185;" d="M100 0h300v400q0 41 -29.5 70.5t-70.5 29.5h-100q-41 0 -70.5 -29.5t-29.5 -70.5v-400zM500 0v1000q0 41 29.5 70.5t70.5 29.5h100q41 0 70.5 -29.5t29.5 -70.5v-1000h-300zM900 0v700q0 41 29.5 70.5t70.5 29.5h100q41 0 70.5 -29.5t29.5 -70.5v-700h-300z" />
<glyph unicode="&#xe186;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 300h300v300h-200v100h200v100h-300v-300h200v-100h-200v-100zM600 300h200v100h100v300h-100v100h-200v-500 zM700 400v300h100v-300h-100z" />
<glyph unicode="&#xe187;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 300h100v200h100v-200h100v500h-100v-200h-100v200h-100v-500zM600 300h200v100h100v300h-100v100h-200v-500 zM700 400v300h100v-300h-100z" />
<glyph unicode="&#xe188;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 300h300v100h-200v300h200v100h-300v-500zM600 300h300v100h-200v300h200v100h-300v-500z" />
<glyph unicode="&#xe189;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 550l300 -150v300zM600 400l300 150l-300 150v-300z" />
<glyph unicode="&#xe190;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 300v500h700v-500h-700zM300 400h130q41 0 68 42t27 107t-28.5 108t-66.5 43h-130v-300zM575 549 q0 -65 27 -107t68 -42h130v300h-130q-38 0 -66.5 -43t-28.5 -108z" />
<glyph unicode="&#xe191;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 300h300v300h-200v100h200v100h-300v-300h200v-100h-200v-100zM601 300h100v100h-100v-100zM700 700h100 v-400h100v500h-200v-100z" />
<glyph unicode="&#xe192;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 300h300v400h-200v100h-100v-500zM301 400v200h100v-200h-100zM601 300h100v100h-100v-100zM700 700h100 v-400h100v500h-200v-100z" />
<glyph unicode="&#xe193;" d="M-100 300v500q0 124 88 212t212 88h700q124 0 212 -88t88 -212v-500q0 -124 -88 -212t-212 -88h-700q-124 0 -212 88t-88 212zM100 200h900v700h-900v-700zM200 700v100h300v-300h-99v-100h-100v100h99v200h-200zM201 300v100h100v-100h-100zM601 300v100h100v-100h-100z M700 700v100h200v-500h-100v400h-100z" />
<glyph unicode="&#xe194;" d="M4 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM186 600q0 -171 121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM400 500v200 l100 100h300v-100h-300v-200h300v-100h-300z" />
<glyph unicode="&#xe195;" d="M0 600q0 162 80 299t217 217t299 80t299 -80t217 -217t80 -299t-80 -299t-217 -217t-299 -80t-299 80t-217 217t-80 299zM182 600q0 -171 121.5 -292.5t292.5 -121.5t292.5 121.5t121.5 292.5t-121.5 292.5t-292.5 121.5t-292.5 -121.5t-121.5 -292.5zM400 400v400h300 l100 -100v-100h-100v100h-200v-100h200v-100h-200v-100h-100zM700 400v100h100v-100h-100z" />
<glyph unicode="&#xe197;" d="M-14 494q0 -80 56.5 -137t135.5 -57h222v300h400v-300h128q120 0 205 86.5t85 207.5t-85 207t-205 86q-46 0 -90 -14q-44 97 -134.5 156.5t-200.5 59.5q-152 0 -260 -107.5t-108 -260.5q0 -25 2 -37q-66 -14 -108.5 -67.5t-42.5 -122.5zM300 200h200v300h200v-300h200 l-300 -300z" />
<glyph unicode="&#xe198;" d="M-14 494q0 -80 56.5 -137t135.5 -57h8l414 414l403 -403q94 26 154.5 104.5t60.5 178.5q0 120 -85 206.5t-205 86.5q-46 0 -90 -14q-44 97 -134.5 156.5t-200.5 59.5q-152 0 -260 -107.5t-108 -260.5q0 -25 2 -37q-66 -14 -108.5 -67.5t-42.5 -122.5zM300 200l300 300 l300 -300h-200v-300h-200v300h-200z" />
<glyph unicode="&#xe199;" d="M100 200h400v-155l-75 -45h350l-75 45v155h400l-270 300h170l-270 300h170l-300 333l-300 -333h170l-270 -300h170z" />
<glyph unicode="&#xe200;" d="M121 700q0 -53 28.5 -97t75.5 -65q-4 -16 -4 -38q0 -74 52.5 -126.5t126.5 -52.5q56 0 100 30v-306l-75 -45h350l-75 45v306q46 -30 100 -30q74 0 126.5 52.5t52.5 126.5q0 24 -9 55q50 32 79.5 83t29.5 112q0 90 -61.5 155.5t-150.5 71.5q-26 89 -99.5 145.5 t-167.5 56.5q-116 0 -197.5 -81.5t-81.5 -197.5q0 -4 1 -11.5t1 -11.5q-14 2 -23 2q-74 0 -126.5 -52.5t-52.5 -126.5z" />
</font>
</defs></svg>

After

Width:  |  Height:  |  Size: 62 KiB

View file

@ -0,0 +1,26 @@
Package.describe({
name: 'telescope-datetimepicker',
summary: 'Custom bootstrap-datetimepicker input type for AutoForm',
version: '1.0.3',
});
Package.onUse(function(api) {
api.use('templating@1.0.0');
api.use('blaze@2.0.0');
api.use('aldeed:autoform@4.0.0');
api.use('fourseven:scss');
// api.use('jquery');
// api.use('tsega:bootstrap3-datetimepicker');
// api.use('chriswessels:glyphicons-halflings');
api.addFiles([
'datetimepicker.scss',
'autoform-bs-datetimepicker.html',
'autoform-bs-datetimepicker.js',
'bootstrap-collapse-transitions.js',
'fonts/glyphicons-halflings-regular.eot',
'fonts/glyphicons-halflings-regular.svg',
'fonts/glyphicons-halflings-regular.ttf',
'fonts/glyphicons-halflings-regular.woff'
], 'client');
});

View file

@ -0,0 +1,119 @@
{
"dependencies": [
[
"aldeed:autoform",
"4.0.2"
],
[
"aldeed:simple-schema",
"1.1.0"
],
[
"base64",
"1.0.1"
],
[
"blaze",
"2.0.3"
],
[
"check",
"1.0.2"
],
[
"ddp",
"1.0.11"
],
[
"deps",
"1.0.5"
],
[
"ejson",
"1.0.4"
],
[
"fourseven:scss",
"1.0.0"
],
[
"geojson-utils",
"1.0.1"
],
[
"htmljs",
"1.0.2"
],
[
"id-map",
"1.0.1"
],
[
"jquery",
"1.0.1"
],
[
"json",
"1.0.1"
],
[
"livedata",
"1.0.11"
],
[
"logging",
"1.0.5"
],
[
"meteor",
"1.1.3"
],
[
"minimongo",
"1.0.5"
],
[
"mrt:moment",
"2.8.1"
],
[
"observe-sequence",
"1.0.3"
],
[
"ordered-dict",
"1.0.1"
],
[
"random",
"1.0.1"
],
[
"reactive-var",
"1.0.3"
],
[
"retry",
"1.0.1"
],
[
"templating",
"1.0.9"
],
[
"tracker",
"1.0.3"
],
[
"ui",
"1.0.4"
],
[
"underscore",
"1.0.1"
]
],
"pluginDependencies": [],
"toolVersion": "meteor-tool@1.0.35",
"format": "1.0"
}

View file

@ -24,7 +24,6 @@ Meteor.startup(function () {
where: 'server',
action: function() {
var post = Posts.findOne(this.params.id);
console.log(templates)
if (!!post) {
html = getEmailTemplate('emailNewPost')(getPostProperties(post));
} else {

View file

@ -57,7 +57,7 @@
color: white;
font-family: Helvetica, sans-serif;
font-weight: bold;
font-size: 20px;
font-size: 20px;
}
.main-container{
line-height: 1.7;
@ -106,11 +106,13 @@
<table border="0" width="600" cellpadding="0" cellspacing="0" class="container" bgcolor="#ffffff">
<tr>
<td class="heading-container">
{{#if logoUrl}}
<img class="logo" src="{{logoUrl}}" height="{{logoHeight}}" width="{{logoWidth}}" alt="{{siteName}}"/>
{{else}}
{{siteName}}
{{/if}}
<a href="{{siteUrl}}">
{{#if logoUrl}}
<img class="logo" src="{{logoUrl}}" height="{{logoHeight}}" width="{{logoWidth}}" alt="{{siteName}}"/>
{{else}}
{{siteName}}
{{/if}}
</a>
</td>
</tr>
<tr>

1
packages/telescope-embedly/.gitignore vendored Normal file
View file

@ -0,0 +1 @@
.build*

View file

@ -0,0 +1,4 @@
{
"thumbnail": "Thumbnail",
"thumbnailUrl": "Thumbnail"
}

View file

@ -0,0 +1,4 @@
{
"thumbnail": "Aperçu",
"thumbnailUrl": "Aperçu"
}

View file

@ -0,0 +1,7 @@
<template name="afPostThumbnail">
<div class="post-thumbnail-container" style="{{style}}">
<img src="{{this.value}}" class="post-thumbnail-preview" style="{{style}}"/>
<div class="post-thumbnail-loading">{{>spinner}}</div>
</div>
<input type="hidden" value="{{this.value}}" {{this.atts}}/>
</template>

View file

@ -0,0 +1,63 @@
AutoForm.addInputType("bootstrap-postthumbnail", {
template: "afPostThumbnail"
});
Template.afPostThumbnail.helpers({
atts: function addFormControlAtts() {
var atts = _.clone(this.atts);
// Add bootstrap class
atts = AutoForm.Utility.addClass(atts, "form-control");
return atts;
},
style: function () {
var thumbnailWidth = getSetting('thumbnailWidth', 200);
var thumbnailHeight = getSetting('thumbnailHeight', 125);
return "width: "+thumbnailWidth+"px; height: "+thumbnailHeight+"px;"
}
});
Template.afPostThumbnail.rendered = function () {
var $img = this.$('.post-thumbnail-preview');
var $thumbnailUrlField = this.$('[name="thumbnailUrl"]');
// note: the following fields are *not* in the current template
var $urlField = $('[name="url"]');
var $titleField = $('[name="title"]');
var $bodyField = $('[name="body"]');
var $thumbnailContainer = $('.post-thumbnail-container');
$urlField.change(function (e) {
var url = $urlField.val();
if (!!url) {
$thumbnailContainer.addClass('loading');
clearSeenMessages();
console.log('getting embedly data for '+url);
Meteor.call('getEmbedlyData', url, function (error, data) {
if (error) {
console.log(error)
flashMessage(error.reason, 'error');
$thumbnailContainer.removeClass('loading');
return
}
if (data) {
// set thumbnail and fill in thumbnailUrl field
$img.attr('src', data.thumbnailUrl);
$thumbnailUrlField.val(data.thumbnailUrl);
// remove loading class
$thumbnailContainer.removeClass('loading');
if (!$titleField.val()) // if title field is empty, fill in title
$titleField.val(data.title);
if (!$bodyField.val()) // if body field is empty, fill in body
$bodyField.val(data.description);
}
});
}
});
}

View file

@ -8,7 +8,6 @@
position: relative;
overflow: hidden;
max-width: 80px;
max-height: 50px;
}
.post-thumbnail-image{
@ -76,4 +75,25 @@
font-size: 40px;
text-align: center;
line-height: 40px;
}
}
.post-thumbnail-container{
position: relative;
}
.post-thumbnail-preview{
background: #eee;
}
.post-thumbnail-loading{
display: none;
position: absolute;
z-index: 100;
top: 0px;
left: 0px;
bottom: 0px;
right: 0px;
.loading & {
background: rgba(255,255,255,0.4);
display: block;
}
}

View file

@ -0,0 +1,70 @@
var thumbnailProperty = {
propertyName: 'thumbnailUrl',
propertySchema: {
type: String,
label: 'thumbnail',
optional: true,
autoform: {
editable: true,
type: 'bootstrap-postthumbnail'
}
}
}
addToPostSchema.push(thumbnailProperty);
var mediaProperty = {
propertyName: 'media',
propertySchema: {
type: Object,
optional: true,
blackbox: true,
hidden: true,
autoform: {
omit: true
}
}
}
addToPostSchema.push(mediaProperty);
postModules.push({
template: 'postThumbnail',
position: 'center-left'
});
var embedlyKeyProperty = {
propertyName: 'embedlyKey',
propertySchema: {
type: String,
optional: true,
autoform: {
group: 'embedly',
private: true
}
}
}
addToSettingsSchema.push(embedlyKeyProperty);
var thumbnailWidthProperty = {
propertyName: 'thumbnailWidth',
propertySchema: {
type: Number,
optional: true,
autoform: {
group: 'embedly'
}
}
}
addToSettingsSchema.push(thumbnailWidthProperty);
var thumbnailHeightProperty = {
propertyName: 'thumbnailHeight',
propertySchema: {
type: Number,
optional: true,
autoform: {
group: 'embedly'
}
}
}
addToSettingsSchema.push(thumbnailHeightProperty);

Some files were not shown because too many files have changed in this diff Show more