CustomJS Callback с помощью Bokeh (не знаю JS!) - PullRequest
0 голосов
/ 06 июля 2018

Я пытаюсь реализовать интерактивную (HTML) карту с помощью Bokeh на Python, чтобы я мог опубликовать ее на своем личном GitHub. Я понял, что не могу использовать обратные вызовы с Python, если я не хочу использовать сервер Bokeh. Я написал функцию обратного вызова или обновления на Python, прежде чем понял эту проблему, и теперь у меня возникают проблемы при перезаписи на JS.

Функция обновляет источник данных для активного глифа на карте после выделения кадра данных pandas из значений, выбранных на ползунках.

def make_data(year=1718, market='dayof', mode='priority_b',
          zone='None', sibling='None'):
"""
This functions subsets the ratexs_dfs by year, market, mode and sibling
parameters refer to the following:
    year =      school year, 1718 or 1819
    market =    time at which cutoffs were calculated:
        any =       anyday, this includes waitlisted applicants and such
        dayof =     these cutoffs were calculated at the day of the lottery
                    #   it should be noted that only the shp_data from SY1718
                        has shp_data from both dayof and any markets.
                        any or dayof markets for SY1819 are identical
                        since the shp_data was obtained before waitlist placements
    mode =      priority system.
        priority =      this system is the native method of the yearself.
                        SY1718 uses the 2 digit priority system,
                        SY1819 uses the 3 digit priority system.
        priority_b =    this is the secondary system. For SY1718, priority
                        and priority_b are identical. However for SY1819,
                        priority is the 3 digit priority system, while
                        priority_b is the SY1819 priorities adapted for the
                        SY1718 priority system.
    sibling =   school where applicant would get sibling priority
"""

school_ratex = ratexs_df[(ratexs_df['year'] == year)
                         & (ratexs_df['market'] == market)
                         & (ratexs_df['prio_mode'] == mode)
                         & ((ratexs_df['nhood'] == zone)
                            | (ratexs_df['nhood'] == 'None'))
                         & ((ratexs_df['sibling'] == sibling)
                            | (ratexs_df['sibling'] == 'None'))]

school_ratex = school_ratex.drop_duplicates(subset='school', keep='first')
school_ratex['ratex'] = (school_ratex['ratex'] * 100).round(2)
school_ratex['ratex_str'] = school_ratex['ratex'].astype(str) + ' %'
school_ratex = school_ratex.sort_values(by='priority')

return ColumnDataSource(school_ratex)

Тогда функция обновления:

def update(attr, old, new):

# make a new def
new_src = make_data(zone=schoolzone_dd.value,
                    sibling=sibling_dd.value)

# update data on map
psource2.data.update(new_src.data)

, где sibling_dd и schoolzone_dd - ползунки:

# define dropdown widgets
# school lists
schools = ['None']
schools.extend(sorted(src['school'].values.tolist(), reverse=False))
sibling_dd = Select(title='Sibling School',
                    value=schools[0],
                    options=schools)

# zones lists
zones = ['None']
zones.extend(sorted(shp_source['app_name'].values.tolist(), reverse=False))
zones.insert(1, schools[1])
schoolzone_dd = Select(title='School Zone',
                       value=zones[0],
                       options=zones)

Есть ли простой способ, с помощью которого я могу разделить фрейм данных аналогичным образом при вызове CustomJS, используя значения двух ползунков параллельно?

Заранее спасибо.

1 Ответ

0 голосов
/ 07 июля 2018

Очевидно, что есть более простой способ сделать это (с Python), использующий пакет PScript, который компилирует Python int Javascript.

Решение состоит в том, чтобы вызвать виджет без каких-либо специальных параметров, т. Е.,

# school lists
schools = ['None']
schools.extend(sorted(source['school'].values.tolist(), reverse=False))
sibling_dd = Select(title='Sibling School',
                    value=schools[1],
                    options=schools)

Определите ваш обратный вызов:

def callback(source=source_bok, original_source=original_source,
             sibling_source=sibling_dd, zone_source=schoolzone_dd):

    # initialize data
    data = original_source.data
    s_data = source.data
    zone = zone_source.value
    sibling = sibling_source.value


    # fetch unique schools
    schools = s_data['school']

    # fetch columns from original data
    orig_schools = data['school']
    orig_nhoods = data['nhood']
    orig_sibling = data['sibling']
    orig_year = data['year']
    orig_market = data['market']
    orig_prio_mode = data['prio_mode']
    orig_ratex = original_source.data['ratex']
    orig_ratex_str = data['ratex_str']
    orig_priority = data['priority']

    no_prio_ratex = []
    no_prio_ratex_str = []
    no_prio_priority = []
    for row in range(len(orig_schools)):
        for j in range(len(schools)):
            if ((orig_nhoods[row] == 'None')
                & (orig_sibling[row] == 'None')
                & (orig_schools[row] == schools[j])
                & (orig_year[row] == 1718)
                & (orig_market[row] == 'dayof')
                & (orig_prio_mode[row] == 'priority_b')):
                    no_prio_ratex.append(orig_ratex[row])
                    no_prio_ratex_str.append(orig_ratex_str[row])
                    no_prio_priority.append(orig_priority[row])

        #return (no_prio_ratex, no_prio_ratex_str, no_prio_priority)
    if (zone == 'None') & (sibling == 'None'):
        s_data['ratex_str'] = no_prio_ratex_str
        s_data['ratex'] = no_prio_ratex
        s_data['priority'] = no_prio_priority

    if (zone == 'None') & (sibling != 'None'):
        s_prio_ratex = []
        s_prio_ratex_str = []
        s_prio_priority = []
        for row in range(len(orig_schools)):
            for j in range(len(schools)):
                if ((orig_nhoods[row] == 'None')
                    & (orig_sibling[row] == sibling)
                    & (orig_schools[row] == schools[j])
                    & (orig_year[row] == 1718)
                    & (orig_market[row] == 'dayof')
                    & (orig_prio_mode[row] == 'priority_b')):
                        s_prio_ratex.append(orig_ratex[row])
                        s_prio_ratex_str.append(orig_ratex_str[row])
                        s_prio_priority.append(orig_priority[row])

        s_data['ratex'] = [s_prio_ratex[0] if s_data['school'][i] == sibling else s_data['ratex'][i] for i in range(len(s_data['school']))]
        s_data['ratex_str'] = [s_prio_ratex_str[0] if s_data['school'][i] == sibling else s_data['ratex_str'][i] for i in range(len(s_data['school']))]
        s_data['priority'] = [s_prio_priority[0] if s_data['school'][i] == sibling else s_data['priority'][i] for i in range(len(s_data['school']))]

    if (zone != 'None') & (sibling == 'None'):
        n_prio_ratex = []
        n_prio_ratex_str = []
        n_prio_priority = []
        for row in range(len(orig_schools)):
            for j in range(len(schools)):
                if ((orig_nhoods[row] == zone)
                    & (orig_sibling[row] == 'None')
                    & (orig_schools[row] == schools[j])
                    & (orig_year[row] == 1718)
                    & (orig_market[row] == 'dayof')
                    & (orig_prio_mode[row] == 'priority_b')):
                        n_prio_ratex.append(orig_ratex[row])
                        n_prio_ratex_str.append(orig_ratex_str[row])
                        n_prio_priority.append(orig_priority[row])

        s_data['ratex'] = [n_prio_ratex[0] if s_data['school'][i] == zone else s_data['ratex'][i] for i in range(len(s_data['school']))]
        s_data['ratex_str'] = [n_prio_ratex_str[0] if s_data['school'][i] == zone else s_data['ratex_str'][i] for i in range(len(s_data['school']))]
        s_data['priority'] = [n_prio_priority[0] if s_data['school'][i] == zone else s_data['priority'][i] for i in range(len(s_data['school']))]

    if (zone != 'None') & (sibling != 'None'):
        ns_prio_ratex = []
        ns_prio_ratex_str = []
        ns_prio_priority = []
        for row in range(len(orig_schools)):
            for j in range(len(schools)):
                if ((orig_nhoods[row] == zone)
                    & (orig_sibling[row] == sibling)
                    & (orig_schools[row] == schools[j])
                    & (orig_year[row] == 1718)
                    & (orig_market[row] == 'dayof')
                    & (orig_prio_mode[row] == 'priority_b')):
                        ns_prio_ratex.append(orig_ratex[row])
                        ns_prio_ratex_str.append(orig_ratex_str[row])
                        ns_prio_priority.append(orig_priority[row])

        s_data['ratex'] = [ns_prio_ratex[0] if (s_data['school'][i] == zone) & (s_data['school'][i] == sibling) else s_data['ratex'][i] for i in range(len(s_data['school']))]
        s_data['ratex_str'] = [ns_prio_ratex_str[0] if (s_data['school'][i] == zone) & (s_data['school'][i] == sibling) else s_data['ratex_str'][i] for i in range(len(s_data['school']))]
        s_data['priority'] = [ns_prio_priority[0] if (s_data['school'][i] == zone) & (s_data['school'][i] == sibling) else s_data['priority'][i] for i in range(len(s_data['school']))]

    if zone == 'None':
        s_data['nhood'] = ['None' for school in s_data['school']]
    else:
        s_data['nhood'] = [zone if school == zone else 'None' for school in s_data['school']]

    if sibling == 'None':
        s_data['sibling'] = ['None' for school in s_data['school']]
    else:
        s_data['sibling'] = [sibling if school == sibling else 'None' for school in s_data['school']]

    source.change.emit()def callback(source=source_bok, original_source=original_source,
             sibling_source=sibling_dd, zone_source=schoolzone_dd):

    # initialize data
    data = original_source.data
    s_data = source.data
    zone = zone_source.value
    sibling = sibling_source.value


    # fetch unique schools
    schools = s_data['school']

    # fetch columns from original data
    orig_schools = data['school']
    orig_nhoods = data['nhood']
    orig_sibling = data['sibling']
    orig_year = data['year']
    orig_market = data['market']
    orig_prio_mode = data['prio_mode']
    orig_ratex = original_source.data['ratex']
    orig_ratex_str = data['ratex_str']
    orig_priority = data['priority']

    no_prio_ratex = []
    no_prio_ratex_str = []
    no_prio_priority = []
    for row in range(len(orig_schools)):
        for j in range(len(schools)):
            if ((orig_nhoods[row] == 'None')
                & (orig_sibling[row] == 'None')
                & (orig_schools[row] == schools[j])
                & (orig_year[row] == 1718)
                & (orig_market[row] == 'dayof')
                & (orig_prio_mode[row] == 'priority_b')):
                    no_prio_ratex.append(orig_ratex[row])
                    no_prio_ratex_str.append(orig_ratex_str[row])
                    no_prio_priority.append(orig_priority[row])

        #return (no_prio_ratex, no_prio_ratex_str, no_prio_priority)
    if (zone == 'None') & (sibling == 'None'):
        s_data['ratex_str'] = no_prio_ratex_str
        s_data['ratex'] = no_prio_ratex
        s_data['priority'] = no_prio_priority

    if (zone == 'None') & (sibling != 'None'):
        s_prio_ratex = []
        s_prio_ratex_str = []
        s_prio_priority = []
        for row in range(len(orig_schools)):
            for j in range(len(schools)):
                if ((orig_nhoods[row] == 'None')
                    & (orig_sibling[row] == sibling)
                    & (orig_schools[row] == schools[j])
                    & (orig_year[row] == 1718)
                    & (orig_market[row] == 'dayof')
                    & (orig_prio_mode[row] == 'priority_b')):
                        s_prio_ratex.append(orig_ratex[row])
                        s_prio_ratex_str.append(orig_ratex_str[row])
                        s_prio_priority.append(orig_priority[row])

        s_data['ratex'] = [s_prio_ratex[0] if s_data['school'][i] == sibling else s_data['ratex'][i] for i in range(len(s_data['school']))]
        s_data['ratex_str'] = [s_prio_ratex_str[0] if s_data['school'][i] == sibling else s_data['ratex_str'][i] for i in range(len(s_data['school']))]
        s_data['priority'] = [s_prio_priority[0] if s_data['school'][i] == sibling else s_data['priority'][i] for i in range(len(s_data['school']))]

    if (zone != 'None') & (sibling == 'None'):
        n_prio_ratex = []
        n_prio_ratex_str = []
        n_prio_priority = []
        for row in range(len(orig_schools)):
            for j in range(len(schools)):
                if ((orig_nhoods[row] == zone)
                    & (orig_sibling[row] == 'None')
                    & (orig_schools[row] == schools[j])
                    & (orig_year[row] == 1718)
                    & (orig_market[row] == 'dayof')
                    & (orig_prio_mode[row] == 'priority_b')):
                        n_prio_ratex.append(orig_ratex[row])
                        n_prio_ratex_str.append(orig_ratex_str[row])
                        n_prio_priority.append(orig_priority[row])

        s_data['ratex'] = [n_prio_ratex[0] if s_data['school'][i] == zone else s_data['ratex'][i] for i in range(len(s_data['school']))]
        s_data['ratex_str'] = [n_prio_ratex_str[0] if s_data['school'][i] == zone else s_data['ratex_str'][i] for i in range(len(s_data['school']))]
        s_data['priority'] = [n_prio_priority[0] if s_data['school'][i] == zone else s_data['priority'][i] for i in range(len(s_data['school']))]

    if (zone != 'None') & (sibling != 'None'):
        ns_prio_ratex = []
        ns_prio_ratex_str = []
        ns_prio_priority = []
        for row in range(len(orig_schools)):
            for j in range(len(schools)):
                if ((orig_nhoods[row] == zone)
                    & (orig_sibling[row] == sibling)
                    & (orig_schools[row] == schools[j])
                    & (orig_year[row] == 1718)
                    & (orig_market[row] == 'dayof')
                    & (orig_prio_mode[row] == 'priority_b')):
                        ns_prio_ratex.append(orig_ratex[row])
                        ns_prio_ratex_str.append(orig_ratex_str[row])
                        ns_prio_priority.append(orig_priority[row])

        s_data['ratex'] = [ns_prio_ratex[0] if (s_data['school'][i] == zone) & (s_data['school'][i] == sibling) else s_data['ratex'][i] for i in range(len(s_data['school']))]
        s_data['ratex_str'] = [ns_prio_ratex_str[0] if (s_data['school'][i] == zone) & (s_data['school'][i] == sibling) else s_data['ratex_str'][i] for i in range(len(s_data['school']))]
        s_data['priority'] = [ns_prio_priority[0] if (s_data['school'][i] == zone) & (s_data['school'][i] == sibling) else s_data['priority'][i] for i in range(len(s_data['school']))]

    if zone == 'None':
        s_data['nhood'] = ['None' for school in s_data['school']]
    else:
        s_data['nhood'] = [zone if school == zone else 'None' for school in s_data['school']]

    if sibling == 'None':
        s_data['sibling'] = ['None' for school in s_data['school']]
    else:
        s_data['sibling'] = [sibling if school == sibling else 'None' for school in s_data['school']]

    source.change.emit()

source.change.emit() ключевая роль (мне было плохо, если я не копался в документации)

Затем, наконец, перед сохранением или рендерингом, позвоните:

sibling_dd.callback = CustomJS.from_py_func(callback)

Убедитесь, что весь питон внутри обратного вызова совместим с PScript !

PS - Чрезвычайно полезным был @ Matt ответ и последующее решение.

...